Zip.class.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582
  1. <?php
  2. namespace Util;
  3. /**
  4. * 文件压缩解压类
  5. */
  6. class Zip
  7. {
  8. private $ctrl_dir = array();
  9. private $datasec = array();
  10. /**
  11. * author xtj
  12. * intro 码包压缩
  13. * param $zip 压缩包名 $shell_name 压缩的文件名
  14. * return array obj json bool string resource
  15. */
  16. public function shell_zip($zip_name, $file_name)
  17. {
  18. $pattern = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLOMNOPQRSTUVWXYZ';
  19. $pwd = '';
  20. for ($i = 0; $i < 4; $i++) {
  21. $pwd .= $pattern{mt_rand(0, 61)};//利用大括号截取字符串
  22. }
  23. $co = "zip -qj -P $pwd $zip_name $file_name";
  24. exec($co, $out);
  25. if ($out) {
  26. return $pwd;
  27. } else {
  28. return false;
  29. }
  30. }
  31. /**********************************************************
  32. * 压缩部分
  33. **********************************************************/
  34. // ------------------------------------------------------ //
  35. // #遍历指定文件夹
  36. //
  37. // $archive = new PHPZip();
  38. // $filelist = $archive->visitFile(文件夹路径);
  39. // print "当前文件夹的文件:<p>\r\n";
  40. // foreach($filelist as $file)
  41. // printf("%s<br>\r\n", $file);
  42. // ------------------------------------------------------ //
  43. public $fileList = array();
  44. public function visitFile($path)
  45. {
  46. global $fileList;
  47. $path = str_replace("\\", "/", $path);
  48. $fdir = dir($path);
  49. while (($file = $fdir->read()) !== false) {
  50. if ($file == '.' || $file == '..') {
  51. continue;
  52. }
  53. $pathSub = preg_replace("*/{2,}*", "/", $path . "/" . $file); // 替换多个反斜杠
  54. $fileList[] = is_dir($pathSub) ? $pathSub . "/" : $pathSub;
  55. if (is_dir($pathSub)) {
  56. $this->visitFile($pathSub);
  57. }
  58. }
  59. $fdir->close();
  60. return $fileList;
  61. }
  62. private function unix2DosTime($unixtime = 0)
  63. {
  64. $timearray = ($unixtime == 0) ? getdate() : getdate($unixtime);
  65. if ($timearray['year'] < 1980) {
  66. $timearray['year'] = 1980;
  67. $timearray['mon'] = 1;
  68. $timearray['mday'] = 1;
  69. $timearray['hours'] = 0;
  70. $timearray['minutes'] = 0;
  71. $timearray['seconds'] = 0;
  72. }
  73. return (($timearray['year'] - 1980) << 25)
  74. | ($timearray['mon'] << 21)
  75. | ($timearray['mday'] << 16)
  76. | ($timearray['hours'] << 11)
  77. | ($timearray['minutes'] << 5)
  78. | ($timearray['seconds'] >> 1);
  79. }
  80. public $old_offset = 0;
  81. private function addFile($data, $filename, $time = 0)
  82. {
  83. $filename = str_replace('\\', '/', $filename);
  84. $dtime = dechex($this->unix2DosTime($time));
  85. $hexdtime = '\x' . $dtime[6] . $dtime[7]
  86. . '\x' . $dtime[4] . $dtime[5]
  87. . '\x' . $dtime[2] . $dtime[3]
  88. . '\x' . $dtime[0] . $dtime[1];
  89. eval('$hexdtime = "' . $hexdtime . '";');
  90. $fr = "\x50\x4b\x03\x04";
  91. $fr .= "\x14\x00";
  92. $fr .= "\x00\x00";
  93. $fr .= "\x08\x00";
  94. $fr .= $hexdtime;
  95. $unc_len = strlen($data);
  96. $crc = crc32($data);
  97. $zdata = gzcompress($data);
  98. $c_len = strlen($zdata);
  99. $zdata = substr(substr($zdata, 0, strlen($zdata) - 4), 2);
  100. $fr .= pack('V', $crc);
  101. $fr .= pack('V', $c_len);
  102. $fr .= pack('V', $unc_len);
  103. $fr .= pack('v', strlen($filename));
  104. $fr .= pack('v', 0);
  105. $fr .= $filename;
  106. $fr .= $zdata;
  107. $fr .= pack('V', $crc);
  108. $fr .= pack('V', $c_len);
  109. $fr .= pack('V', $unc_len);
  110. $this->datasec[] = $fr;
  111. $new_offset = strlen(implode('', $this->datasec));
  112. $cdrec = "\x50\x4b\x01\x02";
  113. $cdrec .= "\x00\x00";
  114. $cdrec .= "\x14\x00";
  115. $cdrec .= "\x00\x00";
  116. $cdrec .= "\x08\x00";
  117. $cdrec .= $hexdtime;
  118. $cdrec .= pack('V', $crc);
  119. $cdrec .= pack('V', $c_len);
  120. $cdrec .= pack('V', $unc_len);
  121. $cdrec .= pack('v', strlen($filename));
  122. $cdrec .= pack('v', 0);
  123. $cdrec .= pack('v', 0);
  124. $cdrec .= pack('v', 0);
  125. $cdrec .= pack('v', 0);
  126. $cdrec .= pack('V', 32);
  127. $cdrec .= pack('V', $this->old_offset);
  128. $this->old_offset = $new_offset;
  129. $cdrec .= $filename;
  130. $this->ctrl_dir[] = $cdrec;
  131. }
  132. public $eof_ctrl_dir = "\x50\x4b\x05\x06\x00\x00\x00\x00";
  133. private function file()
  134. {
  135. $data = implode('', $this->datasec);
  136. $ctrldir = implode('', $this->ctrl_dir);
  137. return $data
  138. . $ctrldir
  139. . $this->eof_ctrl_dir
  140. . pack('v', sizeof($this->ctrl_dir))
  141. . pack('v', sizeof($this->ctrl_dir))
  142. . pack('V', strlen($ctrldir))
  143. . pack('V', strlen($data))
  144. . "\x00\x00";
  145. }
  146. // ------------------------------------------------------ //
  147. // #压缩到服务器
  148. //
  149. // $archive = new PHPZip();
  150. // $archive->Zip("需压缩的文件所在目录", "ZIP压缩文件名");
  151. // ------------------------------------------------------ //
  152. public function Zip($dir, $saveName)
  153. {
  154. if (@!function_exists('gzcompress')) {
  155. return;
  156. }
  157. ob_end_clean();
  158. $filelist = $this->visitFile($dir);
  159. if (count($filelist) == 0) {
  160. return;
  161. }
  162. foreach ($filelist as $file) {
  163. if (!file_exists($file) || !is_file($file)) {
  164. continue;
  165. }
  166. $fd = fopen($file, "rb");
  167. $content = @fread($fd, filesize($file));
  168. fclose($fd);
  169. // 1.删除$dir的字符(./folder/file.txt删除./folder/)
  170. // 2.如果存在/就删除(/file.txt删除/)
  171. $file = substr($file, strlen($dir));
  172. if (substr($file, 0, 1) == "\\" || substr($file, 0, 1) == "/") {
  173. $file = substr($file, 1);
  174. }
  175. $this->addFile($content, $file);
  176. }
  177. $out = $this->file();
  178. $fp = fopen($saveName, "wb");
  179. fwrite($fp, $out, strlen($out));
  180. fclose($fp);
  181. }
  182. // ------------------------------------------------------ //
  183. // #压缩并直接下载
  184. //
  185. // $archive = new PHPZip();
  186. // $archive->ZipAndDownload("需压缩的文件所在目录");
  187. // ------------------------------------------------------ //
  188. public function ZipAndDownload($dir,$filename)
  189. {
  190. if (@!function_exists('gzcompress')) {
  191. return;
  192. }
  193. ob_end_clean();
  194. $filelist = $this->visitFile($dir);
  195. if (count($filelist) == 0) {
  196. return;
  197. }
  198. foreach ($filelist as $file) {
  199. if (!file_exists($file) || !is_file($file)) {
  200. continue;
  201. }
  202. $fd = fopen($file, "rb");
  203. $content = @fread($fd, filesize($file));
  204. fclose($fd);
  205. // 1.删除$dir的字符(./folder/file.txt删除./folder/)
  206. // 2.如果存在/就删除(/file.txt删除/)
  207. $file = substr($file, strlen($dir));
  208. if (substr($file, 0, 1) == "\\" || substr($file, 0, 1) == "/") {
  209. $file = substr($file, 1);
  210. }
  211. $this->addFile($content, $file);
  212. }
  213. $out = $this->file();
  214. @header('Content-Encoding: none');
  215. @header('Content-Type: application/zip');
  216. @header('Content-Disposition: attachment ; filename='.$filename. '.zip');
  217. @header('Pragma: no-cache');
  218. @header('Expires: 0');
  219. print($out);
  220. }
  221. /**********************************************************
  222. * 解压部分
  223. **********************************************************/
  224. // ------------------------------------------------------ //
  225. // ReadCentralDir($zip, $zipfile)
  226. // $zip是经过@fopen($zipfile, 'rb')打开的
  227. // $zipfile是zip文件的路径
  228. // ------------------------------------------------------ //
  229. private function ReadCentralDir($zip, $zipfile)
  230. {
  231. $size = filesize($zipfile);
  232. $max_size = ($size < 277) ? $size : 277;
  233. @fseek($zip, $size - $max_size);
  234. $pos = ftell($zip);
  235. $bytes = 0x00000000;
  236. while ($pos < $size) {
  237. $byte = @fread($zip, 1);
  238. $bytes = ($bytes << 8) | Ord($byte);
  239. $pos++;
  240. if ($bytes == 0x504b0506) {
  241. break;
  242. }
  243. }
  244. $data = unpack('vdisk/vdisk_start/vdisk_entries/ventries/Vsize/Voffset/vcomment_size', fread($zip, 18));
  245. $centd['comment'] = ($data['comment_size'] != 0) ? fread($zip, $data['comment_size']) : ''; // 注释
  246. $centd['entries'] = $data['entries'];
  247. $centd['disk_entries'] = $data['disk_entries'];
  248. $centd['offset'] = $data['offset'];
  249. $centd['disk_start'] = $data['disk_start'];
  250. $centd['size'] = $data['size'];
  251. $centd['disk'] = $data['disk'];
  252. return $centd;
  253. }
  254. private function ReadCentralFileHeaders($zip)
  255. {
  256. $binary_data = fread($zip, 46);
  257. $header = unpack('vchkid/vid/vversion/vversion_extracted/vflag/vcompression/vmtime/vmdate/Vcrc/Vcompressed_size/Vsize/vfilename_len/vextra_len/vcomment_len/vdisk/vinternal/Vexternal/Voffset', $binary_data);
  258. $header['filename'] = ($header['filename_len'] != 0) ? fread($zip, $header['filename_len']) : '';
  259. $header['extra'] = ($header['extra_len'] != 0) ? fread($zip, $header['extra_len']) : '';
  260. $header['comment'] = ($header['comment_len'] != 0) ? fread($zip, $header['comment_len']) : '';
  261. if ($header['mdate'] && $header['mtime']) {
  262. $hour = ($header['mtime'] & 0xF800) >> 11;
  263. $minute = ($header['mtime'] & 0x07E0) >> 5;
  264. $seconde = ($header['mtime'] & 0x001F) * 2;
  265. $year = (($header['mdate'] & 0xFE00) >> 9) + 1980;
  266. $month = ($header['mdate'] & 0x01E0) >> 5;
  267. $day = $header['mdate'] & 0x001F;
  268. $header['mtime'] = mktime($hour, $minute, $seconde, $month, $day, $year);
  269. } else {
  270. $header['mtime'] = time();
  271. }
  272. $header['stored_filename'] = $header['filename'];
  273. $header['status'] = 'ok';
  274. if (substr($header['filename'], -1) == '/') {
  275. $header['external'] = 0x41FF0010;
  276. } // 判断是否文件夹
  277. return $header;
  278. }
  279. private function ReadFileHeader($zip)
  280. {
  281. $binary_data = fread($zip, 30);
  282. $data = unpack('vchk/vid/vversion/vflag/vcompression/vmtime/vmdate/Vcrc/Vcompressed_size/Vsize/vfilename_len/vextra_len', $binary_data);
  283. $header['filename'] = fread($zip, $data['filename_len']);
  284. $header['extra'] = ($data['extra_len'] != 0) ? fread($zip, $data['extra_len']) : '';
  285. $header['compression'] = $data['compression'];
  286. $header['size'] = $data['size'];
  287. $header['compressed_size'] = $data['compressed_size'];
  288. $header['crc'] = $data['crc'];
  289. $header['flag'] = $data['flag'];
  290. $header['mdate'] = $data['mdate'];
  291. $header['mtime'] = $data['mtime'];
  292. if ($header['mdate'] && $header['mtime']) {
  293. $hour = ($header['mtime'] & 0xF800) >> 11;
  294. $minute = ($header['mtime'] & 0x07E0) >> 5;
  295. $seconde = ($header['mtime'] & 0x001F) * 2;
  296. $year = (($header['mdate'] & 0xFE00) >> 9) + 1980;
  297. $month = ($header['mdate'] & 0x01E0) >> 5;
  298. $day = $header['mdate'] & 0x001F;
  299. $header['mtime'] = mktime($hour, $minute, $seconde, $month, $day, $year);
  300. } else {
  301. $header['mtime'] = time();
  302. }
  303. $header['stored_filename'] = $header['filename'];
  304. $header['status'] = "ok";
  305. return $header;
  306. }
  307. private function ExtractFile($header, $to, $zip)
  308. {
  309. $header = $this->readfileheader($zip);
  310. if (substr($to, -1) != "/") {
  311. $to .= "/";
  312. }
  313. if (!@is_dir($to)) {
  314. @mkdir($to, 0777);
  315. }
  316. $pth = explode("/", dirname($header['filename']));
  317. for ($i = 0; isset($pth[$i]); $i++) {
  318. if (!$pth[$i]) {
  319. continue;
  320. }
  321. $pthss .= $pth[$i] . "/";
  322. if (!is_dir($to . $pthss)) {
  323. @mkdir($to . $pthss, 0777);
  324. }
  325. }
  326. if (!($header['external'] == 0x41FF0010) && !($header['external'] == 16)) {
  327. if ($header['compression'] == 0) {
  328. $fp = @fopen($to . $header['filename'], 'wb');
  329. if (!$fp) {
  330. return (-1);
  331. }
  332. $size = $header['compressed_size'];
  333. while ($size != 0) {
  334. $read_size = ($size < 2048 ? $size : 2048);
  335. $buffer = fread($zip, $read_size);
  336. $binary_data = pack('a' . $read_size, $buffer);
  337. @fwrite($fp, $binary_data, $read_size);
  338. $size -= $read_size;
  339. }
  340. fclose($fp);
  341. touch($to . $header['filename'], $header['mtime']);
  342. } else {
  343. $fp = @fopen($to . $header['filename'] . '.gz', 'wb');
  344. if (!$fp) {
  345. return (-1);
  346. }
  347. $binary_data = pack('va1a1Va1a1', 0x8b1f, Chr($header['compression']), Chr(0x00), time(), Chr(0x00), Chr(3));
  348. fwrite($fp, $binary_data, 10);
  349. $size = $header['compressed_size'];
  350. while ($size != 0) {
  351. $read_size = ($size < 1024 ? $size : 1024);
  352. $buffer = fread($zip, $read_size);
  353. $binary_data = pack('a' . $read_size, $buffer);
  354. @fwrite($fp, $binary_data, $read_size);
  355. $size -= $read_size;
  356. }
  357. $binary_data = pack('VV', $header['crc'], $header['size']);
  358. fwrite($fp, $binary_data, 8);
  359. fclose($fp);
  360. $gzp = @gzopen($to . $header['filename'] . '.gz', 'rb') or die("Cette archive est compress!");
  361. if (!$gzp) {
  362. return (-2);
  363. }
  364. $fp = @fopen($to . $header['filename'], 'wb');
  365. if (!$fp) {
  366. return (-1);
  367. }
  368. $size = $header['size'];
  369. while ($size != 0) {
  370. $read_size = ($size < 2048 ? $size : 2048);
  371. $buffer = gzread($gzp, $read_size);
  372. $binary_data = pack('a' . $read_size, $buffer);
  373. @fwrite($fp, $binary_data, $read_size);
  374. $size -= $read_size;
  375. }
  376. fclose($fp);
  377. gzclose($gzp);
  378. touch($to . $header['filename'], $header['mtime']);
  379. @unlink($to . $header['filename'] . '.gz');
  380. }
  381. }
  382. return true;
  383. }
  384. // ------------------------------------------------------ //
  385. // #解压文件
  386. //
  387. // $archive = new PHPZip();
  388. // $zipfile = "ZIP压缩文件名";
  389. // $savepath = "解压缩目录名";
  390. // $zipfile = $unzipfile;
  391. // $savepath = $unziptarget;
  392. // $array = $archive->GetZipInnerFilesInfo($zipfile);
  393. // $filecount = 0;
  394. // $dircount = 0;
  395. // $failfiles = array();
  396. // set_time_limit(0); // 修改为不限制超时时间(默认为30秒)
  397. //
  398. // for($i=0; $i<count($array); $i++) {
  399. // if($array[$i][folder] == 0){
  400. // if($archive->unZip($zipfile, $savepath, $i) > 0){
  401. // $filecount++;
  402. // }else{
  403. // $failfiles[] = $array[$i][filename];
  404. // }
  405. // }else{
  406. // $dircount++;
  407. // }
  408. // }
  409. // set_time_limit(30);
  410. //printf("文件夹:%d&nbsp;&nbsp;&nbsp;&nbsp;解压文件:%d&nbsp;&nbsp;&nbsp;&nbsp;失败:%d<br>\r\n", $dircount, $filecount, count($failfiles));
  411. //if(count($failfiles) > 0){
  412. // foreach($failfiles as $file){
  413. // printf("&middot;%s<br>\r\n", $file);
  414. // }
  415. //}
  416. // ------------------------------------------------------ //
  417. public function unZip($zipfile, $to, $index = array(-1))
  418. {
  419. $ok = 0;
  420. $zip = @fopen($zipfile, 'rb');
  421. if (!$zip) {
  422. return (-1);
  423. }
  424. $cdir = $this->ReadCentralDir($zip, $zipfile);
  425. $pos_entry = $cdir['offset'];
  426. if (!is_array($index)) {
  427. $index = array($index);
  428. }
  429. for ($i = 0; $index[$i]; $i++) {
  430. if (intval($index[$i]) != $index[$i] || $index[$i] > $cdir['entries']) {
  431. return (-1);
  432. }
  433. }
  434. for ($i = 0; $i < $cdir['entries']; $i++) {
  435. @fseek($zip, $pos_entry);
  436. $header = $this->ReadCentralFileHeaders($zip);
  437. $header['index'] = $i;
  438. $pos_entry = ftell($zip);
  439. @rewind($zip);
  440. fseek($zip, $header['offset']);
  441. if (in_array("-1", $index) || in_array($i, $index)) {
  442. $stat[$header['filename']] = $this->ExtractFile($header, $to, $zip);
  443. }
  444. }
  445. fclose($zip);
  446. return $stat;
  447. }
  448. /**********************************************************
  449. * 其它部分
  450. **********************************************************/
  451. // ------------------------------------------------------ //
  452. // #获取被压缩文件的信息
  453. //
  454. // $archive = new PHPZip();
  455. // $array = $archive->GetZipInnerFilesInfo(ZIP压缩文件名);
  456. // for($i=0; $i<count($array); $i++) {
  457. // printf("<b>&middot;%s</b><br>\r\n", $array[$i][filename]);
  458. // foreach($array[$i] as $key => $value)
  459. // printf("%s => %s<br>\r\n", $key, $value);
  460. // print "\r\n<p>------------------------------------<p>\r\n\r\n";
  461. // }
  462. // ------------------------------------------------------ //
  463. public function GetZipInnerFilesInfo($zipfile)
  464. {
  465. $zip = @fopen($zipfile, 'rb');
  466. if (!$zip) {
  467. return (0);
  468. }
  469. $centd = $this->ReadCentralDir($zip, $zipfile);
  470. @rewind($zip);
  471. @fseek($zip, $centd['offset']);
  472. $ret = array();
  473. for ($i = 0; $i < $centd['entries']; $i++) {
  474. $header = $this->ReadCentralFileHeaders($zip);
  475. $header['index'] = $i;
  476. $info = array(
  477. 'filename' => $header['filename'], // 文件名
  478. 'stored_filename' => $header['stored_filename'], // 压缩后文件名
  479. 'size' => $header['size'], // 大小
  480. 'compressed_size' => $header['compressed_size'], // 压缩后大小
  481. 'crc' => strtoupper(dechex($header['crc'])), // CRC32
  482. 'mtime' => date("Y-m-d H:i:s", $header['mtime']), // 文件修改时间
  483. 'comment' => $header['comment'], // 注释
  484. 'folder' => ($header['external'] == 0x41FF0010 || $header['external'] == 16) ? 1 : 0, // 是否为文件夹
  485. 'index' => $header['index'], // 文件索引
  486. 'status' => $header['status'], // 状态
  487. );
  488. $ret[] = $info;
  489. unset($header);
  490. }
  491. fclose($zip);
  492. return $ret;
  493. }
  494. // ------------------------------------------------------ //
  495. // #获取压缩文件的注释
  496. //
  497. // $archive = new PHPZip();
  498. // echo $archive->GetZipComment(ZIP压缩文件名);
  499. // ------------------------------------------------------ //
  500. public function GetZipComment($zipfile)
  501. {
  502. $zip = @fopen($zipfile, 'rb');
  503. if (!$zip) {
  504. return (0);
  505. }
  506. $centd = $this->ReadCentralDir($zip, $zipfile);
  507. fclose($zip);
  508. return $centd[comment];
  509. }
  510. }