diff --git a/typo3/sysext/core/Classes/Exception/Archive/ExtractException.php b/typo3/sysext/core/Classes/Exception/Archive/ExtractException.php new file mode 100644 index 0000000000000000000000000000000000000000..8febbea3d88f38c2eeda7555f67b47630a258deb --- /dev/null +++ b/typo3/sysext/core/Classes/Exception/Archive/ExtractException.php @@ -0,0 +1,25 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Core\Exception\Archive; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Core\Exception; + +/** + * An exception thrown when extracting an archive fails + */ +class ExtractException extends Exception +{ +} diff --git a/typo3/sysext/core/Classes/Service/Archive/ZipService.php b/typo3/sysext/core/Classes/Service/Archive/ZipService.php new file mode 100644 index 0000000000000000000000000000000000000000..7484c4e562081697d149f2eb29035e82822905d3 --- /dev/null +++ b/typo3/sysext/core/Classes/Service/Archive/ZipService.php @@ -0,0 +1,103 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Core\Service\Archive; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Core\Exception\Archive\ExtractException; + +/** + * Service that handles zip creation and extraction + * + * @internal + */ +class ZipService +{ + /** + * Extracts the zip archive to a given directory. This method makes sure a file cannot be placed outside the directory. + * + * @param string $fileName + * @param string $directory + * @return bool + * @throws ExtractException + */ + public function extract(string $fileName, string $directory): bool + { + $this->assertDirectoryIsWritable($directory); + + $zip = new \ZipArchive(); + $state = $zip->open($fileName); + if ($state !== true) { + throw new ExtractException( + sprintf('Unable to open zip file %s, error code %d', $fileName, $state), + 1565709712 + ); + } + + $result = $zip->extractTo($directory); + $zip->close(); + return $result; + } + + /** + * @param string $fileName + * @return bool + * @throws ExtractException + */ + public function verify(string $fileName): bool + { + $zip = new \ZipArchive(); + $state = $zip->open($fileName); + if ($state !== true) { + throw new ExtractException( + sprintf('Unable to open zip file %s, error code %d', $fileName, $state), + 1565709713 + ); + } + + for ($i = 0; $i < $zip->numFiles; $i++) { + $entryName = $zip->getNameIndex($i); + if (preg_match('#/(?:\.{2,})+#', $entryName) // Contains any traversal sequence starting with a slash, e.g. /../, /.., /.../ + || preg_match('#^(?:\.{2,})+/#', $entryName) // Starts with a traversal sequence, e.g. ../, .../ + ) { + throw new ExtractException( + sprintf('Suspicious sequence in zip file %s: %s', $fileName, $entryName), + 1565709714 + ); + } + } + + $zip->close(); + return true; + } + + /** + * @param string $directory + */ + private function assertDirectoryIsWritable(string $directory): void + { + if (!is_dir($directory)) { + throw new \RuntimeException( + sprintf('Directory %s does not exist', $directory), + 1565773005 + ); + } + if (!is_writable($directory)) { + throw new \RuntimeException( + sprintf('Directory %s is not writable', $directory), + 1565773006 + ); + } + } +} diff --git a/typo3/sysext/core/Tests/Functional/Service/Archive/Fixtures/malicious.zip b/typo3/sysext/core/Tests/Functional/Service/Archive/Fixtures/malicious.zip new file mode 100644 index 0000000000000000000000000000000000000000..b050224dc77e7e7a718c4b17c88a92e1a77fb0f1 Binary files /dev/null and b/typo3/sysext/core/Tests/Functional/Service/Archive/Fixtures/malicious.zip differ diff --git a/typo3/sysext/core/Tests/Functional/Service/Archive/Fixtures/my_extension.zip b/typo3/sysext/core/Tests/Functional/Service/Archive/Fixtures/my_extension.zip new file mode 100644 index 0000000000000000000000000000000000000000..852ec4c295ad3606952f0510900a0c7c0fc55fca Binary files /dev/null and b/typo3/sysext/core/Tests/Functional/Service/Archive/Fixtures/my_extension.zip differ diff --git a/typo3/sysext/core/Tests/Functional/Service/Archive/ZipServiceTest.php b/typo3/sysext/core/Tests/Functional/Service/Archive/ZipServiceTest.php new file mode 100644 index 0000000000000000000000000000000000000000..f16bf57f614eb1f24c492ea78bb1274d06fea798 --- /dev/null +++ b/typo3/sysext/core/Tests/Functional/Service/Archive/ZipServiceTest.php @@ -0,0 +1,163 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Core\Tests\Functional\Service\Archive; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use org\bovigo\vfs\vfsStream; +use org\bovigo\vfs\vfsStreamDirectory; +use TYPO3\CMS\Core\Exception\Archive\ExtractException; +use TYPO3\CMS\Core\Service\Archive\ZipService; +use TYPO3\CMS\Core\Utility\GeneralUtility; +use TYPO3\TestingFramework\Core\Functional\FunctionalTestCase; + +class ZipServiceTest extends FunctionalTestCase +{ + /** + * @var vfsStreamDirectory + */ + private $vfs; + + /** + * @var string + */ + private $directory; + + protected function setUp(): void + { + parent::setUp(); + + $structure = [ + 'typo3conf' => [ + 'ext' => [], + ], + ]; + $this->vfs = vfsStream::setup('root', null, $structure); + $this->directory = vfsStream::url('root/typo3conf/ext'); + } + + protected function tearDown(): void + { + parent::tearDown(); + unset($this->vfs, $this->directory); + } + + /** + * @test + */ + public function filesCanNotGetExtractedOutsideTargetDirectory(): void + { + $extensionDirectory = $this->directory . '/malicious'; + GeneralUtility::mkdir($extensionDirectory); + + (new ZipService())->extract( + __DIR__ . '/Fixtures/malicious.zip', + $extensionDirectory + ); + GeneralUtility::fixPermissions($extensionDirectory, true); + + self::assertFileNotExists($extensionDirectory . '/../tool.php'); + self::assertFileExists($extensionDirectory . '/tool.php'); + // This is a smoke test to verify PHP's zip library is broken regarding symlinks + self::assertFileExists($extensionDirectory . '/passwd'); + self::assertFalse(is_link($extensionDirectory . '/passwd')); + } + + /** + * @test + */ + public function fileContentIsExtractedAsExpected(): void + { + $extensionDirectory = $this->directory . '/my_extension'; + GeneralUtility::mkdir($extensionDirectory); + + (new ZipService())->extract( + __DIR__ . '/Fixtures/my_extension.zip', + $extensionDirectory + ); + GeneralUtility::fixPermissions($extensionDirectory, true); + + self::assertDirectoryExists($extensionDirectory . '/Classes'); + self::assertFileExists($extensionDirectory . '/Resources/Public/Css/empty.css'); + self::assertFileExists($extensionDirectory . '/ext_emconf.php'); + } + + /** + * @test + */ + public function nonExistentFileThrowsException(): void + { + $this->expectException(ExtractException::class); + $this->expectExceptionCode(1565709712); + + (new ZipService())->extract( + 'foobar.zip', + vfsStream::url('root') + ); + } + + /** + * @test + */ + public function nonExistentDirectoryThrowsException(): void + { + $this->expectException(\RuntimeException::class); + $this->expectExceptionCode(1565773005); + + (new ZipService())->extract( + __DIR__ . '/Fixtures/my_extension.zip', + vfsStream::url('root/non-existent-directory') + ); + } + + /** + * @test + */ + public function nonWritableDirectoryThrowsException(): void + { + $this->expectException(\RuntimeException::class); + $this->expectExceptionCode(1565773006); + + $extensionDirectory = $this->directory . '/my_extension'; + GeneralUtility::mkdir($extensionDirectory); + chmod($extensionDirectory, 0000); + + (new ZipService())->extract( + __DIR__ . '/Fixtures/my_extension.zip', + $extensionDirectory + ); + self::assertFileExists($extensionDirectory . '/Resources/Public/Css/empty.css'); + } + + /** + * @test + */ + public function verifyDetectsValidArchive(): void + { + self::assertTrue( + (new ZipService())->verify(__DIR__ . '/Fixtures/my_extension.zip') + ); + } + + /** + * @test + */ + public function verifyDetectsSuspiciousSequences(): void + { + $this->expectException(ExtractException::class); + $this->expectExceptionCode(1565709714); + + (new ZipService())->verify(__DIR__ . '/Fixtures/malicious.zip'); + } +} diff --git a/typo3/sysext/extensionmanager/Classes/Utility/FileHandlingUtility.php b/typo3/sysext/extensionmanager/Classes/Utility/FileHandlingUtility.php index 6ca2c77dfda4ed02ae32fafbf1e522d1af4bc140..df0f6b701644d40d2940e81e657bcf809914ca4f 100644 --- a/typo3/sysext/extensionmanager/Classes/Utility/FileHandlingUtility.php +++ b/typo3/sysext/extensionmanager/Classes/Utility/FileHandlingUtility.php @@ -14,8 +14,12 @@ namespace TYPO3\CMS\Extensionmanager\Utility; * The TYPO3 project - inspiring people to share! */ +use Psr\Log\LoggerAwareInterface; +use Psr\Log\LoggerAwareTrait; use TYPO3\CMS\Core\Core\Environment; +use TYPO3\CMS\Core\Exception\Archive\ExtractException; use TYPO3\CMS\Core\Localization\LanguageService; +use TYPO3\CMS\Core\Service\Archive\ZipService; use TYPO3\CMS\Core\SingletonInterface; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Utility\PathUtility; @@ -26,8 +30,10 @@ use TYPO3\CMS\Extensionmanager\Exception\ExtensionManagerException; * Utility for dealing with files and folders * @internal This class is a specific ExtensionManager implementation and is not part of the Public TYPO3 API. */ -class FileHandlingUtility implements SingletonInterface +class FileHandlingUtility implements SingletonInterface, LoggerAwareInterface { + use LoggerAwareTrait; + /** * @var EmConfUtility */ @@ -415,29 +421,18 @@ class FileHandlingUtility implements SingletonInterface public function unzipExtensionFromFile($file, $fileName, $pathType = 'Local') { $extensionDir = $this->makeAndClearExtensionDir($fileName, $pathType); - $zip = zip_open($file); - if (is_resource($zip)) { - while (($zipEntry = zip_read($zip)) !== false) { - if (strpos(zip_entry_name($zipEntry), '/') !== false) { - $last = strrpos(zip_entry_name($zipEntry), '/'); - $dir = substr(zip_entry_name($zipEntry), 0, $last); - $file = substr(zip_entry_name($zipEntry), strrpos(zip_entry_name($zipEntry), '/') + 1); - if (!is_dir($extensionDir . $dir)) { - GeneralUtility::mkdir_deep($extensionDir . $dir); - } - if (trim($file) !== '') { - $return = GeneralUtility::writeFile($extensionDir . $dir . '/' . $file, zip_entry_read($zipEntry, zip_entry_filesize($zipEntry))); - if ($return === false) { - throw new ExtensionManagerException('Could not write file ' . $this->getRelativePath($file), 1344691048); - } - } - } else { - GeneralUtility::writeFile($extensionDir . zip_entry_name($zipEntry), zip_entry_read($zipEntry, zip_entry_filesize($zipEntry))); - } + + try { + $zipService = GeneralUtility::makeInstance(ZipService::class); + if ($zipService->verify($file)) { + $zipService->extract($file, $extensionDir); } - } else { - throw new ExtensionManagerException('Unable to open zip file ' . $this->getRelativePath($file), 1344691049); + } catch (ExtractException $e) { + $this->logger->error('Extracting the extension archive failed', ['exception' => $e]); + throw new ExtensionManagerException('Extracting the extension archive failed: ' . $e->getMessage(), 1565777179, $e); } + + GeneralUtility::fixPermissions($extensionDir, true); } /** diff --git a/typo3/sysext/install/Classes/Service/LanguagePackService.php b/typo3/sysext/install/Classes/Service/LanguagePackService.php index 5d6bcc52ec05f6b4a3391b1feb120aaf213024d8..fc2a2a3427eb3c6f50f1dadc350bfa942f3c81c0 100644 --- a/typo3/sysext/install/Classes/Service/LanguagePackService.php +++ b/typo3/sysext/install/Classes/Service/LanguagePackService.php @@ -26,6 +26,7 @@ use TYPO3\CMS\Core\Http\Uri; use TYPO3\CMS\Core\Localization\Locales; use TYPO3\CMS\Core\Package\PackageManager; use TYPO3\CMS\Core\Registry; +use TYPO3\CMS\Core\Service\Archive\ZipService; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Utility\PathUtility; @@ -361,43 +362,17 @@ class LanguagePackService implements LoggerAwareInterface * * @param string $file path to zip file * @param string $path path to extract to - * @throws \RuntimeException */ protected function unzipTranslationFile(string $file, string $path) { - $zip = zip_open($file); - if (is_resource($zip)) { - if (!is_dir($path)) { - GeneralUtility::mkdir_deep($path); - } - while (($zipEntry = zip_read($zip)) !== false) { - $zipEntryName = zip_entry_name($zipEntry); - if (strpos($zipEntryName, '/') !== false) { - $zipEntryPathSegments = explode('/', $zipEntryName); - $fileName = array_pop($zipEntryPathSegments); - // It is a folder, because the last segment is empty, let's create it - if (empty($fileName)) { - GeneralUtility::mkdir_deep($path . implode('/', $zipEntryPathSegments)); - } else { - $absoluteTargetPath = GeneralUtility::getFileAbsFileName($path . implode('/', $zipEntryPathSegments) . '/' . $fileName); - if (trim($absoluteTargetPath) !== '') { - $return = GeneralUtility::writeFile( - $absoluteTargetPath, - zip_entry_read($zipEntry, zip_entry_filesize($zipEntry)) - ); - if ($return === false) { - throw new \RuntimeException('Could not write file ' . $zipEntryName, 1520170845); - } - } else { - throw new \RuntimeException('Could not write file ' . $zipEntryName, 1520170846); - } - } - } else { - throw new \RuntimeException('Extension directory missing in zip file!', 1520170847); - } - } - } else { - throw new \RuntimeException('Unable to open zip file ' . $file, 1520170848); + if (!is_dir($path)) { + GeneralUtility::mkdir_deep($path); + } + + $zipService = GeneralUtility::makeInstance(ZipService::class); + if ($zipService->verify($file)) { + $zipService->extract($file, $path); } + GeneralUtility::fixPermissions($path, true); } }