123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438 |
- <?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\Package;
- use Composer\Json\JsonFile;
- use Composer\Installer\InstallationManager;
- use Composer\Repository\LockArrayRepository;
- use Composer\Repository\RepositoryManager;
- use Composer\Util\ProcessExecutor;
- use Composer\Package\Dumper\ArrayDumper;
- use Composer\Package\Loader\ArrayLoader;
- use Composer\Util\Git as GitUtil;
- use Composer\IO\IOInterface;
- use Seld\JsonLint\ParsingException;
- /**
- * Reads/writes project lockfile (composer.lock).
- *
- * @author Konstantin Kudryashiv <ever.zet@gmail.com>
- * @author Jordi Boggiano <j.boggiano@seld.be>
- */
- class Locker
- {
- private $lockFile;
- private $repositoryManager;
- private $installationManager;
- private $hash;
- private $contentHash;
- private $loader;
- private $dumper;
- private $process;
- private $lockDataCache;
- /**
- * Initializes packages locker.
- *
- * @param IOInterface $io
- * @param JsonFile $lockFile lockfile loader
- * @param RepositoryManager $repositoryManager repository manager instance
- * @param InstallationManager $installationManager installation manager instance
- * @param string $composerFileContents The contents of the composer file
- */
- public function __construct(IOInterface $io, JsonFile $lockFile, RepositoryManager $repositoryManager, InstallationManager $installationManager, $composerFileContents)
- {
- $this->lockFile = $lockFile;
- $this->repositoryManager = $repositoryManager;
- $this->installationManager = $installationManager;
- $this->hash = md5($composerFileContents);
- $this->contentHash = self::getContentHash($composerFileContents);
- $this->loader = new ArrayLoader(null, true);
- $this->dumper = new ArrayDumper();
- $this->process = new ProcessExecutor($io);
- }
- /**
- * Returns the md5 hash of the sorted content of the composer file.
- *
- * @param string $composerFileContents The contents of the composer file.
- *
- * @return string
- */
- public static function getContentHash($composerFileContents)
- {
- $content = json_decode($composerFileContents, true);
- $relevantKeys = array(
- 'name',
- 'version',
- 'require',
- 'require-dev',
- 'conflict',
- 'replace',
- 'provide',
- 'minimum-stability',
- 'prefer-stable',
- 'repositories',
- 'extra',
- );
- $relevantContent = array();
- foreach (array_intersect($relevantKeys, array_keys($content)) as $key) {
- $relevantContent[$key] = $content[$key];
- }
- if (isset($content['config']['platform'])) {
- $relevantContent['config']['platform'] = $content['config']['platform'];
- }
- ksort($relevantContent);
- return md5(json_encode($relevantContent));
- }
- /**
- * Checks whether locker has been locked (lockfile found).
- *
- * @return bool
- */
- public function isLocked()
- {
- if (!$this->lockFile->exists()) {
- return false;
- }
- $data = $this->getLockData();
- return isset($data['packages']);
- }
- /**
- * Checks whether the lock file is still up to date with the current hash
- *
- * @return bool
- */
- public function isFresh()
- {
- $lock = $this->lockFile->read();
- if (!empty($lock['content-hash'])) {
- // There is a content hash key, use that instead of the file hash
- return $this->contentHash === $lock['content-hash'];
- }
- // BC support for old lock files without content-hash
- if (!empty($lock['hash'])) {
- return $this->hash === $lock['hash'];
- }
- // should not be reached unless the lock file is corrupted, so assume it's out of date
- return false;
- }
- /**
- * Searches and returns an array of locked packages, retrieved from registered repositories.
- *
- * @param bool $withDevReqs true to retrieve the locked dev packages
- * @throws \RuntimeException
- * @return \Composer\Repository\RepositoryInterface
- */
- public function getLockedRepository($withDevReqs = false)
- {
- $lockData = $this->getLockData();
- $packages = new LockArrayRepository();
- $lockedPackages = $lockData['packages'];
- if ($withDevReqs) {
- if (isset($lockData['packages-dev'])) {
- $lockedPackages = array_merge($lockedPackages, $lockData['packages-dev']);
- } else {
- throw new \RuntimeException('The lock file does not contain require-dev information, run install with the --no-dev option or run update to install those packages.');
- }
- }
- if (empty($lockedPackages)) {
- return $packages;
- }
- if (isset($lockedPackages[0]['name'])) {
- foreach ($lockedPackages as $info) {
- $packages->addPackage($this->loader->load($info));
- }
- return $packages;
- }
- throw new \RuntimeException('Your composer.lock was created before 2012-09-15, and is not supported anymore. Run "composer update" to generate a new one.');
- }
- /**
- * Returns the platform requirements stored in the lock file
- *
- * @param bool $withDevReqs if true, the platform requirements from the require-dev block are also returned
- * @return \Composer\Package\Link[]
- */
- public function getPlatformRequirements($withDevReqs = false)
- {
- $lockData = $this->getLockData();
- $requirements = array();
- if (!empty($lockData['platform'])) {
- $requirements = $this->loader->parseLinks(
- '__ROOT__',
- '1.0.0',
- 'requires',
- isset($lockData['platform']) ? $lockData['platform'] : array()
- );
- }
- if ($withDevReqs && !empty($lockData['platform-dev'])) {
- $devRequirements = $this->loader->parseLinks(
- '__ROOT__',
- '1.0.0',
- 'requires',
- isset($lockData['platform-dev']) ? $lockData['platform-dev'] : array()
- );
- $requirements = array_merge($requirements, $devRequirements);
- }
- return $requirements;
- }
- public function getMinimumStability()
- {
- $lockData = $this->getLockData();
- return isset($lockData['minimum-stability']) ? $lockData['minimum-stability'] : 'stable';
- }
- public function getStabilityFlags()
- {
- $lockData = $this->getLockData();
- return isset($lockData['stability-flags']) ? $lockData['stability-flags'] : array();
- }
- public function getPreferStable()
- {
- $lockData = $this->getLockData();
- // return null if not set to allow caller logic to choose the
- // right behavior since old lock files have no prefer-stable
- return isset($lockData['prefer-stable']) ? $lockData['prefer-stable'] : null;
- }
- public function getPreferLowest()
- {
- $lockData = $this->getLockData();
- // return null if not set to allow caller logic to choose the
- // right behavior since old lock files have no prefer-lowest
- return isset($lockData['prefer-lowest']) ? $lockData['prefer-lowest'] : null;
- }
- public function getPlatformOverrides()
- {
- $lockData = $this->getLockData();
- return isset($lockData['platform-overrides']) ? $lockData['platform-overrides'] : array();
- }
- public function getAliases()
- {
- $lockData = $this->getLockData();
- return isset($lockData['aliases']) ? $lockData['aliases'] : array();
- }
- public function getLockData()
- {
- if (null !== $this->lockDataCache) {
- return $this->lockDataCache;
- }
- if (!$this->lockFile->exists()) {
- throw new \LogicException('No lockfile found. Unable to read locked packages');
- }
- return $this->lockDataCache = $this->lockFile->read();
- }
- /**
- * Locks provided data into lockfile.
- *
- * @param array $packages array of packages
- * @param mixed $devPackages array of dev packages or null if installed without --dev
- * @param array $platformReqs array of package name => constraint for required platform packages
- * @param mixed $platformDevReqs array of package name => constraint for dev-required platform packages
- * @param array $aliases array of aliases
- * @param string $minimumStability
- * @param array $stabilityFlags
- * @param bool $preferStable
- * @param bool $preferLowest
- * @param array $platformOverrides
- *
- * @return bool
- */
- public function setLockData(array $packages, $devPackages, array $platformReqs, $platformDevReqs, array $aliases, $minimumStability, array $stabilityFlags, $preferStable, $preferLowest, array $platformOverrides)
- {
- $lock = array(
- '_readme' => array('This file locks the dependencies of your project to a known state',
- 'Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies',
- 'This file is @gener'.'ated automatically', ),
- 'content-hash' => $this->contentHash,
- 'packages' => null,
- 'packages-dev' => null,
- 'aliases' => array(),
- 'minimum-stability' => $minimumStability,
- 'stability-flags' => $stabilityFlags,
- 'prefer-stable' => $preferStable,
- 'prefer-lowest' => $preferLowest,
- );
- foreach ($aliases as $package => $versions) {
- foreach ($versions as $version => $alias) {
- $lock['aliases'][] = array(
- 'alias' => $alias['alias'],
- 'alias_normalized' => $alias['alias_normalized'],
- 'version' => $version,
- 'package' => $package,
- );
- }
- }
- $lock['packages'] = $this->lockPackages($packages);
- if (null !== $devPackages) {
- $lock['packages-dev'] = $this->lockPackages($devPackages);
- }
- $lock['platform'] = $platformReqs;
- $lock['platform-dev'] = $platformDevReqs;
- if ($platformOverrides) {
- $lock['platform-overrides'] = $platformOverrides;
- }
- if (empty($lock['packages']) && empty($lock['packages-dev']) && empty($lock['platform']) && empty($lock['platform-dev'])) {
- if ($this->lockFile->exists()) {
- unlink($this->lockFile->getPath());
- }
- return false;
- }
- try {
- $isLocked = $this->isLocked();
- } catch (ParsingException $e) {
- $isLocked = false;
- }
- if (!$isLocked || $lock !== $this->getLockData()) {
- $this->lockFile->write($lock);
- $this->lockDataCache = null;
- return true;
- }
- return false;
- }
- private function lockPackages(array $packages)
- {
- $locked = array();
- foreach ($packages as $package) {
- if ($package instanceof AliasPackage) {
- continue;
- }
- $name = $package->getPrettyName();
- $version = $package->getPrettyVersion();
- if (!$name || !$version) {
- throw new \LogicException(sprintf(
- 'Package "%s" has no version or name and can not be locked',
- $package
- ));
- }
- $spec = $this->dumper->dump($package);
- unset($spec['version_normalized']);
- // always move time to the end of the package definition
- $time = isset($spec['time']) ? $spec['time'] : null;
- unset($spec['time']);
- if ($package->isDev() && $package->getInstallationSource() === 'source') {
- // use the exact commit time of the current reference if it's a dev package
- $time = $this->getPackageTime($package) ?: $time;
- }
- if (null !== $time) {
- $spec['time'] = $time;
- }
- unset($spec['installation-source']);
- $locked[] = $spec;
- }
- usort($locked, function ($a, $b) {
- $comparison = strcmp($a['name'], $b['name']);
- if (0 !== $comparison) {
- return $comparison;
- }
- // If it is the same package, compare the versions to make the order deterministic
- return strcmp($a['version'], $b['version']);
- });
- return $locked;
- }
- /**
- * Returns the packages's datetime for its source reference.
- *
- * @param PackageInterface $package The package to scan.
- * @return string|null The formatted datetime or null if none was found.
- */
- private function getPackageTime(PackageInterface $package)
- {
- if (!function_exists('proc_open')) {
- return null;
- }
- $path = realpath($this->installationManager->getInstallPath($package));
- $sourceType = $package->getSourceType();
- $datetime = null;
- if ($path && in_array($sourceType, array('git', 'hg'))) {
- $sourceRef = $package->getSourceReference() ?: $package->getDistReference();
- switch ($sourceType) {
- case 'git':
- GitUtil::cleanEnv();
- if (0 === $this->process->execute('git log -n1 --pretty=%ct '.ProcessExecutor::escape($sourceRef), $output, $path) && preg_match('{^\s*\d+\s*$}', $output)) {
- $datetime = new \DateTime('@'.trim($output), new \DateTimeZone('UTC'));
- }
- break;
- case 'hg':
- if (0 === $this->process->execute('hg log --template "{date|hgdate}" -r '.ProcessExecutor::escape($sourceRef), $output, $path) && preg_match('{^\s*(\d+)\s*}', $output, $match)) {
- $datetime = new \DateTime('@'.$match[1], new \DateTimeZone('UTC'));
- }
- break;
- }
- }
- return $datetime ? $datetime->format(DATE_RFC3339) : null;
- }
- }
|