mirror of
				https://github.com/phpbb/phpbb.git
				synced 2025-10-25 05:36:13 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			504 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			504 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?
 | |
| // -------------------------------------------------------------
 | |
| //
 | |
| // $Id$
 | |
| //
 | |
| // FILENAME  : functions_compress.php
 | |
| // STARTED   : Sat Jul 19 2003
 | |
| // COPYRIGHT : © 2003 phpBB Group
 | |
| // WWW       : http://www.phpbb.com/
 | |
| // LICENCE   : GPL vs2.0 [ see /docs/COPYING ] 
 | |
| // 
 | |
| // -------------------------------------------------------------
 | |
| 
 | |
| class compress 
 | |
| {
 | |
| 	var $fp = 0;
 | |
| 
 | |
| 	function add_file($src, $src_rm_prefix = '', $src_add_prefix = '', $skip_files = '')
 | |
| 	{
 | |
| 		global $phpbb_root_path;
 | |
| 
 | |
| 		$skip_files = explode(',', $skip_files);
 | |
| 
 | |
| 		// Remove rm prefix from src path 
 | |
| 		$src_path = ($src_rm_prefix) ? preg_replace('#^(' . preg_quote($src_rm_prefix) . ')#', '', $src) : $src;
 | |
| 		// Add src prefix
 | |
| 		$src_path = ($src_add_prefix) ? ($src_add_prefix . ((substr($src_add_prefix, -1) != '/') ? '/' : '') . $src_path) : $src_path;
 | |
| 		// Remove initial "/" if present
 | |
| 		$src_path = (substr($src_path, 0, 1) == '/') ? substr($src_path, 1) : $src_path;
 | |
| 
 | |
| 		if (is_file($phpbb_root_path . $src))
 | |
| 		{
 | |
| 			if (!($fp = @fopen($phpbb_root_path . $src, 'rb')))
 | |
| 			{
 | |
| 				return false;
 | |
| 			}
 | |
| 
 | |
| 			$data = fread($fp, filesize($phpbb_root_path . $src));
 | |
| 			fclose($fp);
 | |
| 
 | |
| 			$this->data($src_path, $data, filemtime($phpbb_root_path . $src), false);
 | |
| 		}
 | |
| 		else if (is_dir($phpbb_root_path . $src))
 | |
| 		{
 | |
| 			// Clean up path, add closing / if not present
 | |
| 			$src_path = ($src_path && substr($src_path, -1) != '/') ? $src_path . '/' : $src_path;
 | |
| 
 | |
| 			$filelist = array();
 | |
| 			$filelist = filelist($phpbb_root_path . $src, '', '*');
 | |
| 			krsort($filelist);
 | |
| 
 | |
| 			if ($src_path)
 | |
| 			{
 | |
| 				$mtime = (file_exists($phpbb_root_path . $src_path)) ? filemtime($src_path) : time();
 | |
| 				$this->data($src_path, '', $mtime, true);
 | |
| 			}
 | |
| 
 | |
| 			foreach ($filelist as $path => $file_ary)
 | |
| 			{
 | |
| 				if ($path)
 | |
| 				{
 | |
| 					// Same as for src_path
 | |
| 					$path = (substr($path, 0, 1) == '/') ? substr($path, 1) : $path;
 | |
| 					$path = ($path && substr($path, -1) != '/') ? $path . '/' : $path;
 | |
| 
 | |
| 					$this->data("$src_path$path", '', filemtime($phpbb_root_path . $path), true);
 | |
| 				}
 | |
| 
 | |
| 				foreach ($file_ary as $file)
 | |
| 				{
 | |
| 					if (in_array($path . $file, $skip_files))
 | |
| 					{
 | |
| 						continue;
 | |
| 					}
 | |
| 
 | |
| 					$this->data("$src_path$path$file", implode('', file($phpbb_root_path . $src . $path . $file)), filemtime($phpbb_root_path . $src . $path . $file), false);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	function add_data($src, $name)
 | |
| 	{
 | |
| 		$this->data($name, $src);
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	function methods()
 | |
| 	{
 | |
| 		$methods = array('tar');
 | |
| 
 | |
| 		foreach (array('tar.gz' => 'zlib', 'tar.bz2' => 'bz2', 'zip' => 'zlib') as $type => $module)
 | |
| 		{
 | |
| 			if (!@extension_loaded($module))
 | |
| 			{
 | |
| 				break;
 | |
| 			}
 | |
| 			$methods[] = $type;
 | |
| 		}
 | |
| 
 | |
| 		return $methods;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Zip creation class from phpMyAdmin 2.3.0 © Tobias Ratschiller, Olivier Müller, Loïc Chapeaux, 
 | |
| // Marc Delisle, http://www.phpmyadmin.net/
 | |
| //
 | |
| // Modified extensively by psoTFX, © phpBB Group, 2003
 | |
| //
 | |
| // Based on work by Eric Mueller and Denis125
 | |
| // Official ZIP file format: http://www.pkware.com/appnote.txt
 | |
| class compress_zip extends compress
 | |
| {
 | |
| 	var $datasec = array();
 | |
| 	var $ctrl_dir = array();
 | |
| 	var $eof_cdh = "\x50\x4b\x05\x06\x00\x00\x00\x00";
 | |
| 
 | |
| 	var $old_offset = 0;
 | |
| 	var $datasec_len = 0;
 | |
| 
 | |
| 	function compress_zip($mode, $file)
 | |
| 	{
 | |
| 		return $this->fp = @fopen($phpbb_root_path . $file, $mode . 'b');
 | |
| 	}
 | |
| 
 | |
| 	function unix_to_dos_time($time)
 | |
| 	{
 | |
| 		$timearray = (!$time) ? getdate() : getdate($time);
 | |
| 
 | |
| 		if ($timearray['year'] < 1980)
 | |
| 		{
 | |
| 			$timearray['year'] = 1980;
 | |
| 			$timearray['mon'] = $timearray['mday'] = 1;
 | |
| 			$timearray['hours'] = $timearray['minutes'] = $timearray['seconds'] = 0;
 | |
| 		}
 | |
| 
 | |
| 		return (($timearray['year'] - 1980) << 25) | ($timearray['mon'] << 21) | ($timearray['mday'] << 16) | ($timearray['hours'] << 11) | ($timearray['minutes'] << 5) | ($timearray['seconds'] >> 1);
 | |
| 	}
 | |
| 
 | |
| 	function extract($dst)
 | |
| 	{
 | |
| 		$header = $data = '';
 | |
| 		$seek_ary = $mkdir_ary = array();
 | |
| 		$j = 0;
 | |
| 
 | |
| 		fseek($this->fp, -14, SEEK_END);
 | |
| 		$tmp = unpack("ventries/vtotentries/Vctsize/Vctpos", fread($this->fp, 12));
 | |
| 		$entries = (int) trim($tmp['entries']);
 | |
| 		$totentries = (int) trim($tmp['totentries']);
 | |
| 		$ctsize = (int) trim($tmp['ctsize']);
 | |
| 		$ctpos = (int) trim($tmp['ctpos']);
 | |
| 
 | |
| 		fseek($this->fp, $ctpos);
 | |
| 
 | |
| 		// First scan entries, pull out position of data, length, etc.
 | |
| 		// and directory structure
 | |
| 		for ($i = 0; $i < $entries; $i++)
 | |
| 		{
 | |
| 			$buffer = fread($this->fp, 46);
 | |
| 
 | |
| 			$tmp = unpack("vc_method/Vmtime/Vcrc/Vc_size/Vuc_size/vstrlen", substr($buffer, 10, 20));
 | |
| 			$c_method = (int) trim($tmp['c_method']);
 | |
| 			$crc = (int) trim($tmp['crc']);
 | |
| 			$strlen = (int) trim($tmp['strlen']);
 | |
| 			$uc_size = (int) trim($tmp['uc_size']);
 | |
| 			$c_size = (int) trim($tmp['c_size']);
 | |
| 
 | |
| 			$tmp = unpack("Vattrib/Voffset", substr($buffer, 38, 8));
 | |
| 			$attrib = (int) trim($tmp['attrib']);
 | |
| 			$offset = (int) trim($tmp['offset']);
 | |
| 
 | |
| 			$filename =  fread($this->fp, $strlen);
 | |
| 
 | |
| 			if ($attrib == 32)
 | |
| 			{
 | |
| 				$seek_ary[$j]['c_method'] = $c_method;
 | |
| 				$seek_ary[$j]['crc'] = $crc;
 | |
| 				$seek_ary[$j]['strlen'] = $strlen;
 | |
| 				$seek_ary[$j]['uc_size'] = $uc_size;
 | |
| 				$seek_ary[$j]['c_size'] = $c_size;
 | |
| 
 | |
| 				$seek_ary[$j]['offset'] = $offset;
 | |
| 				$seek_ary[$j]['filename'] = "$dst$filename";
 | |
| 
 | |
| 				$j++;
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$mkdir_ary[] = "$dst$filename";
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Create directory structure on fs
 | |
| 		if (is_array($mkdir_ary))
 | |
| 		{
 | |
| 			sort($mkdir_ary);
 | |
| 			foreach ($mkdir_ary as $dir)
 | |
| 			{
 | |
| 				if (!@mkdir($dir, 0777))
 | |
| 				{
 | |
| 					trigger_error("Could not create directory $dir");
 | |
| 				}
 | |
| 				@chmod("$dir", 0777);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Extract files
 | |
| 		foreach ($seek_ary as $seek)
 | |
| 		{
 | |
| 			$filename = $seek['filename'];
 | |
| 
 | |
| //			fseek($this->fp, $seek['offset'] + 8); // To grab file header info
 | |
| //			fseek($this->fp, $seek['offset'] + 30 + $tmp['strlen'] + $tmp['c_size']); // To grab file header info2
 | |
| 
 | |
| 			// Jump to data
 | |
| 			fseek($this->fp, $seek['offset'] + 30 + $seek['strlen']);
 | |
| 
 | |
| 			// Was data compressed? If so we have to fudge a solution thanks
 | |
| 			// to some "issues" with gzuncompress. Else we just write out the
 | |
| 			// data
 | |
| 			if ($seek['c_method'] == 8)
 | |
| 			{
 | |
| 				// Temp gzip file -> .gz header -> data -> gz footer
 | |
| 				if (!($fp = fopen($filename . '.gz', 'wb')))
 | |
| 				{
 | |
| 					trigger_error("Could not open temporary $filename.gz");
 | |
| 				}
 | |
| 				fwrite($fp, pack('va1a1Va1a1', 0x8b1f, chr(0x08), chr(0x00), time(), chr(0x00), chr(3)));
 | |
| 				fwrite($fp, fread($this->fp, $seek['c_size']));
 | |
| 				fwrite($fp, pack("VV", $seek['crc'], $seek['uc_size']));
 | |
| 				fclose($fp);
 | |
| 
 | |
| 				if (!($fp = fopen($filename, 'wb')))
 | |
| 				{
 | |
| 					trigger_error("Could not create $filename");
 | |
| 				}
 | |
| 				@chmod($filename, 0777);
 | |
| 
 | |
| 				if (!($gzfp = gzopen($filename . '.gz', 'rb')))
 | |
| 				{
 | |
| 					die("Could not open temporary $filename.gz");
 | |
| 				}
 | |
| 
 | |
| 				while ($buffer = gzread($gzfp, 1024))
 | |
| 				{
 | |
| 					fwrite($fp, $buffer);
 | |
| 				}
 | |
| 				gzclose($gzfp);
 | |
| 				fclose($fp);
 | |
| 				unlink($filename . '.gz');
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				if (!($fp = fopen($filename, 'wb')))
 | |
| 				{
 | |
| 					trigger_error("Could not create $filename");
 | |
| 				}
 | |
| 				@chmod($filename, 0777);
 | |
| 
 | |
| 				fwrite($fp, fread($this->fp, $seek['uc_size']));
 | |
| 				fclose($fp);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function close()
 | |
| 	{
 | |
| 		// Write out central file directory and footer ... if it exists
 | |
| 		if (sizeof($this->ctrl_dir))
 | |
| 		{
 | |
| 			fwrite($this->fp, $this->file());
 | |
| 		}
 | |
| 		fclose($this->fp);
 | |
| 	}
 | |
| 
 | |
| 	// Create the structures ... note we assume version made by is MSDOS
 | |
| 	function data($name, $data, $mtime = false, $is_dir = false)
 | |
| 	{
 | |
| 		$name = str_replace('\\', '/', $name);
 | |
| 
 | |
| 		$dtime = dechex($this->unix_to_dos_time($mtime));
 | |
| 		$hexdtime = '\x' . $dtime[6] . $dtime[7] . '\x' . $dtime[4] . $dtime[5] . '\x' . $dtime[2] . $dtime[3] . '\x' . $dtime[0] . $dtime[1];
 | |
| 		eval('$hexdtime = "' . $hexdtime . '";');
 | |
| 
 | |
| 		if ($is_dir)
 | |
| 		{
 | |
| 			$unc_len = $c_len = $crc = 0;
 | |
| 			$zdata = '';
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$unc_len = strlen($data);
 | |
| 			$crc = crc32($data);
 | |
| 			$zdata = gzcompress($data);
 | |
| 			$zdata = substr(substr($zdata, 0, strlen($zdata) - 4), 2); // fix crc bug
 | |
| 			$c_len = strlen($zdata);
 | |
| 
 | |
| 			// Did we compress? No, then use data as is
 | |
| 			if ($c_len >= $unc_len)
 | |
| 			{
 | |
| 				$zdata = $data;
 | |
| 				$c_len = $unc_len;
 | |
| 			}
 | |
| 		}
 | |
| 		unset($data);
 | |
| 
 | |
| 		// If we didn't compress set method to store, else deflate
 | |
| 		$c_method = ($c_len == $unc_len) ? "\x00\x00" : "\x08\x00"; 
 | |
| 
 | |
| 		// Are we a file or a directory? Set archive for file
 | |
| 		$attrib = ($is_dir) ? 16 : 32;
 | |
| 		$var_ext = ($is_dir) ? "\x0a" : "\x14";
 | |
| 
 | |
| 		// File Record Header
 | |
| 		$fr = "\x50\x4b\x03\x04";		// Local file header 4bytes
 | |
| 		$fr .= "$var_ext\x00";			// ver needed to extract 2bytes
 | |
| 		$fr .= "\x00\x00";				// gen purpose bit flag 2bytes
 | |
| 		$fr .= $c_method;				// compression method 2bytes
 | |
| 		$fr .= $hexdtime;				// last mod time and date 2+2bytes
 | |
| 		$fr .= pack('V', $crc);			// crc32 4bytes
 | |
| 		$fr .= pack('V', $c_len);		// compressed filesize 4bytes
 | |
| 		$fr .= pack('V', $unc_len);		// uncompressed filesize 4bytes
 | |
| 		$fr .= pack('v', strlen($name));// length of filename 2bytes
 | |
| 
 | |
| 		$fr .= pack('v', 0);			// extra field length 2bytes
 | |
| 		$fr .= $name;
 | |
| 		$fr .= $zdata;
 | |
| 		unset($zdata);
 | |
| 		$fr .= pack('V', $crc);			// crc32 4bytes
 | |
| 		$fr .= pack('V', $c_len);		// compressed filesize 4bytes
 | |
| 		$fr .= pack('V', $unc_len);		// uncompressed filesize 4bytes
 | |
| 
 | |
| 		$this->datasec_len += strlen($fr);
 | |
| 
 | |
| 		// Add data to file ... by writing data out incrementally we save some memory
 | |
| 		fwrite($this->fp, $fr);
 | |
| 		unset($fr);
 | |
| 
 | |
| 		// Central Directory Header
 | |
| 		$cdrec = "\x50\x4b\x01\x02";		// header 4bytes
 | |
| 		$cdrec .= "\x00\x00";               // version made by
 | |
| 		$cdrec .= "$var_ext\x00";           // version needed to extract
 | |
| 		$cdrec .= "\x00\x00";               // gen purpose bit flag
 | |
| 		$cdrec .= $c_method;				// compression method 
 | |
| 		$cdrec .= $hexdtime;                // last mod time & date
 | |
| 		$cdrec .= pack('V', $crc);          // crc32
 | |
| 		$cdrec .= pack('V', $c_len);        // compressed filesize
 | |
| 		$cdrec .= pack('V', $unc_len);      // uncompressed filesize
 | |
| 		$cdrec .= pack('v', strlen($name)); // length of filename
 | |
| 		$cdrec .= pack('v', 0);             // extra field length
 | |
| 		$cdrec .= pack('v', 0);             // file comment length
 | |
| 		$cdrec .= pack('v', 0);             // disk number start
 | |
| 		$cdrec .= pack('v', 0);             // internal file attributes
 | |
| 		$cdrec .= pack('V', $attrib);		// external file attributes
 | |
| 		$cdrec .= pack('V', $this->old_offset); // relative offset of local header
 | |
| 		$cdrec .= $name;
 | |
| 
 | |
| 		// Save to central directory
 | |
| 		$this->ctrl_dir[] = $cdrec;
 | |
| 
 | |
| 		$this->old_offset = $this->datasec_len;
 | |
| 	}
 | |
| 
 | |
| 	function file()
 | |
| 	{
 | |
| 		$ctrldir = implode('', $this->ctrl_dir);
 | |
| 
 | |
| 		return $ctrldir . $this->eof_cdh . 
 | |
| 			pack('v', sizeof($this->ctrl_dir)) .	// total # of entries "on this disk"
 | |
| 			pack('v', sizeof($this->ctrl_dir)) .	// total # of entries overall
 | |
| 			pack('V', strlen($ctrldir)) .			// size of central dir
 | |
| 			pack('V', $this->datasec_len) .			// offset to start of central dir
 | |
| 			"\x00\x00";								// .zip file comment length
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Tar/tar.gz compression routine
 | |
| // Header/checksum creation derived from tarfile.pl, © Tom Horsley, 1994
 | |
| class compress_tar extends compress 
 | |
| {
 | |
| 	var $fzopen = '';
 | |
| 	var $fzclose = '';
 | |
| 	var $fzread = '';
 | |
| 	var $fzwrite = '';
 | |
| 	var $isgz = false;
 | |
| 	var $isbz = false;
 | |
| 
 | |
| 	function compress_tar($mode, $file, $type = '')
 | |
| 	{
 | |
| 		$type = (!$type) ? $file : $type;
 | |
| 		$this->isgz = (strpos($type, '.tar.gz') !== false || strpos($type, '.tgz') !== false) ? true : false;
 | |
| 		$this->isbz = (strpos($type, '.tar.bz2') !== false) ? true : false;
 | |
| 
 | |
| 		$fzopen = ($this->isbz && function_exists('bzopen')) ? 'bzopen' : (($this->isgz && extension_loaded('zlib')) ? 'gzopen' : 'fopen');
 | |
| 		return $this->fp = @$fzopen($phpbb_root_path . $file, $mode . 'b');
 | |
| 	}
 | |
| 
 | |
| 	function extract($dst)
 | |
| 	{
 | |
| 		$fzread = ($this->isbz && function_exists('bzread')) ? 'bzread' : (($this->isgz && extension_loaded('zlib')) ? 'gzread' : 'fread');
 | |
| 
 | |
| 		$size = 0;
 | |
| 		$header = $data = '';
 | |
| 		while ($buffer = $fzread($this->fp, 512))
 | |
| 		{
 | |
| 			$tmp = unpack("A6magic", substr($buffer, 257, 6));
 | |
| 
 | |
| 			if (trim($tmp['magic']) == 'ustar')
 | |
| 			{
 | |
| 				$tmp = unpack("A100name", $buffer);
 | |
| 				$filename = trim($tmp['name']);
 | |
| 
 | |
| 				$tmp = unpack("Atype", substr($buffer, 156, 1));
 | |
| 				$filetype = (int) trim($tmp['type']);
 | |
| 
 | |
| 				if ($filetype == 5)
 | |
| 				{
 | |
| 					if (!@mkdir("$dst$filename", 0777))
 | |
| 					{
 | |
| 						trigger_error("Could not create directory $filename");
 | |
| 					}
 | |
| 					@chmod("$dst$filename", 0777);
 | |
| 					continue;
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					$tmp = unpack("A12size", substr($buffer, 124, 12));
 | |
| 					$filesize = octdec((int) trim($tmp['size']));
 | |
| 
 | |
| 					if (!($fp = fopen("$dst$filename", 'wb')))
 | |
| 					{
 | |
| 						trigger_error("Could create file $filename");
 | |
| 					}
 | |
| 					@chmod("$dst$filename", 0777);
 | |
| 
 | |
| 					$size = 0;
 | |
| 					continue;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			$size += 512;
 | |
| 			$length = ($size > $filesize) ? 512 - ($size - $filesize) : 512;
 | |
| 
 | |
| 			$tmp = unpack("A512data", $buffer);
 | |
| 
 | |
| 			fwrite($fp, (string) $tmp['data'], $length);
 | |
| 			unset($buffer);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function close()
 | |
| 	{
 | |
| 		$fzclose = ($this->isbz && function_exists('bzclose')) ? 'bzclose' : (($this->isgz && extension_loaded('zlib')) ? 'gzclose' : 'fclose');
 | |
| 		$fzclose($this->fp);
 | |
| 	}
 | |
| 
 | |
| 	function data($name, $data, $mtime = false, $is_dir = false)
 | |
| 	{
 | |
| 		$fzwrite = 	($this->isbz && function_exists('bzwrite')) ? 'bzwrite' : (($this->isgz && extension_loaded('zlib')) ? 'gzwrite' : 'fwrite');
 | |
| 
 | |
| 		$mode = ($is_dir) ? '493' : '436';
 | |
| 		$mtime = (!$mtime) ? time() : $mtime;
 | |
| 		$filesize = ($is_dir) ? 0 : strlen($data);
 | |
| 		$typeflag = ($is_dir) ? '5' : '';
 | |
| 
 | |
| 		$header = '';
 | |
| 		$header .= pack("a100", $name);
 | |
| 		$header .= pack("a8", sprintf("%07o", $mode));
 | |
| 		$header .= pack("a8", sprintf("%07o", 0));
 | |
| 		$header .= pack("a8", sprintf("%07o", 0));
 | |
| 		$header .= pack("a12", sprintf("%011o", $filesize));
 | |
| 		$header .= pack("a12", sprintf("%011o", $mtime));
 | |
| 		$header .= '        ';
 | |
| 		$header .= pack("a", $typeflag);
 | |
| 		$header .= pack("a100", '');
 | |
| 		$header .= 'ustar';
 | |
| 		$header .= pack("x");
 | |
| 		$header .= '00';
 | |
| 		$header .= pack("x247");
 | |
| 
 | |
| 		// Checksum
 | |
| 		for ($i = 0; $i < 512; $i++)
 | |
| 		{
 | |
| 			$b = unpack("c1char", substr($header, $i, 1));
 | |
| 			$checksum += $b['char'];
 | |
| 		}
 | |
| 		$header = substr_replace($header, pack("a8",sprintf("%07o", $checksum)), 148, 8);
 | |
| 
 | |
| 		$fzwrite($this->fp, $header);
 | |
| 
 | |
| 		$i = 0;
 | |
| 		// Read the data 512 bytes at a time and write it out
 | |
| 		while ($buffer = substr($data, $i, 512))
 | |
| 		{
 | |
| 			$fzwrite($this->fp, pack("a512", $buffer));
 | |
| 			$i += 512;
 | |
| 		}
 | |
| 		unset($data);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ?>
 |