zip.lib.php 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. <?php
  2. /* $Id: zip.lib.php,v 2.4 2004/11/03 13:56:52 garvinhicking Exp $ */
  3. // vim: expandtab sw=4 ts=4 sts=4:
  4. /**
  5. * Zip file creation class.
  6. * Makes zip files.
  7. *
  8. * Based on :
  9. *
  10. * http://www.zend.com/codex.php?id=535&single=1
  11. * By Eric Mueller <eric@themepark.com>
  12. *
  13. * http://www.zend.com/codex.php?id=470&single=1
  14. * by Denis125 <webmaster@atlant.ru>
  15. *
  16. * a patch from Peter Listiak <mlady@users.sourceforge.net> for last modified
  17. * date and time of the compressed file
  18. *
  19. * Official ZIP file format: http://www.pkware.com/appnote.txt
  20. *
  21. * @access public
  22. */
  23. class zipfile
  24. {
  25. /**
  26. * Array to store compressed data
  27. *
  28. * @var array $datasec
  29. */
  30. var $datasec = array();
  31. /**
  32. * Central directory
  33. *
  34. * @var array $ctrl_dir
  35. */
  36. var $ctrl_dir = array();
  37. /**
  38. * End of central directory record
  39. *
  40. * @var string $eof_ctrl_dir
  41. */
  42. var $eof_ctrl_dir = "\x50\x4b\x05\x06\x00\x00\x00\x00";
  43. /**
  44. * Last offset position
  45. *
  46. * @var integer $old_offset
  47. */
  48. var $old_offset = 0;
  49. /**
  50. * Converts an Unix timestamp to a four byte DOS date and time format (date
  51. * in high two bytes, time in low two bytes allowing magnitude comparison).
  52. *
  53. * @param integer the current Unix timestamp
  54. *
  55. * @return integer the current date in a four byte DOS format
  56. *
  57. * @access private
  58. */
  59. function unix2DosTime($unixtime = 0) {
  60. $timearray = ($unixtime == 0) ? getdate() : getdate($unixtime);
  61. if ($timearray['year'] < 1980) {
  62. $timearray['year'] = 1980;
  63. $timearray['mon'] = 1;
  64. $timearray['mday'] = 1;
  65. $timearray['hours'] = 0;
  66. $timearray['minutes'] = 0;
  67. $timearray['seconds'] = 0;
  68. } // end if
  69. return (($timearray['year'] - 1980) << 25) | ($timearray['mon'] << 21) | ($timearray['mday'] << 16) |
  70. ($timearray['hours'] << 11) | ($timearray['minutes'] << 5) | ($timearray['seconds'] >> 1);
  71. } // end of the 'unix2DosTime()' method
  72. /**
  73. * Adds "file" to archive
  74. *
  75. * @param string file contents
  76. * @param string name of the file in the archive (may contains the path)
  77. * @param integer the current timestamp
  78. *
  79. * @access public
  80. */
  81. function addFile($data, $name, $time = 0)
  82. {
  83. $name = str_replace('\\', '/', $name);
  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"; // ver needed to extract
  92. $fr .= "\x00\x00"; // gen purpose bit flag
  93. $fr .= "\x08\x00"; // compression method
  94. $fr .= $hexdtime; // last mod time and date
  95. // "local file header" segment
  96. $unc_len = strlen($data);
  97. $crc = crc32($data);
  98. $zdata = gzcompress($data);
  99. $zdata = substr(substr($zdata, 0, strlen($zdata) - 4), 2); // fix crc bug
  100. $c_len = strlen($zdata);
  101. $fr .= pack('V', $crc); // crc32
  102. $fr .= pack('V', $c_len); // compressed filesize
  103. $fr .= pack('V', $unc_len); // uncompressed filesize
  104. $fr .= pack('v', strlen($name)); // length of filename
  105. $fr .= pack('v', 0); // extra field length
  106. $fr .= $name;
  107. // "file data" segment
  108. $fr .= $zdata;
  109. // "data descriptor" segment (optional but necessary if archive is not
  110. // served as file)
  111. // nijel(2004-10-19): this seems not to be needed at all and causes
  112. // problems in some cases (bug #1037737)
  113. //$fr .= pack('V', $crc); // crc32
  114. //$fr .= pack('V', $c_len); // compressed filesize
  115. //$fr .= pack('V', $unc_len); // uncompressed filesize
  116. // add this entry to array
  117. $this -> datasec[] = $fr;
  118. // now add to central directory record
  119. $cdrec = "\x50\x4b\x01\x02";
  120. $cdrec .= "\x00\x00"; // version made by
  121. $cdrec .= "\x14\x00"; // version needed to extract
  122. $cdrec .= "\x00\x00"; // gen purpose bit flag
  123. $cdrec .= "\x08\x00"; // compression method
  124. $cdrec .= $hexdtime; // last mod time & date
  125. $cdrec .= pack('V', $crc); // crc32
  126. $cdrec .= pack('V', $c_len); // compressed filesize
  127. $cdrec .= pack('V', $unc_len); // uncompressed filesize
  128. $cdrec .= pack('v', strlen($name) ); // length of filename
  129. $cdrec .= pack('v', 0 ); // extra field length
  130. $cdrec .= pack('v', 0 ); // file comment length
  131. $cdrec .= pack('v', 0 ); // disk number start
  132. $cdrec .= pack('v', 0 ); // internal file attributes
  133. $cdrec .= pack('V', 32 ); // external file attributes - 'archive' bit set
  134. $cdrec .= pack('V', $this -> old_offset ); // relative offset of local header
  135. $this -> old_offset += strlen($fr);
  136. $cdrec .= $name;
  137. // optional extra field, file comment goes here
  138. // save to central directory
  139. $this -> ctrl_dir[] = $cdrec;
  140. } // end of the 'addFile()' method
  141. /**
  142. * Dumps out file
  143. *
  144. * @return string the zipped file
  145. *
  146. * @access public
  147. */
  148. function file()
  149. {
  150. $data = implode('', $this -> datasec);
  151. $ctrldir = implode('', $this -> ctrl_dir);
  152. return
  153. $data .
  154. $ctrldir .
  155. $this -> eof_ctrl_dir .
  156. pack('v', sizeof($this -> ctrl_dir)) . // total # of entries "on this disk"
  157. pack('v', sizeof($this -> ctrl_dir)) . // total # of entries overall
  158. pack('V', strlen($ctrldir)) . // size of central dir
  159. pack('V', strlen($data)) . // offset to start of central dir
  160. "\x00\x00"; // .zip file comment length
  161. } // end of the 'file()' method
  162. } // end of the 'zipfile' class
  163. ?>