switch first / fallback order

main
Guillaume ZITTA 7 years ago
parent 2e8d715c2f
commit f89e01d622

@ -30,8 +30,10 @@ use ZipArchive;
class ZipDownloader extends ArchiveDownloader class ZipDownloader extends ArchiveDownloader
{ {
protected $process; protected $process;
protected static $hasSystemUnzip; public static $hasSystemUnzip;
protected static $hasZipArchive; public static $hasZipArchive;
public static $isWindows;
private $zipArchiveObject;
public function __construct(IOInterface $io, Config $config, EventDispatcher $eventDispatcher = null, Cache $cache = null, ProcessExecutor $process = null, RemoteFilesystem $rfs = null) public function __construct(IOInterface $io, Config $config, EventDispatcher $eventDispatcher = null, Cache $cache = null, ProcessExecutor $process = null, RemoteFilesystem $rfs = null)
{ {
@ -53,6 +55,10 @@ class ZipDownloader extends ArchiveDownloader
self::$hasZipArchive = class_exists('ZipArchive'); self::$hasZipArchive = class_exists('ZipArchive');
} }
if (null === self::$isWindows) {
self::$isWindows = Platform::isWindows();
}
if (!self::$hasZipArchive && !self::$hasSystemUnzip) { if (!self::$hasZipArchive && !self::$hasSystemUnzip) {
// php.ini path is added to the error message to help users find the correct file // php.ini path is added to the error message to help users find the correct file
$iniMessage = IniHelper::getMessage(); $iniMessage = IniHelper::getMessage();
@ -69,62 +75,99 @@ class ZipDownloader extends ArchiveDownloader
* *
* @param string $file File to extract * @param string $file File to extract
* @param string $path Path where to extract file * @param string $path Path where to extract file
* @param bool $isFallback If true it is called as a fallback and should not throw exception * @param bool $isLastChance If true it is called as a fallback and should throw an exception
* @return bool|\Exception True if succeed, an Exception if not * @return bool Success status
*/ */
protected function extractWithSystemUnzip($file, $path, $isFallback) protected function extractWithSystemUnzip($file, $path, $isLastChance)
{ {
if (! self::$hasZipArchive) {
// Force Exception throwing if the Other alternative is not available
$isLastChance = true;
}
if (! self::$hasSystemUnzip && ! $isLastChance) {
// This was call as the favorite extract way, but is not available
// We switch to the alternative
return $this->extractWithZipArchive($file, $path, true);
}
$processError = null; $processError = null;
// When called after a ZipArchive failed, perhaps there is some files to overwrite // When called after a ZipArchive failed, perhaps there is some files to overwrite
$overwrite = $isFallback ? '-o' : ''; $overwrite = $isLastChance ? '-o' : '';
$command = 'unzip -qq '.$overwrite.' '.ProcessExecutor::escape($file).' -d '.ProcessExecutor::escape($path); $command = 'unzip -qq '.$overwrite.' '.ProcessExecutor::escape($file).' -d '.ProcessExecutor::escape($path);
try { try {
if (0 === $this->process->execute($command, $ignoredOutput)) { if (0 === $this->process->execute($command, $ignoredOutput)) {
return TRUE; return true;
} }
$processError = 'Failed to execute ' . $command . "\n\n" . $this->process->getErrorOutput(); $processError = new \RuntimeException('Failed to execute ' . $command . "\n\n" . $this->process->getErrorOutput());
} catch (\Exception $e) { } catch (\Exception $e) {
$processError = 'Failed to execute ' . $command . "\n\n" . $e->getMessage(); $processError = $e;
}
if (! self::$hasZipArchive) {
$isLastChance = true;
} }
if ( $isFallback ) { if ($isLastChance) {
$this->io->write($processError); throw $processError;
} else {
$this->io->write($processError->getMessage());
$this->io->write('Unzip with unzip command failed, falling back to ZipArchive class');
return $this->extractWithZipArchive($file, $path, true);
} }
return new \RuntimeException($processError);
} }
/** /**
* extract $file to $path with ZipArchive * extract $file to $path with ZipArchive
* *
* @param string $file File to extract * @param string $file File to extract
* @param string $path Path where to extract file * @param string $path Path where to extract file
* @return bool|\Exception True if succeed, an Exception if not * @param bool $isLastChance If true it is called as a fallback and should throw an exception
* @return bool Success status
*/ */
protected function extractWithZipArchive($file, $path) protected function extractWithZipArchive($file, $path, $isLastChance)
{ {
$zipArchive = new ZipArchive(); if (! self::$hasSystemUnzip) {
// Force Exception throwing if the Other alternative is not available
$isLastChance = true;
}
if (true !== ($retval = $zipArchive->open($file))) { if (! self::$hasZipArchive && ! $isLastChance) {
return new \UnexpectedValueException(rtrim($this->getErrorMessage($retval, $file)."\n"), $retval); // This was call as the favorite extract way, but is not available
// We switch to the alternative
return $this->extractWithSystemUnzip($file, $path, true);
} }
$extractResult = FALSE; $processError = null;
$zipArchive = $this->zipArchiveObject ?: new ZipArchive();
try { try {
$extractResult = $zipArchive->extractTo($path); if (true === ($retval = $zipArchive->open($file))) {
} catch (\Exception $e ) { $extractResult = $zipArchive->extractTo($path);
return $e;
if (true === $extractResult) {
$zipArchive->close();
return true;
} else {
$processError = new \RuntimeException(rtrim("There was an error extracting the ZIP file, it is either corrupted or using an invalid format.\n"));
}
} else {
$processError = new \UnexpectedValueException(rtrim($this->getErrorMessage($retval, $file)."\n"), $retval);
}
} catch (\Exception $e) {
$processError = $e;
} }
if (true !== $extractResult) { if ($isLastChance) {
return new \RuntimeException(rtrim("There was an error extracting the ZIP file, it is either corrupted or using an invalid format.\n")); throw $processError;
} else {
$this->io->write($processError->getMessage());
$this->io->write('Unzip with ZipArchive class failed, falling back to unzip command');
return $this->extractWithSystemUnzip($file, $path, true);
} }
$zipArchive->close();
return TRUE;
} }
/** /**
@ -133,42 +176,14 @@ class ZipDownloader extends ArchiveDownloader
* @param string $file File to extract * @param string $file File to extract
* @param string $path Path where to extract file * @param string $path Path where to extract file
*/ */
protected function extract($file, $path) public function extract($file, $path)
{ {
$resultZipArchive = NULL; // Each extract calls its alternative if not available or fails
$resultUnzip = NULL; if (self::$isWindows) {
$this->extractWithZipArchive($file, $path, false);
if ( self::$hasZipArchive ) {
// zip module is present
$resultZipArchive = $this->extractWithZipArchive($file, $path);
if ($resultZipArchive === TRUE) {
return;
}
}
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 { } else {
// unZip failed $this->extractWithSystemUnzip($file, $path, false);
// zipArchive not available }
throw $resultUnzip;
};
} }
/** /**

@ -13,6 +13,7 @@
namespace Composer\Test\Downloader; namespace Composer\Test\Downloader;
use Composer\Downloader\ZipDownloader; use Composer\Downloader\ZipDownloader;
use Composer\Package\PackageInterface;
use Composer\TestCase; use Composer\TestCase;
use Composer\Util\Filesystem; use Composer\Util\Filesystem;
@ -22,24 +23,62 @@ class ZipDownloaderTest extends TestCase
* @var string * @var string
*/ */
private $testDir; private $testDir;
private $prophet;
public function setUp() public function setUp()
{ {
$this->testDir = $this->getUniqueTmpDirectory(); $this->testDir = $this->getUniqueTmpDirectory();
$this->io = $this->getMock('Composer\IO\IOInterface');
$this->config = $this->getMock('Composer\Config');
} }
public function tearDown() public function tearDown()
{ {
$fs = new Filesystem; $fs = new Filesystem;
$fs->removeDirectory($this->testDir); $fs->removeDirectory($this->testDir);
ZipDownloader::$hasSystemUnzip = null;
ZipDownloader::$hasZipArchive = null;
}
public function setPrivateProperty($name, $value, $obj = null)
{
$reflectionClass = new \ReflectionClass('Composer\Downloader\ZipDownloader');
$reflectedProperty = $reflectionClass->getProperty($name);
$reflectedProperty->setAccessible(true);
if ($obj === null) {
$reflectedProperty = $reflectedProperty->setValue($value);
} else {
$reflectedProperty = $reflectedProperty->setValue($obj, $value);
}
} }
/**
* @group only
*/
public function testErrorMessages() public function testErrorMessages()
{ {
if (!class_exists('ZipArchive')) { if (!class_exists('ZipArchive')) {
$this->markTestSkipped('zip extension missing'); $this->markTestSkipped('zip extension missing');
} }
$this->config->expects($this->at(0))
->method('get')
->with('disable-tls')
->will($this->returnValue(false));
$this->config->expects($this->at(1))
->method('get')
->with('cafile')
->will($this->returnValue(null));
$this->config->expects($this->at(2))
->method('get')
->with('capath')
->will($this->returnValue(null));
$this->config->expects($this->at(3))
->method('get')
->with('vendor-dir')
->will($this->returnValue($this->testDir));
$packageMock = $this->getMock('Composer\Package\PackageInterface'); $packageMock = $this->getMock('Composer\Package\PackageInterface');
$packageMock->expects($this->any()) $packageMock->expects($this->any())
->method('getDistUrl') ->method('getDistUrl')
@ -54,110 +93,205 @@ class ZipDownloaderTest extends TestCase
->will($this->returnValue(array())) ->will($this->returnValue(array()))
; ;
$io = $this->getMock('Composer\IO\IOInterface'); $downloader = new ZipDownloader($this->io, $this->config);
$config = $this->getMock('Composer\Config');
$config->expects($this->at(0)) ZipDownloader::$hasSystemUnzip = false;
->method('get')
->with('disable-tls')
->will($this->returnValue(false));
$config->expects($this->at(1))
->method('get')
->with('cafile')
->will($this->returnValue(null));
$config->expects($this->at(2))
->method('get')
->with('capath')
->will($this->returnValue(null));
$config->expects($this->at(3))
->method('get')
->with('vendor-dir')
->will($this->returnValue($this->testDir));
$downloader = new ZipDownloader($io, $config);
try { try {
$downloader->download($packageMock, sys_get_temp_dir().'/composer-zip-test'); $downloader->download($packageMock, sys_get_temp_dir().'/composer-zip-test');
$this->fail('Download of invalid zip files should throw an exception'); $this->fail('Download of invalid zip files should throw an exception');
} catch (\UnexpectedValueException $e) { } catch (\Exception $e) {
$this->assertContains('is not a zip archive', $e->getMessage()); $this->assertContains('is not a zip archive', $e->getMessage());
} }
} }
/** /**
* @expectedException \Exception * @expectedException \RuntimeException
* @expectedExceptionMessage ZipArchive Failed * @expectedExceptionMessage There was an error extracting the ZIP file
*/ */
function testZipArchiveOnlyFailed() { public function testZipArchiveOnlyFailed()
$downloader = new TestDownloader($this->getMock('Composer\IO\IOInterface')); {
$e = new \Exception("ZipArchive Failed"); MockedZipDownloader::$hasSystemUnzip = false;
$downloader->setUp(TRUE, FALSE, $e, NULL); MockedZipDownloader::$hasZipArchive = true;
$downloader = new MockedZipDownloader($this->io, $this->config);
$zipArchive = $this->getMock('ZipArchive');
$zipArchive->expects($this->at(0))
->method('open')
->will($this->returnValue(true));
$zipArchive->expects($this->at(1))
->method('extractTo')
->will($this->returnValue(false));
$this->setPrivateProperty('zipArchiveObject', $zipArchive, $downloader);
$downloader->extract('testfile.zip', 'vendor/dir'); $downloader->extract('testfile.zip', 'vendor/dir');
} }
function testZipArchiveOnlyGood() { /**
$downloader = new TestDownloader($this->getMock('Composer\IO\IOInterface')); * @group only
$downloader->setUp(TRUE, FALSE, TRUE, NULL); */
public function testZipArchiveOnlyGood()
{
MockedZipDownloader::$hasSystemUnzip = false;
MockedZipDownloader::$hasZipArchive = true;
$downloader = new MockedZipDownloader($this->io, $this->config);
$zipArchive = $this->getMock('ZipArchive');
$zipArchive->expects($this->at(0))
->method('open')
->will($this->returnValue(true));
$zipArchive->expects($this->at(1))
->method('extractTo')
->will($this->returnValue(true));
$this->setPrivateProperty('zipArchiveObject', $zipArchive, $downloader);
$downloader->extract('testfile.zip', 'vendor/dir'); $downloader->extract('testfile.zip', 'vendor/dir');
} }
/** /**
* @expectedException \Exception * @expectedException \Exception
* @expectedExceptionMessage SystemUnzip Failed * @expectedExceptionMessage Failed to execute unzip
*/ */
function testSystemUnzipOnlyFailed() { public function testSystemUnzipOnlyFailed()
$downloader = new TestDownloader($this->getMock('Composer\IO\IOInterface')); {
$e = new \Exception("SystemUnzip Failed"); MockedZipDownloader::$hasSystemUnzip = true;
$downloader->setUp(FALSE, TRUE, NULL, $e); MockedZipDownloader::$hasZipArchive = false;
$processExecutor = $this->getMock('Composer\Util\ProcessExecutor');
$processExecutor->expects($this->at(0))
->method('execute')
->will($this->returnValue(1));
$downloader = new MockedZipDownloader($this->io, $this->config, null, null, $processExecutor);
$downloader->extract('testfile.zip', 'vendor/dir'); $downloader->extract('testfile.zip', 'vendor/dir');
} }
function testSystemUnzipOnlyGood() { public function testSystemUnzipOnlyGood()
$downloader = new TestDownloader($this->getMock('Composer\IO\IOInterface')); {
$downloader->setUp(FALSE, TRUE, NULL, TRUE); MockedZipDownloader::$hasSystemUnzip = true;
MockedZipDownloader::$hasZipArchive = false;
$processExecutor = $this->getMock('Composer\Util\ProcessExecutor');
$processExecutor->expects($this->at(0))
->method('execute')
->will($this->returnValue(0));
$downloader = new MockedZipDownloader($this->io, $this->config, null, null, $processExecutor);
$downloader->extract('testfile.zip', 'vendor/dir'); $downloader->extract('testfile.zip', 'vendor/dir');
} }
function testSystemUnzipFallbackGood() { public function testNonWindowsFallbackGood()
$downloader = new TestDownloader($this->getMock('Composer\IO\IOInterface')); {
$e = new \Exception("test"); MockedZipDownloader::$isWindows = false;
$downloader->setUp(TRUE, TRUE, $e, TRUE); MockedZipDownloader::$hasSystemUnzip = true;
MockedZipDownloader::$hasZipArchive = true;
$processExecutor = $this->getMock('Composer\Util\ProcessExecutor');
$processExecutor->expects($this->at(0))
->method('execute')
->will($this->returnValue(1));
$zipArchive = $this->getMock('ZipArchive');
$zipArchive->expects($this->at(0))
->method('open')
->will($this->returnValue(true));
$zipArchive->expects($this->at(1))
->method('extractTo')
->will($this->returnValue(true));
$downloader = new MockedZipDownloader($this->io, $this->config, null, null, $processExecutor);
$this->setPrivateProperty('zipArchiveObject', $zipArchive, $downloader);
$downloader->extract('testfile.zip', 'vendor/dir'); $downloader->extract('testfile.zip', 'vendor/dir');
} }
/** /**
* @expectedException \Exception * @expectedException \Exception
* @expectedExceptionMessage ZipArchive Failed * @expectedExceptionMessage There was an error extracting the ZIP file
*/ */
function testSystemUnzipFallbackFailed() { public function testNonWindowsFallbackFailed()
$downloader = new TestDownloader($this->getMock('Composer\IO\IOInterface')); {
$e1 = new \Exception("ZipArchive Failed"); MockedZipDownloader::$isWindows = false;
$e2 = new \Exception("SystemUnzip Failed"); MockedZipDownloader::$hasSystemUnzip = true;
$downloader->setUp(TRUE, TRUE, $e1, $e2); MockedZipDownloader::$hasZipArchive = true;
$processExecutor = $this->getMock('Composer\Util\ProcessExecutor');
$processExecutor->expects($this->at(0))
->method('execute')
->will($this->returnValue(1));
$zipArchive = $this->getMock('ZipArchive');
$zipArchive->expects($this->at(0))
->method('open')
->will($this->returnValue(true));
$zipArchive->expects($this->at(1))
->method('extractTo')
->will($this->returnValue(false));
$downloader = new MockedZipDownloader($this->io, $this->config, null, null, $processExecutor);
$this->setPrivateProperty('zipArchiveObject', $zipArchive, $downloader);
$downloader->extract('testfile.zip', 'vendor/dir'); $downloader->extract('testfile.zip', 'vendor/dir');
} }
}
class TestDownloader extends ZipDownloader { public function testWindowsFallbackGood()
public function __construct($io)
{ {
$this->io = $io; MockedZipDownloader::$isWindows = true;
} MockedZipDownloader::$hasSystemUnzip = true;
MockedZipDownloader::$hasZipArchive = true;
public function extract($file, $path) { $processExecutor = $this->getMock('Composer\Util\ProcessExecutor');
parent::extract($file, $path); $processExecutor->expects($this->atLeastOnce())
->method('execute')
->will($this->returnValue(0));
$zipArchive = $this->getMock('ZipArchive');
$zipArchive->expects($this->at(0))
->method('open')
->will($this->returnValue(true));
$zipArchive->expects($this->at(1))
->method('extractTo')
->will($this->returnValue(false));
$downloader = new MockedZipDownloader($this->io, $this->config, null, null, $processExecutor);
$this->setPrivateProperty('zipArchiveObject', $zipArchive, $downloader);
$downloader->extract('testfile.zip', 'vendor/dir');
} }
public function setUp($zipArchive, $systemUnzip, $zipArchiveResponse, $systemUnzipResponse) { /**
self::$hasZipArchive = $zipArchive; * @expectedException \Exception
self::$hasSystemUnzip = $systemUnzip; * @expectedExceptionMessage Failed to execute unzip
$this->zipArchiveResponse = $zipArchiveResponse; */
$this->systemUnzipResponse = $systemUnzipResponse; public function testWindowsFallbackFailed()
{
MockedZipDownloader::$isWindows = true;
MockedZipDownloader::$hasSystemUnzip = true;
MockedZipDownloader::$hasZipArchive = true;
$processExecutor = $this->getMock('Composer\Util\ProcessExecutor');
$processExecutor->expects($this->atLeastOnce())
->method('execute')
->will($this->returnValue(1));
$zipArchive = $this->getMock('ZipArchive');
$zipArchive->expects($this->at(0))
->method('open')
->will($this->returnValue(true));
$zipArchive->expects($this->at(1))
->method('extractTo')
->will($this->returnValue(false));
$downloader = new MockedZipDownloader($this->io, $this->config, null, null, $processExecutor);
$this->setPrivateProperty('zipArchiveObject', $zipArchive, $downloader);
$downloader->extract('testfile.zip', 'vendor/dir');
} }
}
protected function extractWithZipArchive($file, $path) { class MockedZipDownloader extends ZipDownloader
return $this->zipArchiveResponse; {
public function download(PackageInterface $package, $path, $output = true)
{
return;
} }
protected function extractWithSystemUnzip($file, $path, $fallback) { public function extract($file, $path)
return $this->systemUnzipResponse; {
parent::extract($file, $path);
} }
} }

Loading…
Cancel
Save