ZipFileTest.php 59 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850
  1. <?php
  2. namespace PhpZip;
  3. use PhpZip\Exception\ZipAuthenticationException;
  4. use PhpZip\Model\ZipEntry;
  5. use PhpZip\Util\CryptoUtil;
  6. use PhpZip\Util\FilesUtil;
  7. /**
  8. * ZipFile test
  9. */
  10. class ZipFileTest extends ZipTestCase
  11. {
  12. /**
  13. * @expectedException \PhpZip\Exception\InvalidArgumentException
  14. * @expectedExceptionMessage can't exists
  15. */
  16. public function testOpenFileCantExists()
  17. {
  18. $zipFile = new ZipFile();
  19. $zipFile->openFile(uniqid());
  20. }
  21. /**
  22. * @expectedException \PhpZip\Exception\ZipException
  23. * @expectedExceptionMessage can't open
  24. */
  25. public function testOpenFileCantOpen()
  26. {
  27. self::assertNotFalse(file_put_contents($this->outputFilename, 'content'));
  28. self::assertTrue(chmod($this->outputFilename, 0222));
  29. $zipFile = new ZipFile();
  30. $zipFile->openFile($this->outputFilename);
  31. }
  32. /**
  33. * @expectedException \PhpZip\Exception\ZipException
  34. * @expectedExceptionMessage Invalid zip file
  35. */
  36. public function testOpenFileEmptyFile()
  37. {
  38. self::assertNotFalse(touch($this->outputFilename));
  39. $zipFile = new ZipFile();
  40. $zipFile->openFile($this->outputFilename);
  41. }
  42. /**
  43. * @expectedException \PhpZip\Exception\ZipException
  44. * @expectedExceptionMessage Expected Local File Header or (ZIP64) End Of Central Directory Record
  45. */
  46. public function testOpenFileInvalidZip()
  47. {
  48. self::assertNotFalse(file_put_contents($this->outputFilename, CryptoUtil::randomBytes(255)));
  49. $zipFile = new ZipFile();
  50. $zipFile->openFile($this->outputFilename);
  51. }
  52. /**
  53. * @expectedException \PhpZip\Exception\InvalidArgumentException
  54. * @expectedExceptionMessage Data not available
  55. */
  56. public function testOpenFromStringNullString()
  57. {
  58. $zipFile = new ZipFile();
  59. $zipFile->openFromString(null);
  60. }
  61. /**
  62. * @expectedException \PhpZip\Exception\InvalidArgumentException
  63. * @expectedExceptionMessage Data not available
  64. */
  65. public function testOpenFromStringEmptyString()
  66. {
  67. $zipFile = new ZipFile();
  68. $zipFile->openFromString("");
  69. }
  70. /**
  71. * @expectedException \PhpZip\Exception\ZipException
  72. * @expectedExceptionMessage Expected Local File Header or (ZIP64) End Of Central Directory Record
  73. */
  74. public function testOpenFromStringInvalidZip()
  75. {
  76. $zipFile = new ZipFile();
  77. $zipFile->openFromString(CryptoUtil::randomBytes(255));
  78. }
  79. public function testOpenFromString()
  80. {
  81. $zipFile = new ZipFile();
  82. $zipFile->addFromString('file', 'content');
  83. $zipFile['file2'] = 'content 2';
  84. $zipContents = $zipFile->outputAsString();
  85. $zipFile->close();
  86. $zipFile->openFromString($zipContents);
  87. self::assertEquals($zipFile->count(), 2);
  88. self::assertTrue(isset($zipFile['file']));
  89. self::assertTrue(isset($zipFile['file2']));
  90. self::assertEquals($zipFile['file'], 'content');
  91. self::assertEquals($zipFile['file2'], 'content 2');
  92. $zipFile->close();
  93. }
  94. /**
  95. * @expectedException \PhpZip\Exception\InvalidArgumentException
  96. * @expectedExceptionMessage Invalid stream resource
  97. */
  98. public function testOpenFromStreamNullStream()
  99. {
  100. $zipFile = new ZipFile();
  101. $zipFile->openFromStream(null);
  102. }
  103. /**
  104. * @expectedException \PhpZip\Exception\InvalidArgumentException
  105. * @expectedExceptionMessage Invalid stream resource
  106. */
  107. public function testOpenFromStreamInvalidResourceType()
  108. {
  109. $zipFile = new ZipFile();
  110. $zipFile->openFromStream("stream resource");
  111. }
  112. /**
  113. * @expectedException \PhpZip\Exception\InvalidArgumentException
  114. * @expectedExceptionMessage Resource cannot seekable stream.
  115. */
  116. public function testOpenFromStreamNoSeekable()
  117. {
  118. if (!$fp = @fopen("http://localhost", 'r')) {
  119. if (!$fp = @fopen("http://example.org", 'r')) {
  120. $this->markTestSkipped('not connected to localhost or remote host');
  121. return;
  122. }
  123. }
  124. $zipFile = new ZipFile();
  125. $zipFile->openFromStream($fp);
  126. }
  127. /**
  128. * @expectedException \PhpZip\Exception\ZipException
  129. * @expectedExceptionMessage Invalid zip file
  130. */
  131. public function testOpenFromStreamEmptyContents()
  132. {
  133. $fp = fopen($this->outputFilename, 'w+b');
  134. $zipFile = new ZipFile();
  135. $zipFile->openFromStream($fp);
  136. }
  137. /**
  138. * @expectedException \PhpZip\Exception\ZipException
  139. * @expectedExceptionMessage Expected Local File Header or (ZIP64) End Of Central Directory Record
  140. */
  141. public function testOpenFromStreamInvalidZip()
  142. {
  143. $fp = fopen($this->outputFilename, 'w+b');
  144. fwrite($fp, CryptoUtil::randomBytes(255));
  145. $zipFile = new ZipFile();
  146. $zipFile->openFromStream($fp);
  147. }
  148. public function testOpenFromStream()
  149. {
  150. $zipFile = new ZipFile();
  151. $zipFile
  152. ->addFromString('file', 'content')
  153. ->saveAsFile($this->outputFilename);
  154. $zipFile->close();
  155. $handle = fopen($this->outputFilename, 'rb');
  156. $zipFile->openFromStream($handle);
  157. self::assertEquals($zipFile->count(), 1);
  158. self::assertTrue(isset($zipFile['file']));
  159. self::assertEquals($zipFile['file'], 'content');
  160. $zipFile->close();
  161. }
  162. /**
  163. * Test create, open and extract empty archive.
  164. */
  165. public function testEmptyArchive()
  166. {
  167. $zipFile = new ZipFile();
  168. $zipFile->saveAsFile($this->outputFilename);
  169. $zipFile->close();
  170. self::assertCorrectEmptyZip($this->outputFilename);
  171. self::assertTrue(mkdir($this->outputDirname, 0755, true));
  172. $zipFile->openFile($this->outputFilename);
  173. self::assertEquals($zipFile->count(), 0);
  174. $zipFile->extractTo($this->outputDirname);
  175. $zipFile->close();
  176. self::assertTrue(FilesUtil::isEmptyDir($this->outputDirname));
  177. }
  178. /**
  179. * No modified archive
  180. *
  181. * @see ZipOutputFile::create()
  182. */
  183. public function testNoModifiedArchive()
  184. {
  185. self::assertTrue(mkdir($this->outputDirname, 0755, true));
  186. $fileActual = $this->outputDirname . DIRECTORY_SEPARATOR . 'file_actual.zip';
  187. $fileExpected = $this->outputDirname . DIRECTORY_SEPARATOR . 'file_expected.zip';
  188. $zipFile = new ZipFile();
  189. $zipFile->addDirRecursive(__DIR__);
  190. $zipFile->saveAsFile($fileActual);
  191. self::assertCorrectZipArchive($fileActual);
  192. $zipFile->close();
  193. $zipFile->openFile($fileActual);
  194. $zipFile->saveAsFile($fileExpected);
  195. self::assertCorrectZipArchive($fileExpected);
  196. $zipFileExpected = new ZipFile();
  197. $zipFileExpected->openFile($fileExpected);
  198. self::assertEquals($zipFileExpected->count(), $zipFile->count());
  199. self::assertEquals($zipFileExpected->getListFiles(), $zipFile->getListFiles());
  200. foreach ($zipFile as $entryName => $content) {
  201. self::assertEquals($zipFileExpected[$entryName], $content);
  202. }
  203. $zipFileExpected->close();
  204. $zipFile->close();
  205. }
  206. /**
  207. * Create archive and add files.
  208. *
  209. * @see ZipOutputFile::addFromString()
  210. * @see ZipOutputFile::addFromFile()
  211. * @see ZipOutputFile::addFromStream()
  212. * @see ZipFile::getEntryContent()
  213. */
  214. public function testCreateArchiveAndAddFiles()
  215. {
  216. $outputFromString = file_get_contents(__FILE__);
  217. $outputFromString2 = file_get_contents(dirname(dirname(__DIR__)) . DIRECTORY_SEPARATOR . 'README.md');
  218. $outputFromFile = file_get_contents(dirname(dirname(__DIR__)) . DIRECTORY_SEPARATOR . 'bootstrap.xml');
  219. $outputFromStream = file_get_contents(dirname(dirname(__DIR__)) . DIRECTORY_SEPARATOR . 'composer.json');
  220. $filenameFromString = basename(__FILE__);
  221. $filenameFromString2 = 'test_file.txt';
  222. $filenameFromFile = 'data/test file.txt';
  223. $filenameFromStream = 'data/ডিরেক্টরি/αρχείο.json';
  224. $emptyDirName = 'empty dir/пустой каталог/空目錄/ไดเรกทอรีที่ว่างเปล่า/';
  225. $emptyDirName2 = 'empty dir/пустой каталог/';
  226. $emptyDirName3 = 'empty dir/пустой каталог/ещё один пустой каталог/';
  227. $tempFile = tempnam(sys_get_temp_dir(), 'txt');
  228. file_put_contents($tempFile, $outputFromFile);
  229. $tempStream = tmpfile();
  230. fwrite($tempStream, $outputFromStream);
  231. $zipFile = new ZipFile;
  232. $zipFile->addFromString($filenameFromString, $outputFromString);
  233. $zipFile->addFile($tempFile, $filenameFromFile);
  234. $zipFile->addFromStream($tempStream, $filenameFromStream);
  235. $zipFile->addEmptyDir($emptyDirName);
  236. $zipFile[$filenameFromString2] = $outputFromString2;
  237. $zipFile[$emptyDirName2] = null;
  238. $zipFile[$emptyDirName3] = 'this content ignoring';
  239. $zipFile->saveAsFile($this->outputFilename);
  240. $zipFile->close();
  241. unlink($tempFile);
  242. self::assertCorrectZipArchive($this->outputFilename);
  243. $zipFile->openFile($this->outputFilename);
  244. self::assertEquals(count($zipFile), 7);
  245. self::assertEquals($zipFile[$filenameFromString], $outputFromString);
  246. self::assertEquals($zipFile[$filenameFromFile], $outputFromFile);
  247. self::assertEquals($zipFile[$filenameFromStream], $outputFromStream);
  248. self::assertEquals($zipFile[$filenameFromString2], $outputFromString2);
  249. self::assertTrue(isset($zipFile[$emptyDirName]));
  250. self::assertTrue(isset($zipFile[$emptyDirName2]));
  251. self::assertTrue(isset($zipFile[$emptyDirName3]));
  252. self::assertTrue($zipFile->isDirectory($emptyDirName));
  253. self::assertTrue($zipFile->isDirectory($emptyDirName2));
  254. self::assertTrue($zipFile->isDirectory($emptyDirName3));
  255. $listFiles = $zipFile->getListFiles();
  256. self::assertEquals($listFiles[0], $filenameFromString);
  257. self::assertEquals($listFiles[1], $filenameFromFile);
  258. self::assertEquals($listFiles[2], $filenameFromStream);
  259. self::assertEquals($listFiles[3], $emptyDirName);
  260. self::assertEquals($listFiles[4], $filenameFromString2);
  261. self::assertEquals($listFiles[5], $emptyDirName2);
  262. self::assertEquals($listFiles[6], $emptyDirName3);
  263. $zipFile->close();
  264. }
  265. /**
  266. * Test compression method from image file.
  267. */
  268. public function testCompressionMethodFromImageMimeType()
  269. {
  270. if (!function_exists('mime_content_type')) {
  271. $this->markTestSkipped('Function mime_content_type not exists');
  272. }
  273. $outputFilename = $this->outputFilename;
  274. $this->outputFilename .= '.gif';
  275. self::assertNotFalse(
  276. file_put_contents(
  277. $this->outputFilename,
  278. base64_decode('R0lGODlhAQABAJAAAP8AAAAAACH5BAUQAAAALAAAAAABAAEAAAICBAEAOw==')
  279. )
  280. );
  281. $basename = basename($this->outputFilename);
  282. $zipFile = new ZipFile();
  283. $zipFile->addFile($this->outputFilename, $basename);
  284. $zipFile->saveAsFile($outputFilename);
  285. unlink($this->outputFilename);
  286. $this->outputFilename = $outputFilename;
  287. $zipFile->openFile($this->outputFilename);
  288. $info = $zipFile->getEntryInfo($basename);
  289. self::assertEquals($info->getMethod(), 'No compression');
  290. $zipFile->close();
  291. }
  292. /**
  293. * Rename zip entry name.
  294. */
  295. public function testRename()
  296. {
  297. $oldName = basename(__FILE__);
  298. $newName = 'tests/' . $oldName;
  299. $zipFile = new ZipFile();
  300. $zipFile->addDirRecursive(__DIR__);
  301. $zipFile->saveAsFile($this->outputFilename);
  302. $zipFile->close();
  303. self::assertCorrectZipArchive($this->outputFilename);
  304. $zipFile->openFile($this->outputFilename);
  305. $zipFile->rename($oldName, $newName);
  306. $zipFile->addFromString('file1.txt', 'content');
  307. $zipFile->rename('file1.txt', 'file2.txt');
  308. $zipFile->saveAsFile($this->outputFilename);
  309. $zipFile->close();
  310. self::assertCorrectZipArchive($this->outputFilename);
  311. $zipFile->openFile($this->outputFilename);
  312. self::assertFalse(isset($zipFile[$oldName]));
  313. self::assertTrue(isset($zipFile[$newName]));
  314. self::assertFalse(isset($zipFile['file1.txt']));
  315. self::assertTrue(isset($zipFile['file2.txt']));
  316. $zipFile->close();
  317. }
  318. /**
  319. * @expectedException \PhpZip\Exception\InvalidArgumentException
  320. * @expectedExceptionMessage name is null
  321. */
  322. public function testRenameEntryNull()
  323. {
  324. $zipFile = new ZipFile();
  325. $zipFile->rename(null, 'new-file');
  326. }
  327. /**
  328. * @expectedException \PhpZip\Exception\InvalidArgumentException
  329. * @expectedExceptionMessage name is null
  330. */
  331. public function testRenameEntryNull2()
  332. {
  333. $zipFile = new ZipFile();
  334. $zipFile->rename('old-file', null);
  335. }
  336. /**
  337. * @expectedException \PhpZip\Exception\InvalidArgumentException
  338. * @expectedExceptionMessage is exists
  339. */
  340. public function testRenameEntryNewEntyExists()
  341. {
  342. $zipFile = new ZipFile();
  343. $zipFile['file'] = 'content';
  344. $zipFile['file2'] = 'content 2';
  345. $zipFile->saveAsFile($this->outputFilename);
  346. $zipFile->close();
  347. $zipFile = new ZipFile();
  348. $zipFile->openFile($this->outputFilename);
  349. $zipFile->rename('file2', 'file');
  350. }
  351. /**
  352. * @expectedException \PhpZip\Exception\ZipNotFoundEntry
  353. * @expectedExceptionMessage Not found entry
  354. */
  355. public function testRenameEntryNotFound()
  356. {
  357. $zipFile = new ZipFile();
  358. $zipFile['file'] = 'content';
  359. $zipFile['file2'] = 'content 2';
  360. $zipFile->saveAsFile($this->outputFilename);
  361. $zipFile->close();
  362. $zipFile = new ZipFile();
  363. $zipFile->openFile($this->outputFilename);
  364. $zipFile->rename('file2.bak', 'file3');
  365. }
  366. /**
  367. * Delete entry from name.
  368. */
  369. public function testDeleteFromName()
  370. {
  371. $inputDir = dirname(dirname(__DIR__)) . DIRECTORY_SEPARATOR;
  372. $deleteEntryName = 'composer.json';
  373. $zipFile = new ZipFile();
  374. $zipFile->addDir($inputDir);
  375. $zipFile->saveAsFile($this->outputFilename);
  376. $zipFile->close();
  377. self::assertCorrectZipArchive($this->outputFilename);
  378. $zipFile->openFile($this->outputFilename);
  379. $zipFile->deleteFromName($deleteEntryName);
  380. $zipFile->saveAsFile($this->outputFilename);
  381. $zipFile->close();
  382. self::assertCorrectZipArchive($this->outputFilename);
  383. $zipFile->openFile($this->outputFilename);
  384. self::assertFalse(isset($zipFile[$deleteEntryName]));
  385. $zipFile->close();
  386. }
  387. public function testDeleteNewEntry(){
  388. $zipFile = new ZipFile();
  389. $zipFile['entry1'] = '';
  390. $zipFile['entry2'] = '';
  391. $zipFile->deleteFromName('entry2');
  392. $zipFile->saveAsFile($this->outputFilename);
  393. $zipFile->close();
  394. $zipFile->openFile($this->outputFilename);
  395. self::assertEquals(sizeof($zipFile), 1);
  396. self::assertTrue(isset($zipFile['entry1']));
  397. self::assertFalse(isset($zipFile['entry2']));
  398. $zipFile->close();
  399. }
  400. /**
  401. * @expectedException \PhpZip\Exception\ZipNotFoundEntry
  402. * @expectedExceptionMessage Not found entry entry
  403. */
  404. public function testDeleteFromNameNotFoundEntry(){
  405. $zipFile = new ZipFile();
  406. $zipFile->deleteFromName('entry');
  407. }
  408. /**
  409. * Delete zip entries from glob pattern
  410. */
  411. public function testDeleteFromGlob()
  412. {
  413. $inputDir = dirname(dirname(__DIR__));
  414. $zipFile = new ZipFile();
  415. $zipFile->addFilesFromGlobRecursive($inputDir, '**.{php,xml,json}', '/');
  416. $zipFile->saveAsFile($this->outputFilename);
  417. $zipFile->close();
  418. self::assertCorrectZipArchive($this->outputFilename);
  419. $zipFile->openFile($this->outputFilename);
  420. $zipFile->deleteFromGlob('**.{xml,json}');
  421. $zipFile->saveAsFile($this->outputFilename);
  422. $zipFile->close();
  423. self::assertCorrectZipArchive($this->outputFilename);
  424. $zipFile->openFile($this->outputFilename);
  425. self::assertFalse(isset($zipFile['composer.json']));
  426. self::assertFalse(isset($zipFile['bootstrap.xml']));
  427. $zipFile->close();
  428. }
  429. /**
  430. * @expectedException \PhpZip\Exception\InvalidArgumentException
  431. * @expectedExceptionMessage Glob pattern is empty
  432. */
  433. public function testDeleteFromGlobFailNull()
  434. {
  435. $zipFile = new ZipFile();
  436. $zipFile->deleteFromGlob(null);
  437. }
  438. /**
  439. * @expectedException \PhpZip\Exception\InvalidArgumentException
  440. * @expectedExceptionMessage Glob pattern is empty
  441. */
  442. public function testDeleteFromGlobFailEmpty()
  443. {
  444. $zipFile = new ZipFile();
  445. $zipFile->deleteFromGlob('');
  446. }
  447. /**
  448. * Delete entries from regex pattern
  449. */
  450. public function testDeleteFromRegex()
  451. {
  452. $inputDir = dirname(dirname(__DIR__));
  453. $zipFile = new ZipFile();
  454. $zipFile->addFilesFromRegexRecursive($inputDir, '~\.(xml|php|json)$~i', 'Path');
  455. $zipFile->saveAsFile($this->outputFilename);
  456. $zipFile->close();
  457. self::assertCorrectZipArchive($this->outputFilename);
  458. $zipFile->openFile($this->outputFilename);
  459. $zipFile->deleteFromRegex('~\.(json)$~i');
  460. $zipFile->addFromString('test.txt', 'content');
  461. $zipFile->deleteFromRegex('~\.txt$~');
  462. $zipFile->saveAsFile($this->outputFilename);
  463. $zipFile->close();
  464. self::assertCorrectZipArchive($this->outputFilename);
  465. $zipFile->openFile($this->outputFilename);
  466. self::assertFalse(isset($zipFile['Path/composer.json']));
  467. self::assertFalse(isset($zipFile['Path/test.txt']));
  468. self::assertTrue(isset($zipFile['Path/bootstrap.xml']));
  469. $zipFile->close();
  470. }
  471. /**
  472. * @expectedException \PhpZip\Exception\InvalidArgumentException
  473. * @expectedExceptionMessage Regex pattern is empty.
  474. */
  475. public function testDeleteFromRegexFailNull()
  476. {
  477. $zipFile = new ZipFile();
  478. $zipFile->deleteFromRegex(null);
  479. }
  480. /**
  481. * @expectedException \PhpZip\Exception\InvalidArgumentException
  482. * @expectedExceptionMessage Regex pattern is empty.
  483. */
  484. public function testDeleteFromRegexFailEmpty()
  485. {
  486. $zipFile = new ZipFile();
  487. $zipFile->deleteFromRegex('');
  488. }
  489. /**
  490. * Delete all entries
  491. */
  492. public function testDeleteAll()
  493. {
  494. $zipFile = new ZipFile();
  495. $zipFile->addDirRecursive(__DIR__);
  496. $zipFile->saveAsFile($this->outputFilename);
  497. $zipFile->close();
  498. self::assertCorrectZipArchive($this->outputFilename);
  499. $zipFile->openFile($this->outputFilename);
  500. self::assertTrue($zipFile->count() > 0);
  501. $zipFile->deleteAll();
  502. $zipFile->saveAsFile($this->outputFilename);
  503. $zipFile->close();
  504. self::assertCorrectEmptyZip($this->outputFilename);
  505. $zipFile->openFile($this->outputFilename);
  506. self::assertEquals($zipFile->count(), 0);
  507. $zipFile->close();
  508. }
  509. /**
  510. * Test zip archive comment.
  511. */
  512. public function testArchiveComment()
  513. {
  514. $comment = "This zip file comment" . PHP_EOL
  515. . "Αυτό το σχόλιο αρχείο zip" . PHP_EOL
  516. . "Это комментарий zip архива" . PHP_EOL
  517. . "這個ZIP文件註釋" . PHP_EOL
  518. . "ეს zip ფაილის კომენტარი" . PHP_EOL
  519. . "このzipファイルにコメント" . PHP_EOL
  520. . "ความคิดเห็นนี้ไฟล์ซิป";
  521. $zipFile = new ZipFile();
  522. $zipFile->setArchiveComment($comment);
  523. $zipFile->addFile(__FILE__);
  524. $zipFile->saveAsFile($this->outputFilename);
  525. $zipFile->close();
  526. self::assertCorrectZipArchive($this->outputFilename);
  527. $zipFile->openFile($this->outputFilename);
  528. self::assertEquals($zipFile->getArchiveComment(), $comment);
  529. $zipFile->setArchiveComment(null); // remove archive comment
  530. $zipFile->saveAsFile($this->outputFilename);
  531. $zipFile->close();
  532. self::assertCorrectZipArchive($this->outputFilename);
  533. // check empty comment
  534. $zipFile->openFile($this->outputFilename);
  535. self::assertEquals($zipFile->getArchiveComment(), "");
  536. $zipFile->close();
  537. }
  538. /**
  539. * Test very long archive comment.
  540. *
  541. * @expectedException \PhpZip\Exception\InvalidArgumentException
  542. */
  543. public function testVeryLongArchiveComment()
  544. {
  545. $comment = "Very long comment" . PHP_EOL .
  546. "Очень длинный комментарий" . PHP_EOL;
  547. $comment = str_repeat($comment, ceil(0xffff / strlen($comment)) + strlen($comment) + 1);
  548. $zipFile = new ZipFile();
  549. $zipFile->setArchiveComment($comment);
  550. }
  551. /**
  552. * Test zip entry comment.
  553. */
  554. public function testEntryComment()
  555. {
  556. $entries = [
  557. '文件1.txt' => [
  558. 'data' => CryptoUtil::randomBytes(255),
  559. 'comment' => "這是註釋的條目。",
  560. ],
  561. 'file2.txt' => [
  562. 'data' => CryptoUtil::randomBytes(255),
  563. 'comment' => null
  564. ],
  565. 'file3.txt' => [
  566. 'data' => CryptoUtil::randomBytes(255),
  567. 'comment' => CryptoUtil::randomBytes(255),
  568. ],
  569. 'file4.txt' => [
  570. 'data' => CryptoUtil::randomBytes(255),
  571. 'comment' => "Комментарий файла"
  572. ],
  573. 'file5.txt' => [
  574. 'data' => CryptoUtil::randomBytes(255),
  575. 'comment' => "ไฟล์แสดงความคิดเห็น"
  576. ],
  577. 'file6 emoji 🙍🏼.txt' => [
  578. 'data' => CryptoUtil::randomBytes(255),
  579. 'comment' => "Emoji comment file - 😀 ⛈ ❤️ 🤴🏽"
  580. ],
  581. ];
  582. // create archive with entry comments
  583. $zipFile = new ZipFile();
  584. foreach ($entries as $entryName => $item) {
  585. $zipFile->addFromString($entryName, $item['data']);
  586. $zipFile->setEntryComment($entryName, $item['comment']);
  587. }
  588. $zipFile->saveAsFile($this->outputFilename);
  589. $zipFile->close();
  590. self::assertCorrectZipArchive($this->outputFilename);
  591. // check and modify comments
  592. $zipFile->openFile($this->outputFilename);
  593. foreach ($zipFile->getListFiles() as $entryName) {
  594. $entriesItem = $entries[$entryName];
  595. self::assertNotEmpty($entriesItem);
  596. self::assertEquals($zipFile[$entryName], $entriesItem['data']);
  597. self::assertEquals($zipFile->getEntryComment($entryName), (string)$entriesItem['comment']);
  598. }
  599. // modify comment
  600. $entries['file5.txt']['comment'] = mt_rand(1, 100000000);
  601. $zipFile->setEntryComment('file5.txt', $entries['file5.txt']['comment']);
  602. $zipFile->saveAsFile($this->outputFilename);
  603. $zipFile->close();
  604. self::assertCorrectZipArchive($this->outputFilename);
  605. // check modify comments
  606. $zipFile->openFile($this->outputFilename);
  607. foreach ($entries as $entryName => $entriesItem) {
  608. self::assertTrue(isset($zipFile[$entryName]));
  609. self::assertEquals($zipFile->getEntryComment($entryName), (string)$entriesItem['comment']);
  610. self::assertEquals($zipFile[$entryName], $entriesItem['data']);
  611. }
  612. $zipFile->close();
  613. }
  614. /**
  615. * Test zip entry very long comment.
  616. *
  617. * @expectedException \PhpZip\Exception\ZipException
  618. * @expectedExceptionMessage Comment too long
  619. */
  620. public function testVeryLongEntryComment()
  621. {
  622. $comment = "Very long comment" . PHP_EOL .
  623. "Очень длинный комментарий" . PHP_EOL;
  624. $comment = str_repeat($comment, ceil(0xffff / strlen($comment)) + strlen($comment) + 1);
  625. $zipFile = new ZipFile();
  626. $zipFile->addFile(__FILE__, 'test');
  627. $zipFile->setEntryComment('test', $comment);
  628. }
  629. /**
  630. * @expectedException \PhpZip\Exception\ZipException
  631. * @expectedExceptionMessage Not found entry
  632. */
  633. public function testSetEntryCommentNotFoundEntry()
  634. {
  635. $zipFile = new ZipFile();
  636. $zipFile->setEntryComment('test', 'comment');
  637. }
  638. /**
  639. * Test all available support compression methods.
  640. */
  641. public function testCompressionMethod()
  642. {
  643. $entries = [
  644. '1' => [
  645. 'data' => CryptoUtil::randomBytes(255),
  646. 'method' => ZipFile::METHOD_STORED,
  647. 'expected' => 'No compression',
  648. ],
  649. '2' => [
  650. 'data' => CryptoUtil::randomBytes(255),
  651. 'method' => ZipFile::METHOD_DEFLATED,
  652. 'expected' => 'Deflate',
  653. ],
  654. ];
  655. if (extension_loaded("bz2")) {
  656. $entries['3'] = [
  657. 'data' => CryptoUtil::randomBytes(255),
  658. 'method' => ZipFile::METHOD_BZIP2,
  659. 'expected' => 'Bzip2',
  660. ];
  661. }
  662. $zipFile = new ZipFile();
  663. foreach ($entries as $entryName => $item) {
  664. $zipFile->addFromString($entryName, $item['data'], $item['method']);
  665. }
  666. $zipFile->saveAsFile($this->outputFilename);
  667. $zipFile->close();
  668. self::assertCorrectZipArchive($this->outputFilename);
  669. $zipFile->openFile($this->outputFilename);
  670. $zipFile->setCompressionLevel(ZipFile::LEVEL_BEST_COMPRESSION);
  671. $zipAllInfo = $zipFile->getAllInfo();
  672. foreach ($zipAllInfo as $entryName => $info) {
  673. self::assertEquals($zipFile[$entryName], $entries[$entryName]['data']);
  674. self::assertEquals($info->getMethod(), $entries[$entryName]['expected']);
  675. $entryInfo = $zipFile->getEntryInfo($entryName);
  676. self::assertEquals($entryInfo, $info);
  677. }
  678. $zipFile->close();
  679. }
  680. /**
  681. * @expectedException \PhpZip\Exception\InvalidArgumentException
  682. * @expectedExceptionMessage Invalid compression level. Minimum level -1. Maximum level 9
  683. */
  684. public function testSetInvalidCompressionLevel(){
  685. $zipFile = new ZipFile();
  686. $zipFile->setCompressionLevel(-2);
  687. }
  688. /**
  689. /**
  690. * @expectedException \PhpZip\Exception\InvalidArgumentException
  691. * @expectedExceptionMessage Invalid compression level. Minimum level -1. Maximum level 9
  692. */
  693. public function testSetInvalidCompressionLevel2(){
  694. $zipFile = new ZipFile();
  695. $zipFile->setCompressionLevel(10);
  696. }
  697. /**
  698. * Test extract all files.
  699. */
  700. public function testExtract()
  701. {
  702. $entries = [
  703. 'test1.txt' => CryptoUtil::randomBytes(255),
  704. 'test2.txt' => CryptoUtil::randomBytes(255),
  705. 'test/test 2/test3.txt' => CryptoUtil::randomBytes(255),
  706. 'test empty/dir' => null,
  707. ];
  708. $extractPath = sys_get_temp_dir() . '/zipExtract' . uniqid();
  709. if (!is_dir($extractPath)) {
  710. mkdir($extractPath, 0755, true);
  711. }
  712. $zipFile = new ZipFile();
  713. foreach ($entries as $entryName => $value) {
  714. if ($value === null) {
  715. $zipFile->addEmptyDir($entryName);
  716. } else {
  717. $zipFile->addFromString($entryName, $value);
  718. }
  719. }
  720. $zipFile->saveAsFile($this->outputFilename);
  721. $zipFile->close();
  722. $zipFile->openFile($this->outputFilename);
  723. $zipFile->extractTo($extractPath);
  724. foreach ($entries as $entryName => $value) {
  725. $fullExtractedFilename = $extractPath . DIRECTORY_SEPARATOR . $entryName;
  726. if ($value === null) {
  727. self::assertTrue(is_dir($fullExtractedFilename));
  728. self::assertTrue(FilesUtil::isEmptyDir($fullExtractedFilename));
  729. } else {
  730. self::assertTrue(is_file($fullExtractedFilename));
  731. $contents = file_get_contents($fullExtractedFilename);
  732. self::assertEquals($contents, $value);
  733. }
  734. }
  735. $zipFile->close();
  736. FilesUtil::removeDir($extractPath);
  737. }
  738. /**
  739. * Test extract some files
  740. */
  741. public function testExtractSomeFiles()
  742. {
  743. $entries = [
  744. 'test1.txt' => CryptoUtil::randomBytes(255),
  745. 'test2.txt' => CryptoUtil::randomBytes(255),
  746. 'test3.txt' => CryptoUtil::randomBytes(255),
  747. 'test4.txt' => CryptoUtil::randomBytes(255),
  748. 'test5.txt' => CryptoUtil::randomBytes(255),
  749. 'test/test/test.txt' => CryptoUtil::randomBytes(255),
  750. 'test/test/test 2.txt' => CryptoUtil::randomBytes(255),
  751. 'test empty/dir/' => null,
  752. 'test empty/dir2/' => null,
  753. ];
  754. $extractEntries = [
  755. 'test1.txt',
  756. 'test3.txt',
  757. 'test5.txt',
  758. 'test/test/test 2.txt',
  759. 'test empty/dir2/'
  760. ];
  761. $extractPath = sys_get_temp_dir() . '/zipExtractTest';
  762. if (is_dir($extractPath)) {
  763. FilesUtil::removeDir($extractPath);
  764. }
  765. self::assertTrue(mkdir($extractPath, 0755, true));
  766. $zipFile = new ZipFile();
  767. $zipFile->addAll($entries);
  768. $zipFile->saveAsFile($this->outputFilename);
  769. $zipFile->close();
  770. $zipFile->openFile($this->outputFilename);
  771. $zipFile->extractTo($extractPath, $extractEntries);
  772. foreach ($entries as $entryName => $value) {
  773. $fullExtractFilename = $extractPath . DIRECTORY_SEPARATOR . $entryName;
  774. if (in_array($entryName, $extractEntries)) {
  775. if ($value === null) {
  776. self::assertTrue(is_dir($fullExtractFilename));
  777. self::assertTrue(FilesUtil::isEmptyDir($fullExtractFilename));
  778. } else {
  779. self::assertTrue(is_file($fullExtractFilename));
  780. $contents = file_get_contents($fullExtractFilename);
  781. self::assertEquals($contents, $value);
  782. }
  783. } else {
  784. if ($value === null) {
  785. self::assertFalse(is_dir($fullExtractFilename));
  786. } else {
  787. self::assertFalse(is_file($fullExtractFilename));
  788. }
  789. }
  790. }
  791. self::assertFalse(is_file($extractPath . DIRECTORY_SEPARATOR . 'test/test/test.txt'));
  792. $zipFile->extractTo($extractPath, 'test/test/test.txt');
  793. self::assertTrue(is_file($extractPath . DIRECTORY_SEPARATOR . 'test/test/test.txt'));
  794. $zipFile->close();
  795. FilesUtil::removeDir($extractPath);
  796. }
  797. /**
  798. * @expectedException \PhpZip\Exception\ZipException
  799. * @expectedExceptionMessage not found
  800. */
  801. public function testExtractFail()
  802. {
  803. $zipFile = new ZipFile();
  804. $zipFile['file'] = 'content';
  805. $zipFile->saveAsFile($this->outputFilename);
  806. $zipFile->close();
  807. $zipFile->openFile($this->outputFilename);
  808. $zipFile->extractTo('path/to/path');
  809. }
  810. /**
  811. * @expectedException \PhpZip\Exception\ZipException
  812. * @expectedExceptionMessage Destination is not directory
  813. */
  814. public function testExtractFail2()
  815. {
  816. $zipFile = new ZipFile();
  817. $zipFile['file'] = 'content';
  818. $zipFile->saveAsFile($this->outputFilename);
  819. $zipFile->close();
  820. $zipFile->openFile($this->outputFilename);
  821. $zipFile->extractTo($this->outputFilename);
  822. }
  823. /**
  824. * @expectedException \PhpZip\Exception\ZipException
  825. * @expectedExceptionMessage Destination is not writable directory
  826. */
  827. public function testExtractFail3()
  828. {
  829. $zipFile = new ZipFile();
  830. $zipFile['file'] = 'content';
  831. $zipFile->saveAsFile($this->outputFilename);
  832. $zipFile->close();
  833. $extractPath = sys_get_temp_dir() . '/zipExtractTest';
  834. if (is_dir($extractPath)) {
  835. FilesUtil::removeDir($extractPath);
  836. }
  837. self::assertTrue(mkdir($extractPath, 0444, true));
  838. self::assertTrue(chmod($extractPath, 0444));
  839. $zipFile->openFile($this->outputFilename);
  840. $zipFile->extractTo($extractPath);
  841. }
  842. /**
  843. * Test archive password.
  844. */
  845. public function testSetPassword()
  846. {
  847. $password = base64_encode(CryptoUtil::randomBytes(100));
  848. $badPassword = "sdgt43r23wefe";
  849. // create encryption password with ZipCrypto
  850. $zipFile = new ZipFile();
  851. $zipFile->addDirRecursive(__DIR__);
  852. $zipFile->withNewPassword($password, ZipFile::ENCRYPTION_METHOD_TRADITIONAL);
  853. $zipFile->saveAsFile($this->outputFilename);
  854. $zipFile->close();
  855. self::assertCorrectZipArchive($this->outputFilename, $password);
  856. // check bad password for ZipCrypto
  857. $zipFile->openFile($this->outputFilename);
  858. $zipFile->withReadPassword($badPassword);
  859. foreach ($zipFile->getListFiles() as $entryName) {
  860. try {
  861. $zipFile[$entryName];
  862. self::fail("Expected Exception has not been raised.");
  863. } catch (ZipAuthenticationException $ae) {
  864. self::assertNotNull($ae);
  865. }
  866. }
  867. // check correct password for ZipCrypto
  868. $zipFile->withReadPassword($password);
  869. foreach ($zipFile->getAllInfo() as $info) {
  870. self::assertTrue($info->isEncrypted());
  871. self::assertContains('ZipCrypto', $info->getMethod());
  872. $decryptContent = $zipFile[$info->getPath()];
  873. self::assertNotEmpty($decryptContent);
  874. self::assertContains('<?php', $decryptContent);
  875. }
  876. // change encryption method to WinZip Aes and update file
  877. $zipFile->withNewPassword($password, ZipFile::ENCRYPTION_METHOD_WINZIP_AES);
  878. $zipFile->saveAsFile($this->outputFilename);
  879. $zipFile->close();
  880. self::assertCorrectZipArchive($this->outputFilename, $password);
  881. // check from WinZip AES encryption
  882. $zipFile->openFile($this->outputFilename);
  883. // set bad password WinZip AES
  884. $zipFile->withReadPassword($badPassword);
  885. foreach ($zipFile->getListFiles() as $entryName) {
  886. try {
  887. $zipFile[$entryName];
  888. self::fail("Expected Exception has not been raised.");
  889. } catch (ZipAuthenticationException $ae) {
  890. self::assertNotNull($ae);
  891. }
  892. }
  893. // set correct password WinZip AES
  894. $zipFile->withReadPassword($password);
  895. foreach ($zipFile->getAllInfo() as $info) {
  896. self::assertTrue($info->isEncrypted());
  897. self::assertContains('WinZip', $info->getMethod());
  898. $decryptContent = $zipFile[$info->getPath()];
  899. self::assertNotEmpty($decryptContent);
  900. self::assertContains('<?php', $decryptContent);
  901. }
  902. // clear password
  903. $zipFile->addFromString('file1', '');
  904. $zipFile->withoutPassword();
  905. $zipFile->addFromString('file2', '');
  906. $zipFile->saveAsFile($this->outputFilename);
  907. $zipFile->close();
  908. self::assertCorrectZipArchive($this->outputFilename);
  909. // check remove password
  910. $zipFile->openFile($this->outputFilename);
  911. foreach ($zipFile->getAllInfo() as $info) {
  912. self::assertFalse($info->isEncrypted());
  913. }
  914. $zipFile->close();
  915. }
  916. /**
  917. * @expectedException \PhpZip\Exception\ZipException
  918. * @expectedExceptionMessage Invalid encryption method
  919. */
  920. public function testSetEncryptionMethodInvalid(){
  921. $zipFile = new ZipFile();
  922. $encryptionMethod = 9999;
  923. $zipFile->withNewPassword('pass', $encryptionMethod);
  924. $zipFile['entry'] = 'content';
  925. $zipFile->outputAsString();
  926. }
  927. /**
  928. * @expectedException \PhpZip\Exception\InvalidArgumentException
  929. * @expectedExceptionMessage entryName is null
  930. */
  931. public function testAddFromArrayAccessNullName()
  932. {
  933. $zipFile = new ZipFile();
  934. $zipFile[null] = 'content';
  935. }
  936. /**
  937. * @expectedException \PhpZip\Exception\InvalidArgumentException
  938. * @expectedExceptionMessage entryName is empty
  939. */
  940. public function testAddFromArrayAccessEmptyName()
  941. {
  942. $zipFile = new ZipFile();
  943. $zipFile[''] = 'content';
  944. }
  945. /**
  946. * @expectedException \PhpZip\Exception\InvalidArgumentException
  947. * @expectedExceptionMessage Contents is null
  948. */
  949. public function testAddFromStringNullContents()
  950. {
  951. $zipFile = new ZipFile();
  952. $zipFile->addFromString('file', null);
  953. }
  954. /**
  955. * @expectedException \PhpZip\Exception\InvalidArgumentException
  956. * @expectedExceptionMessage Incorrect entry name
  957. */
  958. public function testAddFromStringNullEntryName()
  959. {
  960. $zipFile = new ZipFile();
  961. $zipFile->addFromString(null, 'contents');
  962. }
  963. /**
  964. * @expectedException \PhpZip\Exception\ZipUnsupportMethod
  965. * @expectedExceptionMessage Unsupported method
  966. */
  967. public function testAddFromStringUnsupportedMethod()
  968. {
  969. $zipFile = new ZipFile();
  970. $zipFile->addFromString('file', 'contents', ZipEntry::METHOD_WINZIP_AES);
  971. }
  972. /**
  973. * @expectedException \PhpZip\Exception\InvalidArgumentException
  974. * @expectedExceptionMessage Incorrect entry name
  975. */
  976. public function testAddFromStringEmptyEntryName()
  977. {
  978. $zipFile = new ZipFile();
  979. $zipFile->addFromString('', 'contents');
  980. }
  981. /**
  982. * Test compression method from add string.
  983. */
  984. public function testAddFromStringCompressionMethod()
  985. {
  986. $fileStored = sys_get_temp_dir() . '/zip-stored.txt';
  987. $fileDeflated = sys_get_temp_dir() . '/zip-deflated.txt';
  988. self::assertNotFalse(file_put_contents($fileStored, 'content'));
  989. self::assertNotFalse(file_put_contents($fileDeflated, str_repeat('content', 200)));
  990. $zipFile = new ZipFile();
  991. $zipFile->addFromString(basename($fileStored), file_get_contents($fileStored));
  992. $zipFile->addFromString(basename($fileDeflated), file_get_contents($fileDeflated));
  993. $zipFile->saveAsFile($this->outputFilename);
  994. $zipFile->close();
  995. unlink($fileStored);
  996. unlink($fileDeflated);
  997. $zipFile->openFile($this->outputFilename);
  998. $infoStored = $zipFile->getEntryInfo(basename($fileStored));
  999. $infoDeflated = $zipFile->getEntryInfo(basename($fileDeflated));
  1000. self::assertEquals($infoStored->getMethod(), 'No compression');
  1001. self::assertEquals($infoDeflated->getMethod(), 'Deflate');
  1002. $zipFile->close();
  1003. }
  1004. /**
  1005. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1006. * @expectedExceptionMessage stream is not resource
  1007. */
  1008. public function testAddFromStreamInvalidResource()
  1009. {
  1010. $zipFile = new ZipFile();
  1011. $zipFile->addFromStream("invalid resource", "name");
  1012. }
  1013. /**
  1014. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1015. * @expectedExceptionMessage Incorrect entry name
  1016. */
  1017. public function testAddFromStreamEmptyEntryName()
  1018. {
  1019. $handle = fopen(__FILE__, 'rb');
  1020. $zipFile = new ZipFile();
  1021. $zipFile->addFromStream($handle, "");
  1022. }
  1023. /**
  1024. * @expectedException \PhpZip\Exception\ZipUnsupportMethod
  1025. * @expectedExceptionMessage Unsupported method
  1026. */
  1027. public function testAddFromStreamUnsupportedMethod()
  1028. {
  1029. $handle = fopen(__FILE__, 'rb');
  1030. $zipFile = new ZipFile();
  1031. $zipFile->addFromStream($handle, basename(__FILE__), ZipEntry::METHOD_WINZIP_AES);
  1032. }
  1033. /**
  1034. * Test compression method from add stream.
  1035. */
  1036. public function testAddFromStreamCompressionMethod()
  1037. {
  1038. $fileStored = sys_get_temp_dir() . '/zip-stored.txt';
  1039. $fileDeflated = sys_get_temp_dir() . '/zip-deflated.txt';
  1040. self::assertNotFalse(file_put_contents($fileStored, 'content'));
  1041. self::assertNotFalse(file_put_contents($fileDeflated, str_repeat('content', 200)));
  1042. $fpStored = fopen($fileStored, 'rb');
  1043. $fpDeflated = fopen($fileDeflated, 'rb');
  1044. $zipFile = new ZipFile();
  1045. $zipFile->addFromStream($fpStored, basename($fileStored));
  1046. $zipFile->addFromStream($fpDeflated, basename($fileDeflated));
  1047. $zipFile->saveAsFile($this->outputFilename);
  1048. $zipFile->close();
  1049. unlink($fileStored);
  1050. unlink($fileDeflated);
  1051. $zipFile->openFile($this->outputFilename);
  1052. $infoStored = $zipFile->getEntryInfo(basename($fileStored));
  1053. $infoDeflated = $zipFile->getEntryInfo(basename($fileDeflated));
  1054. self::assertEquals($infoStored->getMethod(), 'No compression');
  1055. self::assertEquals($infoDeflated->getMethod(), 'Deflate');
  1056. $zipFile->close();
  1057. }
  1058. /**
  1059. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1060. * @expectedExceptionMessage Filename is null
  1061. */
  1062. public function testAddFileNullFileName()
  1063. {
  1064. $zipFile = new ZipFile();
  1065. $zipFile->addFile(null);
  1066. }
  1067. /**
  1068. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1069. * @expectedExceptionMessage is not exists
  1070. */
  1071. public function testAddFileCantExists()
  1072. {
  1073. $zipFile = new ZipFile();
  1074. $zipFile->addFile('path/to/file');
  1075. }
  1076. /**
  1077. * @expectedException \PhpZip\Exception\ZipUnsupportMethod
  1078. * @expectedExceptionMessage Unsupported method
  1079. */
  1080. public function testAddFileUnsupportedMethod()
  1081. {
  1082. $zipFile = new ZipFile();
  1083. $zipFile->addFile(__FILE__, null, ZipEntry::METHOD_WINZIP_AES);
  1084. }
  1085. /**
  1086. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1087. * @expectedExceptionMessage can not open
  1088. */
  1089. public function testAddFileCantOpen()
  1090. {
  1091. self::assertNotFalse(file_put_contents($this->outputFilename, ''));
  1092. self::assertTrue(chmod($this->outputFilename, 0244));
  1093. $zipFile = new ZipFile();
  1094. $zipFile->addFile($this->outputFilename);
  1095. }
  1096. /**
  1097. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1098. * @expectedExceptionMessage Input dir empty
  1099. */
  1100. public function testAddDirNullDirname()
  1101. {
  1102. $zipFile = new ZipFile();
  1103. $zipFile->addDir(null);
  1104. }
  1105. /**
  1106. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1107. * @expectedExceptionMessage Input dir empty
  1108. */
  1109. public function testAddDirEmptyDirname()
  1110. {
  1111. $zipFile = new ZipFile();
  1112. $zipFile->addDir("");
  1113. }
  1114. /**
  1115. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1116. * @expectedExceptionMessage can't exists
  1117. */
  1118. public function testAddDirCantExists()
  1119. {
  1120. $zipFile = new ZipFile();
  1121. $zipFile->addDir(uniqid());
  1122. }
  1123. /**
  1124. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1125. * @expectedExceptionMessage Input dir empty
  1126. */
  1127. public function testAddDirRecursiveNullDirname()
  1128. {
  1129. $zipFile = new ZipFile();
  1130. $zipFile->addDirRecursive(null);
  1131. }
  1132. /**
  1133. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1134. * @expectedExceptionMessage Input dir empty
  1135. */
  1136. public function testAddDirRecursiveEmptyDirname()
  1137. {
  1138. $zipFile = new ZipFile();
  1139. $zipFile->addDirRecursive("");
  1140. }
  1141. /**
  1142. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1143. * @expectedExceptionMessage can't exists
  1144. */
  1145. public function testAddDirRecursiveCantExists()
  1146. {
  1147. $zipFile = new ZipFile();
  1148. $zipFile->addDirRecursive(uniqid());
  1149. }
  1150. /**
  1151. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1152. * @expectedExceptionMessage Input dir empty
  1153. */
  1154. public function testAddFilesFromGlobNull()
  1155. {
  1156. $zipFile = new ZipFile();
  1157. $zipFile->addFilesFromGlob(null, '*.png');
  1158. }
  1159. /**
  1160. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1161. * @expectedExceptionMessage Input dir empty
  1162. */
  1163. public function testAddFilesFromGlobEmpty()
  1164. {
  1165. $zipFile = new ZipFile();
  1166. $zipFile->addFilesFromGlob("", '*.png');
  1167. }
  1168. /**
  1169. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1170. * @expectedExceptionMessage can't exists
  1171. */
  1172. public function testAddFilesFromGlobCantExists()
  1173. {
  1174. $zipFile = new ZipFile();
  1175. $zipFile->addFilesFromGlob("path/to/path", '*.png');
  1176. }
  1177. /**
  1178. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1179. * @expectedExceptionMessage glob pattern empty
  1180. */
  1181. public function testAddFilesFromGlobNullPattern()
  1182. {
  1183. $zipFile = new ZipFile();
  1184. $zipFile->addFilesFromGlob(__DIR__, null);
  1185. }
  1186. /**
  1187. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1188. * @expectedExceptionMessage glob pattern empty
  1189. */
  1190. public function testAddFilesFromGlobEmptyPattern()
  1191. {
  1192. $zipFile = new ZipFile();
  1193. $zipFile->addFilesFromGlob(__DIR__, '');
  1194. }
  1195. /**
  1196. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1197. * @expectedExceptionMessage Input dir empty
  1198. */
  1199. public function testAddFilesFromGlobRecursiveNull()
  1200. {
  1201. $zipFile = new ZipFile();
  1202. $zipFile->addFilesFromGlobRecursive(null, '*.png');
  1203. }
  1204. /**
  1205. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1206. * @expectedExceptionMessage Input dir empty
  1207. */
  1208. public function testAddFilesFromGlobRecursiveEmpty()
  1209. {
  1210. $zipFile = new ZipFile();
  1211. $zipFile->addFilesFromGlobRecursive("", '*.png');
  1212. }
  1213. /**
  1214. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1215. * @expectedExceptionMessage can't exists
  1216. */
  1217. public function testAddFilesFromGlobRecursiveCantExists()
  1218. {
  1219. $zipFile = new ZipFile();
  1220. $zipFile->addFilesFromGlobRecursive("path/to/path", '*.png');
  1221. }
  1222. /**
  1223. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1224. * @expectedExceptionMessage glob pattern empty
  1225. */
  1226. public function testAddFilesFromGlobRecursiveNullPattern()
  1227. {
  1228. $zipFile = new ZipFile();
  1229. $zipFile->addFilesFromGlobRecursive(__DIR__, null);
  1230. }
  1231. /**
  1232. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1233. * @expectedExceptionMessage glob pattern empty
  1234. */
  1235. public function testAddFilesFromGlobRecursiveEmptyPattern()
  1236. {
  1237. $zipFile = new ZipFile();
  1238. $zipFile->addFilesFromGlobRecursive(__DIR__, '');
  1239. }
  1240. /**
  1241. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1242. * @expectedExceptionMessage Input dir empty
  1243. */
  1244. public function testAddFilesFromRegexDirectoryNull()
  1245. {
  1246. $zipFile = new ZipFile();
  1247. $zipFile->addFilesFromRegex(null, '~\.png$~i');
  1248. }
  1249. /**
  1250. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1251. * @expectedExceptionMessage Input dir empty
  1252. */
  1253. public function testAddFilesFromRegexDirectoryEmpty()
  1254. {
  1255. $zipFile = new ZipFile();
  1256. $zipFile->addFilesFromRegex("", '~\.png$~i');
  1257. }
  1258. /**
  1259. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1260. * @expectedExceptionMessage can't exists
  1261. */
  1262. public function testAddFilesFromRegexCantExists()
  1263. {
  1264. $zipFile = new ZipFile();
  1265. $zipFile->addFilesFromRegex("path/to/path", '~\.png$~i');
  1266. }
  1267. /**
  1268. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1269. * @expectedExceptionMessage regex pattern empty
  1270. */
  1271. public function testAddFilesFromRegexNullPattern()
  1272. {
  1273. $zipFile = new ZipFile();
  1274. $zipFile->addFilesFromRegex(__DIR__, null);
  1275. }
  1276. /**
  1277. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1278. * @expectedExceptionMessage regex pattern empty
  1279. */
  1280. public function testAddFilesFromRegexEmptyPattern()
  1281. {
  1282. $zipFile = new ZipFile();
  1283. $zipFile->addFilesFromRegex(__DIR__, '');
  1284. }
  1285. /**
  1286. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1287. * @expectedExceptionMessage Input dir empty
  1288. */
  1289. public function testAddFilesFromRegexRecursiveDirectoryNull()
  1290. {
  1291. $zipFile = new ZipFile();
  1292. $zipFile->addFilesFromRegexRecursive(null, '~\.png$~i');
  1293. }
  1294. /**
  1295. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1296. * @expectedExceptionMessage Input dir empty
  1297. */
  1298. public function testAddFilesFromRegexRecursiveEmpty()
  1299. {
  1300. $zipFile = new ZipFile();
  1301. $zipFile->addFilesFromRegexRecursive("", '~\.png$~i');
  1302. }
  1303. /**
  1304. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1305. * @expectedExceptionMessage can't exists
  1306. */
  1307. public function testAddFilesFromRegexRecursiveCantExists()
  1308. {
  1309. $zipFile = new ZipFile();
  1310. $zipFile->addFilesFromGlobRecursive("path/to/path", '~\.png$~i');
  1311. }
  1312. /**
  1313. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1314. * @expectedExceptionMessage regex pattern empty
  1315. */
  1316. public function testAddFilesFromRegexRecursiveNullPattern()
  1317. {
  1318. $zipFile = new ZipFile();
  1319. $zipFile->addFilesFromRegexRecursive(__DIR__, null);
  1320. }
  1321. /**
  1322. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1323. * @expectedExceptionMessage regex pattern empty
  1324. */
  1325. public function testAddFilesFromRegexRecursiveEmptyPattern()
  1326. {
  1327. $zipFile = new ZipFile();
  1328. $zipFile->addFilesFromRegexRecursive(__DIR__, '');
  1329. }
  1330. /**
  1331. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1332. * @expectedExceptionMessage handle is not resource
  1333. */
  1334. public function testSaveAsStreamBadStream()
  1335. {
  1336. $zipFile = new ZipFile();
  1337. $zipFile->saveAsStream("bad stream");
  1338. }
  1339. /**
  1340. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1341. * @expectedExceptionMessage can not open from write
  1342. */
  1343. public function testSaveAsFileNotWritable()
  1344. {
  1345. $this->outputFilename = sys_get_temp_dir() . '/zipExtractTest';
  1346. if (is_dir($this->outputFilename)) {
  1347. FilesUtil::removeDir($this->outputFilename);
  1348. }
  1349. self::assertTrue(mkdir($this->outputFilename, 0444, true));
  1350. self::assertTrue(chmod($this->outputFilename, 0444));
  1351. $this->outputFilename .= '/' . uniqid() . '.zip';
  1352. $zipFile = new ZipFile();
  1353. $zipFile->saveAsFile($this->outputFilename);
  1354. }
  1355. /**
  1356. * Test `ZipFile` implemented \ArrayAccess, \Countable and |iterator.
  1357. */
  1358. public function testZipFileArrayAccessAndCountableAndIterator()
  1359. {
  1360. $files = [];
  1361. $numFiles = mt_rand(20, 100);
  1362. for ($i = 0; $i < $numFiles; $i++) {
  1363. $files['file' . $i . '.txt'] = CryptoUtil::randomBytes(255);
  1364. }
  1365. $methods = [ZipFile::METHOD_STORED, ZipFile::METHOD_DEFLATED];
  1366. if (extension_loaded("bz2")) {
  1367. $methods[] = ZipFile::METHOD_BZIP2;
  1368. }
  1369. $zipFile = new ZipFile();
  1370. $zipFile->setCompressionLevel(ZipFile::LEVEL_BEST_SPEED);
  1371. foreach ($files as $entryName => $content) {
  1372. $zipFile->addFromString($entryName, $content, $methods[array_rand($methods)]);
  1373. }
  1374. $zipFile->saveAsFile($this->outputFilename);
  1375. $zipFile->close();
  1376. self::assertCorrectZipArchive($this->outputFilename);
  1377. $zipFile->openFile($this->outputFilename);
  1378. // Test \Countable
  1379. self::assertEquals($zipFile->count(), $numFiles);
  1380. self::assertEquals(count($zipFile), $numFiles);
  1381. // Test \ArrayAccess
  1382. reset($files);
  1383. foreach ($zipFile as $entryName => $content) {
  1384. self::assertEquals($entryName, key($files));
  1385. self::assertEquals($content, current($files));
  1386. next($files);
  1387. }
  1388. // Test \Iterator
  1389. reset($files);
  1390. $iterator = new \ArrayIterator($zipFile);
  1391. $iterator->rewind();
  1392. while ($iterator->valid()) {
  1393. $key = $iterator->key();
  1394. $value = $iterator->current();
  1395. self::assertEquals($key, key($files));
  1396. self::assertEquals($value, current($files));
  1397. next($files);
  1398. $iterator->next();
  1399. }
  1400. $zipFile->close();
  1401. $zipFile = new ZipFile();
  1402. $zipFile['file1.txt'] = 'content 1';
  1403. $zipFile['dir/file2.txt'] = 'content 1';
  1404. $zipFile['dir/empty dir/'] = null;
  1405. $zipFile->saveAsFile($this->outputFilename);
  1406. $zipFile->close();
  1407. self::assertCorrectZipArchive($this->outputFilename);
  1408. $zipFile->openFile($this->outputFilename);
  1409. self::assertTrue(isset($zipFile['file1.txt']));
  1410. self::assertTrue(isset($zipFile['dir/file2.txt']));
  1411. self::assertTrue(isset($zipFile['dir/empty dir/']));
  1412. self::assertFalse(isset($zipFile['dir/empty dir/2/']));
  1413. $zipFile['dir/empty dir/2/'] = null;
  1414. unset($zipFile['dir/file2.txt'], $zipFile['dir/empty dir/']);
  1415. $zipFile->saveAsFile($this->outputFilename);
  1416. $zipFile->close();
  1417. self::assertCorrectZipArchive($this->outputFilename);
  1418. $zipFile->openFile($this->outputFilename);
  1419. self::assertTrue(isset($zipFile['file1.txt']));
  1420. self::assertFalse(isset($zipFile['dir/file2.txt']));
  1421. self::assertFalse(isset($zipFile['dir/empty dir/']));
  1422. self::assertTrue(isset($zipFile['dir/empty dir/2/']));
  1423. $zipFile->close();
  1424. }
  1425. public function testArrayAccessAddFile()
  1426. {
  1427. $entryName = 'path/to/file.dat';
  1428. $zipFile = new ZipFile();
  1429. $zipFile[$entryName] = new \SplFileInfo(__FILE__);
  1430. $zipFile->saveAsFile($this->outputFilename);
  1431. $zipFile->close();
  1432. self::assertCorrectZipArchive($this->outputFilename);
  1433. $zipFile->openFile($this->outputFilename);
  1434. self::assertEquals(sizeof($zipFile), 1);
  1435. self::assertTrue(isset($zipFile[$entryName]));
  1436. self::assertEquals($zipFile[$entryName], file_get_contents(__FILE__));
  1437. $zipFile->close();
  1438. }
  1439. /**
  1440. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1441. * @expectedExceptionMessage DirName empty
  1442. */
  1443. public function testAddEmptyDirNullName()
  1444. {
  1445. $zipFile = new ZipFile();
  1446. $zipFile->addEmptyDir(null);
  1447. }
  1448. /**
  1449. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1450. * @expectedExceptionMessage DirName empty
  1451. */
  1452. public function testAddEmptyDirEmptyName()
  1453. {
  1454. $zipFile = new ZipFile();
  1455. $zipFile->addEmptyDir("");
  1456. }
  1457. /**
  1458. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1459. * @expectedExceptionMessage Output filename is empty.
  1460. */
  1461. public function testOutputAsAttachmentNullName()
  1462. {
  1463. $zipFile = new ZipFile();
  1464. $zipFile->outputAsAttachment(null);
  1465. }
  1466. /**
  1467. * @expectedException \PhpZip\Exception\InvalidArgumentException
  1468. * @expectedExceptionMessage Output filename is empty.
  1469. */
  1470. public function testOutputAsAttachmentEmptyName()
  1471. {
  1472. $zipFile = new ZipFile();
  1473. $zipFile->outputAsAttachment('');
  1474. }
  1475. /**
  1476. * @expectedException \PhpZip\Exception\ZipNotFoundEntry
  1477. * @expectedExceptionMessage Zip entry bad entry name not found
  1478. */
  1479. public function testNotFoundEntry(){
  1480. $zipFile = new ZipFile();
  1481. $zipFile['bad entry name'];
  1482. }
  1483. /**
  1484. * Test rewrite input file.
  1485. */
  1486. public function testRewriteFile()
  1487. {
  1488. $zipFile = new ZipFile();
  1489. $zipFile['file'] = 'content';
  1490. $zipFile['file2'] = 'content2';
  1491. $zipFile->saveAsFile($this->outputFilename);
  1492. $zipFile->close();
  1493. $md5file = md5_file($this->outputFilename);
  1494. $zipFile->openFile($this->outputFilename);
  1495. self::assertEquals(count($zipFile), 2);
  1496. self::assertTrue(isset($zipFile['file']));
  1497. self::assertTrue(isset($zipFile['file2']));
  1498. $zipFile['file3'] = 'content3';
  1499. self::assertEquals(count($zipFile), 2);
  1500. $zipFile = $zipFile->rewrite();
  1501. self::assertEquals(count($zipFile), 3);
  1502. self::assertTrue(isset($zipFile['file']));
  1503. self::assertTrue(isset($zipFile['file2']));
  1504. self::assertTrue(isset($zipFile['file3']));
  1505. $zipFile->close();
  1506. self::assertNotEquals(md5_file($this->outputFilename), $md5file);
  1507. }
  1508. /**
  1509. * Test rewrite for string.
  1510. */
  1511. public function testRewriteString()
  1512. {
  1513. $zipFile = new ZipFile();
  1514. $zipFile['file'] = 'content';
  1515. $zipFile['file2'] = 'content2';
  1516. $zipFile->saveAsFile($this->outputFilename);
  1517. $zipFile->close();
  1518. $zipFile->openFromString(file_get_contents($this->outputFilename));
  1519. self::assertEquals(count($zipFile), 2);
  1520. self::assertTrue(isset($zipFile['file']));
  1521. self::assertTrue(isset($zipFile['file2']));
  1522. $zipFile['file3'] = 'content3';
  1523. $zipFile = $zipFile->rewrite();
  1524. self::assertEquals(count($zipFile), 3);
  1525. self::assertTrue(isset($zipFile['file']));
  1526. self::assertTrue(isset($zipFile['file2']));
  1527. self::assertTrue(isset($zipFile['file3']));
  1528. $zipFile->close();
  1529. }
  1530. /**
  1531. * @expectedException \PhpZip\Exception\ZipException
  1532. * @expectedExceptionMessage input stream is null
  1533. */
  1534. public function testRewriteNullStream(){
  1535. $zipFile = new ZipFile();
  1536. $zipFile->rewrite();
  1537. }
  1538. /**
  1539. * Test zip alignment.
  1540. */
  1541. public function testZipAlign()
  1542. {
  1543. $zipFile = new ZipFile();
  1544. for ($i = 0; $i < 100; $i++) {
  1545. $zipFile->addFromString(
  1546. 'entry' . $i . '.txt',
  1547. CryptoUtil::randomBytes(mt_rand(100, 4096)),
  1548. ZipFile::METHOD_STORED
  1549. );
  1550. }
  1551. $zipFile->saveAsFile($this->outputFilename);
  1552. $zipFile->close();
  1553. self::assertCorrectZipArchive($this->outputFilename);
  1554. $result = self::doZipAlignVerify($this->outputFilename);
  1555. if ($result === null) return; // zip align not installed
  1556. // check not zip align
  1557. self::assertFalse($result);
  1558. $zipFile->openFile($this->outputFilename);
  1559. $zipFile->setZipAlign(4);
  1560. $zipFile->saveAsFile($this->outputFilename);
  1561. $zipFile->close();
  1562. self::assertCorrectZipArchive($this->outputFilename);
  1563. $result = self::doZipAlignVerify($this->outputFilename, true);
  1564. self::assertNotNull($result);
  1565. // check zip align
  1566. self::assertTrue($result);
  1567. $zipFile = new ZipFile();
  1568. for ($i = 0; $i < 100; $i++) {
  1569. $zipFile->addFromString(
  1570. 'entry' . $i . '.txt',
  1571. CryptoUtil::randomBytes(mt_rand(100, 4096)),
  1572. ZipFile::METHOD_STORED
  1573. );
  1574. }
  1575. $zipFile->setZipAlign(4);
  1576. $zipFile->saveAsFile($this->outputFilename);
  1577. $zipFile->close();
  1578. self::assertCorrectZipArchive($this->outputFilename);
  1579. $result = self::doZipAlignVerify($this->outputFilename);
  1580. // check not zip align
  1581. self::assertTrue($result);
  1582. }
  1583. /**
  1584. * Test support ZIP64 ext (slow test - normal).
  1585. * Create > 65535 files in archive and open and extract to /dev/null.
  1586. */
  1587. public function testCreateAndOpenZip64Ext()
  1588. {
  1589. $countFiles = 0xffff + 1;
  1590. $zipFile = new ZipFile();
  1591. for ($i = 0; $i < $countFiles; $i++) {
  1592. $zipFile[$i . '.txt'] = $i;
  1593. }
  1594. $zipFile->saveAsFile($this->outputFilename);
  1595. $zipFile->close();
  1596. self::assertCorrectZipArchive($this->outputFilename);
  1597. $zipFile->openFile($this->outputFilename);
  1598. self::assertEquals($zipFile->count(), $countFiles);
  1599. foreach ($zipFile as $entry => $content) {
  1600. }
  1601. $zipFile->close();
  1602. }
  1603. }