123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 |
- <?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\Downloader;
- use Composer\Package\PackageInterface;
- use Composer\Util\Svn as SvnUtil;
- use Composer\Repository\VcsRepository;
- use Composer\Util\ProcessExecutor;
- /**
- * @author Ben Bieker <mail@ben-bieker.de>
- * @author Till Klampaeckel <till@php.net>
- */
- class SvnDownloader extends VcsDownloader
- {
- protected $cacheCredentials = true;
- /**
- * {@inheritDoc}
- */
- protected function doDownload(PackageInterface $package, $path, $url, PackageInterface $prevPackage = null)
- {
- }
- /**
- * {@inheritDoc}
- */
- protected function doInstall(PackageInterface $package, $path, $url)
- {
- SvnUtil::cleanEnv();
- $ref = $package->getSourceReference();
- $repo = $package->getRepository();
- if ($repo instanceof VcsRepository) {
- $repoConfig = $repo->getRepoConfig();
- if (array_key_exists('svn-cache-credentials', $repoConfig)) {
- $this->cacheCredentials = (bool) $repoConfig['svn-cache-credentials'];
- }
- }
- $this->io->writeError(" Checking out ".$package->getSourceReference());
- $this->execute($package, $url, "svn co", sprintf("%s/%s", $url, $ref), null, $path);
- }
- /**
- * {@inheritDoc}
- */
- protected function doUpdate(PackageInterface $initial, PackageInterface $target, $path, $url)
- {
- SvnUtil::cleanEnv();
- $ref = $target->getSourceReference();
- if (!$this->hasMetadataRepository($path)) {
- throw new \RuntimeException('The .svn directory is missing from '.$path.', see https://getcomposer.org/commit-deps for more information');
- }
- $util = new SvnUtil($url, $this->io, $this->config);
- $flags = "";
- if (version_compare($util->binaryVersion(), '1.7.0', '>=')) {
- $flags .= ' --ignore-ancestry';
- }
- $this->io->writeError(" Checking out " . $ref);
- $this->execute($target, $url, "svn switch" . $flags, sprintf("%s/%s", $url, $ref), $path);
- }
- /**
- * {@inheritDoc}
- */
- public function getLocalChanges(PackageInterface $package, $path)
- {
- if (!$this->hasMetadataRepository($path)) {
- return null;
- }
- $this->process->execute('svn status --ignore-externals', $output, $path);
- return preg_match('{^ *[^X ] +}m', $output) ? $output : null;
- }
- /**
- * Execute an SVN command and try to fix up the process with credentials
- * if necessary.
- *
- * @param string $baseUrl Base URL of the repository
- * @param string $command SVN command to run
- * @param string $url SVN url
- * @param string $cwd Working directory
- * @param string $path Target for a checkout
- * @throws \RuntimeException
- * @return string
- */
- protected function execute(PackageInterface $package, $baseUrl, $command, $url, $cwd = null, $path = null)
- {
- $util = new SvnUtil($baseUrl, $this->io, $this->config);
- $util->setCacheCredentials($this->cacheCredentials);
- try {
- return $util->execute($command, $url, $cwd, $path, $this->io->isVerbose());
- } catch (\RuntimeException $e) {
- throw new \RuntimeException(
- $package->getPrettyName().' could not be downloaded, '.$e->getMessage()
- );
- }
- }
- /**
- * {@inheritDoc}
- */
- protected function cleanChanges(PackageInterface $package, $path, $update)
- {
- if (!$changes = $this->getLocalChanges($package, $path)) {
- return;
- }
- if (!$this->io->isInteractive()) {
- if (true === $this->config->get('discard-changes')) {
- return $this->discardChanges($path);
- }
- return parent::cleanChanges($package, $path, $update);
- }
- $changes = array_map(function ($elem) {
- return ' '.$elem;
- }, preg_split('{\s*\r?\n\s*}', $changes));
- $countChanges = count($changes);
- $this->io->writeError(sprintf(' <error>'.$package->getPrettyName().' has modified file%s:</error>', $countChanges === 1 ? '' : 's'));
- $this->io->writeError(array_slice($changes, 0, 10));
- if ($countChanges > 10) {
- $remainingChanges = $countChanges - 10;
- $this->io->writeError(
- sprintf(
- ' <info>'.$remainingChanges.' more file%s modified, choose "v" to view the full list</info>',
- $remainingChanges === 1 ? '' : 's'
- )
- );
- }
- while (true) {
- switch ($this->io->ask(' <info>Discard changes [y,n,v,?]?</info> ', '?')) {
- case 'y':
- $this->discardChanges($path);
- break 2;
- case 'n':
- throw new \RuntimeException('Update aborted');
- case 'v':
- $this->io->writeError($changes);
- break;
- case '?':
- default:
- $this->io->writeError(array(
- ' y - discard changes and apply the '.($update ? 'update' : 'uninstall'),
- ' n - abort the '.($update ? 'update' : 'uninstall').' and let you manually clean things up',
- ' v - view modified files',
- ' ? - print help',
- ));
- break;
- }
- }
- }
- /**
- * {@inheritDoc}
- */
- protected function getCommitLogs($fromReference, $toReference, $path)
- {
- if (preg_match('{@(\d+)$}', $fromReference) && preg_match('{@(\d+)$}', $toReference)) {
- // retrieve the svn base url from the checkout folder
- $command = sprintf('svn info --non-interactive --xml %s', ProcessExecutor::escape($path));
- if (0 !== $this->process->execute($command, $output, $path)) {
- throw new \RuntimeException(
- 'Failed to execute ' . $command . "\n\n" . $this->process->getErrorOutput()
- );
- }
- $urlPattern = '#<url>(.*)</url>#';
- if (preg_match($urlPattern, $output, $matches)) {
- $baseUrl = $matches[1];
- } else {
- throw new \RuntimeException(
- 'Unable to determine svn url for path '. $path
- );
- }
- // strip paths from references and only keep the actual revision
- $fromRevision = preg_replace('{.*@(\d+)$}', '$1', $fromReference);
- $toRevision = preg_replace('{.*@(\d+)$}', '$1', $toReference);
- $command = sprintf('svn log -r%s:%s --incremental', ProcessExecutor::escape($fromRevision), ProcessExecutor::escape($toRevision));
- $util = new SvnUtil($baseUrl, $this->io, $this->config);
- $util->setCacheCredentials($this->cacheCredentials);
- try {
- return $util->executeLocal($command, $path, null, $this->io->isVerbose());
- } catch (\RuntimeException $e) {
- throw new \RuntimeException(
- 'Failed to execute ' . $command . "\n\n".$e->getMessage()
- );
- }
- }
- return "Could not retrieve changes between $fromReference and $toReference due to missing revision information";
- }
- protected function discardChanges($path)
- {
- if (0 !== $this->process->execute('svn revert -R .', $output, $path)) {
- throw new \RuntimeException("Could not reset changes\n\n:".$this->process->getErrorOutput());
- }
- }
- /**
- * {@inheritDoc}
- */
- protected function hasMetadataRepository($path)
- {
- return is_dir($path.'/.svn');
- }
- }
|