2011-05-06 17:55:49 +00:00
|
|
|
<?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;
|
|
|
|
|
2012-10-21 15:56:57 +00:00
|
|
|
use Composer\Config;
|
2012-11-10 22:17:36 +00:00
|
|
|
use Composer\Cache;
|
2013-08-14 15:42:11 +00:00
|
|
|
use Composer\EventDispatcher\EventDispatcher;
|
2016-02-25 13:05:26 +00:00
|
|
|
use Composer\Package\PackageInterface;
|
2016-11-19 20:50:15 +00:00
|
|
|
use Composer\Util\IniHelper;
|
2016-02-03 21:39:16 +00:00
|
|
|
use Composer\Util\Platform;
|
2012-01-18 15:37:55 +00:00
|
|
|
use Composer\Util\ProcessExecutor;
|
2016-01-10 20:06:10 +00:00
|
|
|
use Composer\Util\RemoteFilesystem;
|
2012-01-18 16:11:26 +00:00
|
|
|
use Composer\IO\IOInterface;
|
2016-02-25 13:05:26 +00:00
|
|
|
use Symfony\Component\Process\ExecutableFinder;
|
2012-03-29 13:08:47 +00:00
|
|
|
use ZipArchive;
|
2011-05-06 17:55:49 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @author Jordi Boggiano <j.boggiano@seld.be>
|
|
|
|
*/
|
2012-02-17 22:10:02 +00:00
|
|
|
class ZipDownloader extends ArchiveDownloader
|
2011-05-06 17:55:49 +00:00
|
|
|
{
|
2012-01-18 15:37:55 +00:00
|
|
|
protected $process;
|
2016-02-25 14:04:29 +00:00
|
|
|
protected static $hasSystemUnzip;
|
2017-02-13 14:54:55 +00:00
|
|
|
protected static $hasZipArchive;
|
2012-01-18 15:37:55 +00:00
|
|
|
|
2016-01-10 20:06:10 +00:00
|
|
|
public function __construct(IOInterface $io, Config $config, EventDispatcher $eventDispatcher = null, Cache $cache = null, ProcessExecutor $process = null, RemoteFilesystem $rfs = null)
|
2012-01-18 15:37:55 +00:00
|
|
|
{
|
2013-08-10 00:43:40 +00:00
|
|
|
$this->process = $process ?: new ProcessExecutor($io);
|
2016-01-10 20:06:10 +00:00
|
|
|
parent::__construct($io, $config, $eventDispatcher, $cache, $rfs);
|
2012-01-18 15:37:55 +00:00
|
|
|
}
|
|
|
|
|
2016-02-25 13:05:26 +00:00
|
|
|
/**
|
|
|
|
* {@inheritDoc}
|
|
|
|
*/
|
2016-12-06 22:41:46 +00:00
|
|
|
public function download(PackageInterface $package, $path, $output = true)
|
2016-02-25 13:05:26 +00:00
|
|
|
{
|
2016-02-25 14:04:29 +00:00
|
|
|
if (null === self::$hasSystemUnzip) {
|
2016-02-25 13:05:26 +00:00
|
|
|
$finder = new ExecutableFinder;
|
2016-02-25 14:04:29 +00:00
|
|
|
self::$hasSystemUnzip = (bool) $finder->find('unzip');
|
2016-02-25 13:05:26 +00:00
|
|
|
}
|
|
|
|
|
2017-02-13 14:54:55 +00:00
|
|
|
if (null === self::$hasZipArchive) {
|
|
|
|
self::$hasZipArchive = class_exists('ZipArchive');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!self::$hasZipArchive && !self::$hasSystemUnzip) {
|
2016-03-11 09:37:00 +00:00
|
|
|
// php.ini path is added to the error message to help users find the correct file
|
2016-11-19 20:50:15 +00:00
|
|
|
$iniMessage = IniHelper::getMessage();
|
2016-03-11 09:37:00 +00:00
|
|
|
$error = "The zip extension and unzip command are both missing, skipping.\n" . $iniMessage;
|
|
|
|
|
|
|
|
throw new \RuntimeException($error);
|
2016-02-25 13:05:26 +00:00
|
|
|
}
|
|
|
|
|
2016-12-06 22:41:46 +00:00
|
|
|
return parent::download($package, $path, $output);
|
2016-02-25 13:05:26 +00:00
|
|
|
}
|
|
|
|
|
2017-02-13 13:00:48 +00:00
|
|
|
/**
|
|
|
|
* extract $file to $path with "unzip" command
|
2017-02-13 12:43:36 +00:00
|
|
|
*
|
2017-02-13 14:54:55 +00:00
|
|
|
* @param string $file File to extract
|
|
|
|
* @param string $path Path where to extract file
|
|
|
|
* @param bool $isFallback If true it is called as a fallback and should not throw exception
|
2017-02-14 16:44:56 +00:00
|
|
|
* @return bool|\Exception True if succeed, an Exception if not
|
2017-02-13 12:43:36 +00:00
|
|
|
*/
|
2017-02-13 14:54:55 +00:00
|
|
|
protected function extractWithSystemUnzip($file, $path, $isFallback)
|
2011-05-06 17:55:49 +00:00
|
|
|
{
|
2013-01-31 09:57:59 +00:00
|
|
|
$processError = null;
|
2017-02-13 14:54:55 +00:00
|
|
|
// When called after a ZipArchive failed, perhaps there is some files to overwrite
|
|
|
|
$overwrite = $isFallback ? '-o' : '';
|
|
|
|
|
|
|
|
$command = 'unzip -qq '.$overwrite.' '.ProcessExecutor::escape($file).' -d '.ProcessExecutor::escape($path);
|
2013-01-31 09:57:59 +00:00
|
|
|
|
2017-02-13 13:00:48 +00:00
|
|
|
try {
|
|
|
|
if (0 === $this->process->execute($command, $ignoredOutput)) {
|
|
|
|
return TRUE;
|
2014-05-20 08:15:44 +00:00
|
|
|
}
|
2013-01-31 09:57:59 +00:00
|
|
|
|
2017-02-13 13:00:48 +00:00
|
|
|
$processError = 'Failed to execute ' . $command . "\n\n" . $this->process->getErrorOutput();
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
$processError = 'Failed to execute ' . $command . "\n\n" . $e->getMessage();
|
2011-05-06 17:55:49 +00:00
|
|
|
}
|
|
|
|
|
2017-02-13 14:54:55 +00:00
|
|
|
if ( $isFallback ) {
|
|
|
|
$this->io->write($processError);
|
|
|
|
}
|
|
|
|
return new \RuntimeException($processError);
|
2017-02-13 13:00:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* extract $file to $path with ZipArchive
|
|
|
|
*
|
|
|
|
* @param string $file File to extract
|
|
|
|
* @param string $path Path where to extract file
|
2017-02-14 16:44:56 +00:00
|
|
|
* @return bool|\Exception True if succeed, an Exception if not
|
2017-02-13 13:00:48 +00:00
|
|
|
*/
|
|
|
|
protected function extractWithZipArchive($file, $path)
|
|
|
|
{
|
2012-03-29 13:08:47 +00:00
|
|
|
$zipArchive = new ZipArchive();
|
2011-05-06 17:55:49 +00:00
|
|
|
|
2011-09-28 22:48:17 +00:00
|
|
|
if (true !== ($retval = $zipArchive->open($file))) {
|
2017-02-13 14:54:55 +00:00
|
|
|
return new \UnexpectedValueException(rtrim($this->getErrorMessage($retval, $file)."\n"), $retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
$extractResult = FALSE;
|
|
|
|
try {
|
|
|
|
$extractResult = $zipArchive->extractTo($path);
|
|
|
|
} catch (\Exception $e ) {
|
|
|
|
return $e;
|
2011-05-06 17:55:49 +00:00
|
|
|
}
|
2011-09-28 22:48:17 +00:00
|
|
|
|
2017-02-13 14:54:55 +00:00
|
|
|
if (true !== $extractResult) {
|
|
|
|
return new \RuntimeException(rtrim("There was an error extracting the ZIP file, it is either corrupted or using an invalid format.\n"));
|
2012-08-17 20:06:58 +00:00
|
|
|
}
|
|
|
|
|
2011-09-28 22:48:17 +00:00
|
|
|
$zipArchive->close();
|
2017-02-13 13:00:48 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* extract $file to $path
|
|
|
|
*
|
|
|
|
* @param string $file File to extract
|
|
|
|
* @param string $path Path where to extract file
|
|
|
|
*/
|
2017-02-14 17:02:22 +00:00
|
|
|
protected function extract($file, $path)
|
2017-02-13 13:00:48 +00:00
|
|
|
{
|
2017-02-13 14:54:55 +00:00
|
|
|
$resultZipArchive = NULL;
|
|
|
|
$resultUnzip = NULL;
|
|
|
|
|
|
|
|
if ( self::$hasZipArchive ) {
|
|
|
|
// zip module is present
|
|
|
|
$resultZipArchive = $this->extractWithZipArchive($file, $path);
|
|
|
|
if ($resultZipArchive === TRUE) {
|
2017-02-13 13:00:48 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-13 14:54:55 +00:00
|
|
|
if ( self::$hasSystemUnzip ) {
|
|
|
|
// we have unzip in the path
|
|
|
|
$isFallback=FALSE;
|
|
|
|
if ( $resultZipArchive !== NULL) {
|
|
|
|
$this->io->writeError("\nUnzip using ZipArchive failed, trying with unzip");
|
|
|
|
$isFallback=TRUE;
|
|
|
|
};
|
|
|
|
$resultUnzip = $this->extractWithSystemUnzip($file, $path, $isFallback);
|
|
|
|
if ( $resultUnzip === TRUE ) {
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// extract functions return TRUE or an exception
|
|
|
|
if ( $resultZipArchive !== NULL ) {
|
|
|
|
// zipArchive failed
|
|
|
|
// unZip not present or failed too
|
|
|
|
throw $resultZipArchive;
|
|
|
|
} else {
|
|
|
|
// unZip failed
|
|
|
|
// zipArchive not available
|
|
|
|
throw $resultUnzip;
|
|
|
|
};
|
2011-05-06 17:55:49 +00:00
|
|
|
}
|
2012-03-29 12:19:41 +00:00
|
|
|
|
2012-03-29 12:22:26 +00:00
|
|
|
/**
|
2012-03-29 13:08:47 +00:00
|
|
|
* Give a meaningful error message to the user.
|
2012-03-29 12:22:26 +00:00
|
|
|
*
|
2012-05-22 10:07:08 +00:00
|
|
|
* @param int $retval
|
|
|
|
* @param string $file
|
2012-03-29 13:08:47 +00:00
|
|
|
* @return string
|
2012-03-29 12:22:26 +00:00
|
|
|
*/
|
2012-03-29 13:08:47 +00:00
|
|
|
protected function getErrorMessage($retval, $file)
|
2012-03-29 12:19:41 +00:00
|
|
|
{
|
|
|
|
switch ($retval) {
|
2012-03-29 13:08:47 +00:00
|
|
|
case ZipArchive::ER_EXISTS:
|
|
|
|
return sprintf("File '%s' already exists.", $file);
|
|
|
|
case ZipArchive::ER_INCONS:
|
|
|
|
return sprintf("Zip archive '%s' is inconsistent.", $file);
|
|
|
|
case ZipArchive::ER_INVAL:
|
|
|
|
return sprintf("Invalid argument (%s)", $file);
|
|
|
|
case ZipArchive::ER_MEMORY:
|
|
|
|
return sprintf("Malloc failure (%s)", $file);
|
|
|
|
case ZipArchive::ER_NOENT:
|
|
|
|
return sprintf("No such zip file: '%s'", $file);
|
|
|
|
case ZipArchive::ER_NOZIP:
|
|
|
|
return sprintf("'%s' is not a zip archive.", $file);
|
|
|
|
case ZipArchive::ER_OPEN:
|
|
|
|
return sprintf("Can't open zip file: %s", $file);
|
|
|
|
case ZipArchive::ER_READ:
|
|
|
|
return sprintf("Zip read error (%s)", $file);
|
|
|
|
case ZipArchive::ER_SEEK:
|
|
|
|
return sprintf("Zip seek error (%s)", $file);
|
|
|
|
default:
|
|
|
|
return sprintf("'%s' is not a valid zip archive, got error code: %s", $file, $retval);
|
2012-03-29 12:19:41 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-17 13:12:45 +00:00
|
|
|
}
|