make.dangerous.tar.php.inc 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. <?php
  2. // stolen from PEAR2_Pyrus_Developer_Creator_Tar by Greg Beaver, the original author, for use in unit tests
  3. // this tarmaker makes a malicious tar with a header designed to overflow the buffer
  4. class danger_tarmaker
  5. {
  6. /**
  7. * Path to archive file
  8. *
  9. * @var string
  10. */
  11. protected $archive;
  12. /**
  13. * Temporary stream used for creating the archive
  14. *
  15. * @var stream
  16. */
  17. protected $tmp;
  18. protected $path;
  19. protected $compress;
  20. function __construct($path, $compress = 'zlib')
  21. {
  22. $this->compress = $compress;
  23. if ($compress === 'bz2' && !function_exists('bzopen')) {
  24. throw new PEAR2_Pyrus_Developer_Creator_Exception(
  25. 'bzip2 extension not available');
  26. }
  27. if ($compress === 'zlib' && !function_exists('gzopen')) {
  28. throw new PEAR2_Pyrus_Developer_Creator_Exception(
  29. 'zlib extension not available');
  30. }
  31. $this->path = $path;
  32. }
  33. /**
  34. * save a file inside this package
  35. *
  36. * This code is modified from Vincent Lascaux's File_Archive
  37. * package, which is licensed under the LGPL license.
  38. * @param string relative path within the package
  39. * @param string|resource file contents or open file handle
  40. */
  41. function addFile($path, $fileOrStream, $stat = null)
  42. {
  43. clearstatcache();
  44. if ($stat === null) {
  45. if (is_resource($fileOrStream)) {
  46. $stat = fstat($fileOrStream);
  47. } else {
  48. $stat = array(
  49. 'mode' => 0x8000 + 0644,
  50. 'uid' => 0,
  51. 'gid' => 0,
  52. 'size' => strlen($fileOrStream),
  53. 'mtime' => time(),
  54. );
  55. }
  56. }
  57. $link = null;
  58. if ($stat['mode'] & 0x4000) {
  59. $type = 5; // Directory
  60. } else if ($stat['mode'] & 0x8000) {
  61. $type = 0; // Regular
  62. } else if ($stat['mode'] & 0xA000) {
  63. $type = 1; // Link
  64. $link = @readlink($current);
  65. } else {
  66. $type = 9; // Unknown
  67. }
  68. $filePrefix = '';
  69. if (strlen($path) > 255) {
  70. throw new Exception(
  71. "$path is too long, must be 255 characters or less"
  72. );
  73. } else if (strlen($path) > 100) {
  74. $filePrefix = substr($path, 0, strlen($path)-100);
  75. $path = substr($path, -100);
  76. }
  77. $block = pack('a100a8a8a8a12A12',
  78. $path,
  79. '12345678', // have a mode that allows the name to overflow
  80. sprintf('%6s ',decoct($stat['uid'])),
  81. sprintf('%6s ',decoct($stat['gid'])),
  82. sprintf('%11s ',decoct($stat['size'])),
  83. sprintf('%11s ',decoct($stat['mtime']))
  84. );
  85. $blockend = pack('a1a100a6a2a32a32a8a8a155a12',
  86. $type,
  87. $link,
  88. 'ustar',
  89. '00',
  90. 'Pyrus',
  91. 'Pyrus',
  92. '',
  93. '',
  94. $filePrefix,
  95. '123456789abc'); // malicious block
  96. $checkheader = array_merge(str_split($block), str_split($blockend));
  97. if (!function_exists('_pear2tarchecksum')) {
  98. function _pear2tarchecksum($a, $b) {return $a + ord($b);}
  99. }
  100. $checksum = 256; // 8 * ord(' ');
  101. $checksum += array_reduce($checkheader, '_pear2tarchecksum');
  102. $checksum = pack('a8', sprintf('%6s ', decoct($checksum)));
  103. fwrite($this->tmp, (binary)$block . $checksum . $blockend, 512);
  104. if (is_resource($fileOrStream)) {
  105. stream_copy_to_stream($fileOrStream, $this->tmp);
  106. if ($stat['size'] % 512) {
  107. fwrite($this->tmp, (binary)str_repeat("\0", 512 - $stat['size'] % 512));
  108. }
  109. } else {
  110. fwrite($this->tmp, (binary)$fileOrStream);
  111. if (strlen($fileOrStream) % 512) {
  112. fwrite($this->tmp, (binary)str_repeat("\0", 512 - strlen($fileOrStream) % 512));
  113. }
  114. }
  115. }
  116. /**
  117. * Initialize the package creator
  118. */
  119. function init()
  120. {
  121. switch ($this->compress) {
  122. case 'zlib' :
  123. $this->tmp = gzopen($this->path, 'wb');
  124. break;
  125. case 'bz2' :
  126. $this->tmp = bzopen($this->path, 'w');
  127. break;
  128. case 'none' :
  129. $this->tmp = fopen($this->path, 'wb');
  130. break;
  131. default :
  132. throw new Exception(
  133. 'unknown compression type ' . $this->compress);
  134. }
  135. }
  136. /**
  137. * Create an internal directory, creating parent directories as needed
  138. *
  139. * @param string $dir
  140. */
  141. function mkdir($dir)
  142. {
  143. $this->addFile($dir, "", array(
  144. 'mode' => 0x4000 + 0644,
  145. 'uid' => 0,
  146. 'gid' => 0,
  147. 'size' => 0,
  148. 'mtime' => time(),
  149. ));
  150. }
  151. /**
  152. * Finish saving the package
  153. */
  154. function close()
  155. {
  156. fwrite($this->tmp, pack('a1024', ''));
  157. fclose($this->tmp);
  158. }
  159. }