PageRenderTime 40ms CodeModel.GetById 15ms RepoModel.GetById 0ms app.codeStats 0ms

/application/libraries/getid3/module.archive.gzip.php

https://github.com/libis/Flandrica
PHP | 296 lines | 177 code | 55 blank | 64 comment | 33 complexity | 2004b0e712030e80989c6812dfeb112e MD5 | raw file
Possible License(s): GPL-3.0, LGPL-2.1, Apache-2.0
  1. <?php
  2. // +----------------------------------------------------------------------+
  3. // | PHP version 5 |
  4. // +----------------------------------------------------------------------+
  5. // | Copyright (c) 2002-2009 James Heinrich, Allan Hansen |
  6. // +----------------------------------------------------------------------+
  7. // | This source file is subject to version 2 of the GPL license, |
  8. // | that is bundled with this package in the file license.txt and is |
  9. // | available through the world-wide-web at the following url: |
  10. // | http://www.gnu.org/copyleft/gpl.html |
  11. // +----------------------------------------------------------------------+
  12. // | getID3() - http://getid3.sourceforge.net or http://www.getid3.org |
  13. // +----------------------------------------------------------------------+
  14. // | Authors: James Heinrich <infoŘgetid3*org> |
  15. // | Allan Hansen <ahŘartemis*dk> |
  16. // +----------------------------------------------------------------------+
  17. // | module.archive.gzip.php |
  18. // | module for analyzing GZIP files |
  19. // | dependencies: PHP compiled with zlib support (optional) |
  20. // +----------------------------------------------------------------------+
  21. // | Module originally written by Mike Mozolin <teddybearŘmail*ru> |
  22. // +----------------------------------------------------------------------+
  23. //
  24. // $Id: module.archive.gzip.php,v 1.4 2006/12/04 16:00:35 ah Exp $
  25. class getid3_gzip extends getid3_handler
  26. {
  27. // public: Optional file list - disable for speed.
  28. public $option_gzip_parse_contents = true; // decode gzipped files, if possible, and parse recursively (.tar.gz for example)
  29. // Reads the gzip-file
  30. function Analyze() {
  31. $info = &$this->getid3->info;
  32. $info['fileformat'] = 'gzip';
  33. $start_length = 10;
  34. $unpack_header = 'a1id1/a1id2/a1cmethod/a1flags/a4mtime/a1xflags/a1os';
  35. //+---+---+---+---+---+---+---+---+---+---+
  36. //|ID1|ID2|CM |FLG| MTIME |XFL|OS |
  37. //+---+---+---+---+---+---+---+---+---+---+
  38. @fseek($this->getid3->fp, 0);
  39. $buffer = @fread($this->getid3->fp, $info['filesize']);
  40. $arr_members = explode("\x1F\x8B\x08", $buffer);
  41. while (true) {
  42. $is_wrong_members = false;
  43. $num_members = intval(count($arr_members));
  44. for ($i = 0; $i < $num_members; $i++) {
  45. if (strlen($arr_members[$i]) == 0) {
  46. continue;
  47. }
  48. $buf = "\x1F\x8B\x08".$arr_members[$i];
  49. $attr = unpack($unpack_header, substr($buf, 0, $start_length));
  50. if (!$this->get_os_type(ord($attr['os']))) {
  51. // Merge member with previous if wrong OS type
  52. $arr_members[$i - 1] .= $buf;
  53. $arr_members[$i] = '';
  54. $is_wrong_members = true;
  55. continue;
  56. }
  57. }
  58. if (!$is_wrong_members) {
  59. break;
  60. }
  61. }
  62. $fpointer = 0;
  63. $idx = 0;
  64. for ($i = 0; $i < $num_members; $i++) {
  65. if (strlen($arr_members[$i]) == 0) {
  66. continue;
  67. }
  68. $info_gzip_member_header_idx = &$info['gzip']['member_header'][++$idx];
  69. $buff = "\x1F\x8B\x08".$arr_members[$i];
  70. $attr = unpack($unpack_header, substr($buff, 0, $start_length));
  71. $info_gzip_member_header_idx['filemtime'] = getid3_lib::LittleEndian2Int($attr['mtime']);
  72. $info_gzip_member_header_idx['raw']['id1'] = ord($attr['cmethod']);
  73. $info_gzip_member_header_idx['raw']['id2'] = ord($attr['cmethod']);
  74. $info_gzip_member_header_idx['raw']['cmethod'] = ord($attr['cmethod']);
  75. $info_gzip_member_header_idx['raw']['os'] = ord($attr['os']);
  76. $info_gzip_member_header_idx['raw']['xflags'] = ord($attr['xflags']);
  77. $info_gzip_member_header_idx['raw']['flags'] = ord($attr['flags']);
  78. $info_gzip_member_header_idx['flags']['crc16'] = (bool) ($info_gzip_member_header_idx['raw']['flags'] & 0x02);
  79. $info_gzip_member_header_idx['flags']['extra'] = (bool) ($info_gzip_member_header_idx['raw']['flags'] & 0x04);
  80. $info_gzip_member_header_idx['flags']['filename'] = (bool) ($info_gzip_member_header_idx['raw']['flags'] & 0x08);
  81. $info_gzip_member_header_idx['flags']['comment'] = (bool) ($info_gzip_member_header_idx['raw']['flags'] & 0x10);
  82. $info_gzip_member_header_idx['compression'] = $this->get_xflag_type($info_gzip_member_header_idx['raw']['xflags']);
  83. $info_gzip_member_header_idx['os'] = $this->get_os_type($info_gzip_member_header_idx['raw']['os']);
  84. if (!$info_gzip_member_header_idx['os']) {
  85. $info['error'][] = 'Read error on gzip file';
  86. return false;
  87. }
  88. $fpointer = 10;
  89. $arr_xsubfield = array ();
  90. // bit 2 - FLG.FEXTRA
  91. //+---+---+=================================+
  92. //| XLEN |...XLEN bytes of "extra field"...|
  93. //+---+---+=================================+
  94. if ($info_gzip_member_header_idx['flags']['extra']) {
  95. $w_xlen = substr($buff, $fpointer, 2);
  96. $xlen = getid3_lib::LittleEndian2Int($w_xlen);
  97. $fpointer += 2;
  98. $info_gzip_member_header_idx['raw']['xfield'] = substr($buff, $fpointer, $xlen);
  99. // Extra SubFields
  100. //+---+---+---+---+==================================+
  101. //|SI1|SI2| LEN |... LEN bytes of subfield data ...|
  102. //+---+---+---+---+==================================+
  103. $idx = 0;
  104. while (true) {
  105. if ($idx >= $xlen) {
  106. break;
  107. }
  108. $si1 = ord(substr($buff, $fpointer + $idx++, 1));
  109. $si2 = ord(substr($buff, $fpointer + $idx++, 1));
  110. if (($si1 == 0x41) && ($si2 == 0x70)) {
  111. $w_xsublen = substr($buff, $fpointer+$idx, 2);
  112. $xsublen = getid3_lib::LittleEndian2Int($w_xsublen);
  113. $idx += 2;
  114. $arr_xsubfield[] = substr($buff, $fpointer+$idx, $xsublen);
  115. $idx += $xsublen;
  116. } else {
  117. break;
  118. }
  119. }
  120. $fpointer += $xlen;
  121. }
  122. // bit 3 - FLG.FNAME
  123. //+=========================================+
  124. //|...original file name, zero-terminated...|
  125. //+=========================================+
  126. // GZIP files may have only one file, with no filename, so assume original filename is current filename without .gz
  127. $info_gzip_member_header_idx['filename'] = preg_replace('#\.gz$#i', '', @$info['filename']);
  128. if ($info_gzip_member_header_idx['flags']['filename']) {
  129. while (true) {
  130. if (ord($buff[$fpointer]) == 0) {
  131. $fpointer++;
  132. break;
  133. }
  134. $info_gzip_member_header_idx['filename'] .= $buff[$fpointer];
  135. $fpointer++;
  136. }
  137. }
  138. // bit 4 - FLG.FCOMMENT
  139. //+===================================+
  140. //|...file comment, zero-terminated...|
  141. //+===================================+
  142. if ($info_gzip_member_header_idx['flags']['comment']) {
  143. while (true) {
  144. if (ord($buff[$fpointer]) == 0) {
  145. $fpointer++;
  146. break;
  147. }
  148. $info_gzip_member_header_idx['comment'] .= $buff[$fpointer];
  149. $fpointer++;
  150. }
  151. }
  152. // bit 1 - FLG.FHCRC
  153. //+---+---+
  154. //| CRC16 |
  155. //+---+---+
  156. if ($info_gzip_member_header_idx['flags']['crc16']) {
  157. $w_crc = substr($buff, $fpointer, 2);
  158. $info_gzip_member_header_idx['crc16'] = getid3_lib::LittleEndian2Int($w_crc);
  159. $fpointer += 2;
  160. }
  161. // bit 0 - FLG.FTEXT
  162. //if ($info_gzip_member_header_idx['raw']['flags'] & 0x01) {
  163. // Ignored...
  164. //}
  165. // bits 5, 6, 7 - reserved
  166. $info_gzip_member_header_idx['crc32'] = getid3_lib::LittleEndian2Int(substr($buff, strlen($buff) - 8, 4));
  167. $info_gzip_member_header_idx['filesize'] = getid3_lib::LittleEndian2Int(substr($buff, strlen($buff) - 4));
  168. if ($this->option_gzip_parse_contents) {
  169. // Try to inflate GZip
  170. if (!function_exists('gzinflate')) {
  171. $this->getid3->warning('PHP does not have zlib support - contents not parsed.');
  172. return true;
  173. }
  174. $csize = 0;
  175. $inflated = '';
  176. $chkcrc32 = '';
  177. $cdata = substr($buff, $fpointer);
  178. $cdata = substr($cdata, 0, strlen($cdata) - 8);
  179. $csize = strlen($cdata);
  180. $inflated = gzinflate($cdata);
  181. // Calculate CRC32 for inflated content
  182. $info_gzip_member_header_idx['crc32_valid'] = (bool) (sprintf('%u', crc32($inflated)) == $info_gzip_member_header_idx['crc32']);
  183. //// Analyse contents
  184. // write content to temp file
  185. if (($temp_file_name = tempnam((function_exists('sys_get_temp_dir') ? sys_get_temp_dir() : ini_get('upload_tmp_dir')), 'getID3')) === false) {
  186. throw new getid3_exception('Unable to create temporary file.');
  187. }
  188. if ($tmp = fopen($temp_file_name, 'wb')) {
  189. fwrite($tmp, $inflated);
  190. fclose($tmp);
  191. // clone getid3 - we want same settings
  192. $clone = clone $this->getid3;
  193. unset($clone->info);
  194. try {
  195. $clone->Analyze($temp_file_name);
  196. $info_gzip_member_header_idx['parsed_content'] = $clone->info;
  197. }
  198. catch (getid3_exception $e) {
  199. // unable to parse contents
  200. }
  201. unlink($temp_file_name);
  202. }
  203. // Unknown/unhandled format
  204. else {
  205. }
  206. }
  207. }
  208. return true;
  209. }
  210. // Converts the OS type
  211. public static function get_os_type($key) {
  212. static $os_type = array (
  213. '0' => 'FAT filesystem (MS-DOS, OS/2, NT/Win32)',
  214. '1' => 'Amiga',
  215. '2' => 'VMS (or OpenVMS)',
  216. '3' => 'Unix',
  217. '4' => 'VM/CMS',
  218. '5' => 'Atari TOS',
  219. '6' => 'HPFS filesystem (OS/2, NT)',
  220. '7' => 'Macintosh',
  221. '8' => 'Z-System',
  222. '9' => 'CP/M',
  223. '10' => 'TOPS-20',
  224. '11' => 'NTFS filesystem (NT)',
  225. '12' => 'QDOS',
  226. '13' => 'Acorn RISCOS',
  227. '255' => 'unknown'
  228. );
  229. return @$os_type[$key];
  230. }
  231. // Converts the eXtra FLags
  232. public static function get_xflag_type($key) {
  233. static $xflag_type = array (
  234. '0' => 'unknown',
  235. '2' => 'maximum compression',
  236. '4' => 'fastest algorithm'
  237. );
  238. return @$xflag_type[$key];
  239. }
  240. }
  241. ?>