Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed
<?php// --------------------------------------------------------------------------------// PhpConcept Library - Zip Module 2.8.2// --------------------------------------------------------------------------------// License GNU/LGPL - Vincent Blavet - August 2009// http://www.phpconcept.net// --------------------------------------------------------------------------------//// Presentation :// PclZip is a PHP library that manage ZIP archives.// So far tests show that archives generated by PclZip are readable by// WinZip application and other tools.//// Description :// See readme.txt and http://www.phpconcept.net//// Warning :// This library and the associated files are non commercial, non professional// work.// It should not have unexpected results. However if any damage is caused by// this software the author can not be responsible.// The use of this software is at the risk of the user.//// --------------------------------------------------------------------------------// $Id: pclzip.lib.php,v 1.60 2009/09/30 21:01:04 vblavet Exp $// --------------------------------------------------------------------------------// ----- Constantsif (!defined('PCLZIP_READ_BLOCK_SIZE')) {define( 'PCLZIP_READ_BLOCK_SIZE', 2048 );}// ----- File list separator// In version 1.x of PclZip, the separator for file list is a space// (which is not a very smart choice, specifically for windows paths !).// A better separator should be a comma (,). This constant gives you the// abilty to change that.// However notice that changing this value, may have impact on existing// scripts, using space separated filenames.// Recommanded values for compatibility with older versions ://define( 'PCLZIP_SEPARATOR', ' ' );// Recommanded values for smart separation of filenames.if (!defined('PCLZIP_SEPARATOR')) {define( 'PCLZIP_SEPARATOR', ',' );}// ----- Error configuration// 0 : PclZip Class integrated error handling// 1 : PclError external library error handling. By enabling this// you must ensure that you have included PclError library.// [2,...] : reserved for futur useif (!defined('PCLZIP_ERROR_EXTERNAL')) {define( 'PCLZIP_ERROR_EXTERNAL', 0 );}// ----- Optional static temporary directory// By default temporary files are generated in the script current// path.// If defined :// - MUST BE terminated by a '/'.// - MUST be a valid, already created directory// Samples :// define( 'PCLZIP_TEMPORARY_DIR', '/temp/' );// define( 'PCLZIP_TEMPORARY_DIR', 'C:/Temp/' );if (!defined('PCLZIP_TEMPORARY_DIR')) {define( 'PCLZIP_TEMPORARY_DIR', '' );}// ----- Optional threshold ratio for use of temporary files// Pclzip sense the size of the file to add/extract and decide to// use or not temporary file. The algorythm is looking for// memory_limit of PHP and apply a ratio.// threshold = memory_limit * ratio.// Recommended values are under 0.5. Default 0.47.// Samples :// define( 'PCLZIP_TEMPORARY_FILE_RATIO', 0.5 );if (!defined('PCLZIP_TEMPORARY_FILE_RATIO')) {define( 'PCLZIP_TEMPORARY_FILE_RATIO', 0.47 );}// --------------------------------------------------------------------------------// ***** UNDER THIS LINE NOTHING NEEDS TO BE MODIFIED *****// --------------------------------------------------------------------------------// ----- Global variables$g_pclzip_version = "2.8.2";// ----- Error codes// -1 : Unable to open file in binary write mode// -2 : Unable to open file in binary read mode// -3 : Invalid parameters// -4 : File does not exist// -5 : Filename is too long (max. 255)// -6 : Not a valid zip file// -7 : Invalid extracted file size// -8 : Unable to create directory// -9 : Invalid archive extension// -10 : Invalid archive format// -11 : Unable to delete file (unlink)// -12 : Unable to rename file (rename)// -13 : Invalid header checksum// -14 : Invalid archive sizedefine( 'PCLZIP_ERR_USER_ABORTED', 2 );define( 'PCLZIP_ERR_NO_ERROR', 0 );define( 'PCLZIP_ERR_WRITE_OPEN_FAIL', -1 );define( 'PCLZIP_ERR_READ_OPEN_FAIL', -2 );define( 'PCLZIP_ERR_INVALID_PARAMETER', -3 );define( 'PCLZIP_ERR_MISSING_FILE', -4 );define( 'PCLZIP_ERR_FILENAME_TOO_LONG', -5 );define( 'PCLZIP_ERR_INVALID_ZIP', -6 );define( 'PCLZIP_ERR_BAD_EXTRACTED_FILE', -7 );define( 'PCLZIP_ERR_DIR_CREATE_FAIL', -8 );define( 'PCLZIP_ERR_BAD_EXTENSION', -9 );define( 'PCLZIP_ERR_BAD_FORMAT', -10 );define( 'PCLZIP_ERR_DELETE_FILE_FAIL', -11 );define( 'PCLZIP_ERR_RENAME_FILE_FAIL', -12 );define( 'PCLZIP_ERR_BAD_CHECKSUM', -13 );define( 'PCLZIP_ERR_INVALID_ARCHIVE_ZIP', -14 );define( 'PCLZIP_ERR_MISSING_OPTION_VALUE', -15 );define( 'PCLZIP_ERR_INVALID_OPTION_VALUE', -16 );define( 'PCLZIP_ERR_ALREADY_A_DIRECTORY', -17 );define( 'PCLZIP_ERR_UNSUPPORTED_COMPRESSION', -18 );define( 'PCLZIP_ERR_UNSUPPORTED_ENCRYPTION', -19 );define( 'PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE', -20 );define( 'PCLZIP_ERR_DIRECTORY_RESTRICTION', -21 );// ----- Options valuesdefine( 'PCLZIP_OPT_PATH', 77001 );define( 'PCLZIP_OPT_ADD_PATH', 77002 );define( 'PCLZIP_OPT_REMOVE_PATH', 77003 );define( 'PCLZIP_OPT_REMOVE_ALL_PATH', 77004 );define( 'PCLZIP_OPT_SET_CHMOD', 77005 );define( 'PCLZIP_OPT_EXTRACT_AS_STRING', 77006 );define( 'PCLZIP_OPT_NO_COMPRESSION', 77007 );define( 'PCLZIP_OPT_BY_NAME', 77008 );define( 'PCLZIP_OPT_BY_INDEX', 77009 );define( 'PCLZIP_OPT_BY_EREG', 77010 );define( 'PCLZIP_OPT_BY_PREG', 77011 );define( 'PCLZIP_OPT_COMMENT', 77012 );define( 'PCLZIP_OPT_ADD_COMMENT', 77013 );define( 'PCLZIP_OPT_PREPEND_COMMENT', 77014 );define( 'PCLZIP_OPT_EXTRACT_IN_OUTPUT', 77015 );define( 'PCLZIP_OPT_REPLACE_NEWER', 77016 );define( 'PCLZIP_OPT_STOP_ON_ERROR', 77017 );// Having big trouble with crypt. Need to multiply 2 long int// which is not correctly supported by PHP ...//define( 'PCLZIP_OPT_CRYPT', 77018 );define( 'PCLZIP_OPT_EXTRACT_DIR_RESTRICTION', 77019 );define( 'PCLZIP_OPT_TEMP_FILE_THRESHOLD', 77020 );define( 'PCLZIP_OPT_ADD_TEMP_FILE_THRESHOLD', 77020 ); // aliasdefine( 'PCLZIP_OPT_TEMP_FILE_ON', 77021 );define( 'PCLZIP_OPT_ADD_TEMP_FILE_ON', 77021 ); // aliasdefine( 'PCLZIP_OPT_TEMP_FILE_OFF', 77022 );define( 'PCLZIP_OPT_ADD_TEMP_FILE_OFF', 77022 ); // alias// ----- File description attributesdefine( 'PCLZIP_ATT_FILE_NAME', 79001 );define( 'PCLZIP_ATT_FILE_NEW_SHORT_NAME', 79002 );define( 'PCLZIP_ATT_FILE_NEW_FULL_NAME', 79003 );define( 'PCLZIP_ATT_FILE_MTIME', 79004 );define( 'PCLZIP_ATT_FILE_CONTENT', 79005 );define( 'PCLZIP_ATT_FILE_COMMENT', 79006 );// ----- Call backs valuesdefine( 'PCLZIP_CB_PRE_EXTRACT', 78001 );define( 'PCLZIP_CB_POST_EXTRACT', 78002 );define( 'PCLZIP_CB_PRE_ADD', 78003 );define( 'PCLZIP_CB_POST_ADD', 78004 );/* For futur usedefine( 'PCLZIP_CB_PRE_LIST', 78005 );define( 'PCLZIP_CB_POST_LIST', 78006 );define( 'PCLZIP_CB_PRE_DELETE', 78007 );define( 'PCLZIP_CB_POST_DELETE', 78008 );*/// --------------------------------------------------------------------------------// Class : PclZip// Description :// PclZip is the class that represent a Zip archive.// The public methods allow the manipulation of the archive.// Attributes :// Attributes must not be accessed directly.// Methods :// PclZip() : Object creator// create() : Creates the Zip archive// listContent() : List the content of the Zip archive// extract() : Extract the content of the archive// properties() : List the properties of the archive// --------------------------------------------------------------------------------class PclZip{// ----- Filename of the zip filevar $zipname = '';// ----- File descriptor of the zip filevar $zip_fd = 0;// ----- Internal error handlingvar $error_code = 1;var $error_string = '';// ----- Current status of the magic_quotes_runtime// This value store the php configuration for magic_quotes// The class can then disable the magic_quotes and reset it aftervar $magic_quotes_status;// --------------------------------------------------------------------------------// Function : PclZip()// Description :// Creates a PclZip object and set the name of the associated Zip archive// filename.// Note that no real action is taken, if the archive does not exist it is not// created. Use create() for that.// --------------------------------------------------------------------------------function PclZip($p_zipname){// ----- Tests the zlibif (!function_exists('gzopen')){die('Abort '.basename(__FILE__).' : Missing zlib extensions');}// ----- Set the attributes$this->zipname = $p_zipname;$this->zip_fd = 0;$this->magic_quotes_status = -1;// ----- Returnreturn;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function :// create($p_filelist, $p_add_dir="", $p_remove_dir="")// create($p_filelist, $p_option, $p_option_value, ...)// Description :// This method supports two different synopsis. The first one is historical.// This method creates a Zip Archive. The Zip file is created in the// filesystem. The files and directories indicated in $p_filelist// are added in the archive. See the parameters description for the// supported format of $p_filelist.// When a directory is in the list, the directory and its content is added// in the archive.// In this synopsis, the function takes an optional variable list of// options. See bellow the supported options.// Parameters :// $p_filelist : An array containing file or directory names, or// a string containing one filename or one directory name, or// a string containing a list of filenames and/or directory// names separated by spaces.// $p_add_dir : A path to add before the real path of the archived file,// in order to have it memorized in the archive.// $p_remove_dir : A path to remove from the real path of the file to archive,// in order to have a shorter path memorized in the archive.// When $p_add_dir and $p_remove_dir are set, $p_remove_dir// is removed first, before $p_add_dir is added.// Options :// PCLZIP_OPT_ADD_PATH :// PCLZIP_OPT_REMOVE_PATH :// PCLZIP_OPT_REMOVE_ALL_PATH :// PCLZIP_OPT_COMMENT :// PCLZIP_CB_PRE_ADD :// PCLZIP_CB_POST_ADD :// Return Values :// 0 on failure,// The list of the added files, with a status of the add action.// (see PclZip::listContent() for list entry format)// --------------------------------------------------------------------------------function create($p_filelist){$v_result=1;// ----- Reset the error handler$this->privErrorReset();// ----- Set default values$v_options = array();$v_options[PCLZIP_OPT_NO_COMPRESSION] = FALSE;// ----- Look for variable options arguments$v_size = func_num_args();// ----- Look for argumentsif ($v_size > 1) {// ----- Get the arguments$v_arg_list = func_get_args();// ----- Remove from the options list the first argumentarray_shift($v_arg_list);$v_size--;// ----- Look for first argif ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) {// ----- Parse the options$v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options,array (PCLZIP_OPT_REMOVE_PATH => 'optional',PCLZIP_OPT_REMOVE_ALL_PATH => 'optional',PCLZIP_OPT_ADD_PATH => 'optional',PCLZIP_CB_PRE_ADD => 'optional',PCLZIP_CB_POST_ADD => 'optional',PCLZIP_OPT_NO_COMPRESSION => 'optional',PCLZIP_OPT_COMMENT => 'optional',PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional',PCLZIP_OPT_TEMP_FILE_ON => 'optional',PCLZIP_OPT_TEMP_FILE_OFF => 'optional'//, PCLZIP_OPT_CRYPT => 'optional'));if ($v_result != 1) {return 0;}}// ----- Look for 2 args// Here we need to support the first historic synopsis of the// method.else {// ----- Get the first argument$v_options[PCLZIP_OPT_ADD_PATH] = $v_arg_list[0];// ----- Look for the optional second argumentif ($v_size == 2) {$v_options[PCLZIP_OPT_REMOVE_PATH] = $v_arg_list[1];}else if ($v_size > 2) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER,"Invalid number / type of arguments");return 0;}}}// ----- Look for default option values$this->privOptionDefaultThreshold($v_options);// ----- Init$v_string_list = array();$v_att_list = array();$v_filedescr_list = array();$p_result_list = array();// ----- Look if the $p_filelist is really an arrayif (is_array($p_filelist)) {// ----- Look if the first element is also an array// This will mean that this is a file description entryif (isset($p_filelist[0]) && is_array($p_filelist[0])) {$v_att_list = $p_filelist;}// ----- The list is a list of string nameselse {$v_string_list = $p_filelist;}}// ----- Look if the $p_filelist is a stringelse if (is_string($p_filelist)) {// ----- Create a list from the string$v_string_list = explode(PCLZIP_SEPARATOR, $p_filelist);}// ----- Invalid variable type for $p_filelistelse {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type p_filelist");return 0;}// ----- Reformat the string listif (sizeof($v_string_list) != 0) {foreach ($v_string_list as $v_string) {if ($v_string != '') {$v_att_list[][PCLZIP_ATT_FILE_NAME] = $v_string;}else {}}}// ----- For each file in the list check the attributes$v_supported_attributes= array ( PCLZIP_ATT_FILE_NAME => 'mandatory',PCLZIP_ATT_FILE_NEW_SHORT_NAME => 'optional',PCLZIP_ATT_FILE_NEW_FULL_NAME => 'optional',PCLZIP_ATT_FILE_MTIME => 'optional',PCLZIP_ATT_FILE_CONTENT => 'optional',PCLZIP_ATT_FILE_COMMENT => 'optional');foreach ($v_att_list as $v_entry) {$v_result = $this->privFileDescrParseAtt($v_entry,$v_filedescr_list[],$v_options,$v_supported_attributes);if ($v_result != 1) {return 0;}}// ----- Expand the filelist (expand directories)$v_result = $this->privFileDescrExpand($v_filedescr_list, $v_options);if ($v_result != 1) {return 0;}// ----- Call the create fct$v_result = $this->privCreate($v_filedescr_list, $p_result_list, $v_options);if ($v_result != 1) {return 0;}// ----- Returnreturn $p_result_list;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function :// add($p_filelist, $p_add_dir="", $p_remove_dir="")// add($p_filelist, $p_option, $p_option_value, ...)// Description :// This method supports two synopsis. The first one is historical.// This methods add the list of files in an existing archive.// If a file with the same name already exists, it is added at the end of the// archive, the first one is still present.// If the archive does not exist, it is created.// Parameters :// $p_filelist : An array containing file or directory names, or// a string containing one filename or one directory name, or// a string containing a list of filenames and/or directory// names separated by spaces.// $p_add_dir : A path to add before the real path of the archived file,// in order to have it memorized in the archive.// $p_remove_dir : A path to remove from the real path of the file to archive,// in order to have a shorter path memorized in the archive.// When $p_add_dir and $p_remove_dir are set, $p_remove_dir// is removed first, before $p_add_dir is added.// Options :// PCLZIP_OPT_ADD_PATH :// PCLZIP_OPT_REMOVE_PATH :// PCLZIP_OPT_REMOVE_ALL_PATH :// PCLZIP_OPT_COMMENT :// PCLZIP_OPT_ADD_COMMENT :// PCLZIP_OPT_PREPEND_COMMENT :// PCLZIP_CB_PRE_ADD :// PCLZIP_CB_POST_ADD :// Return Values :// 0 on failure,// The list of the added files, with a status of the add action.// (see PclZip::listContent() for list entry format)// --------------------------------------------------------------------------------function add($p_filelist){$v_result=1;// ----- Reset the error handler$this->privErrorReset();// ----- Set default values$v_options = array();$v_options[PCLZIP_OPT_NO_COMPRESSION] = FALSE;// ----- Look for variable options arguments$v_size = func_num_args();// ----- Look for argumentsif ($v_size > 1) {// ----- Get the arguments$v_arg_list = func_get_args();// ----- Remove form the options list the first argumentarray_shift($v_arg_list);$v_size--;// ----- Look for first argif ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) {// ----- Parse the options$v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options,array (PCLZIP_OPT_REMOVE_PATH => 'optional',PCLZIP_OPT_REMOVE_ALL_PATH => 'optional',PCLZIP_OPT_ADD_PATH => 'optional',PCLZIP_CB_PRE_ADD => 'optional',PCLZIP_CB_POST_ADD => 'optional',PCLZIP_OPT_NO_COMPRESSION => 'optional',PCLZIP_OPT_COMMENT => 'optional',PCLZIP_OPT_ADD_COMMENT => 'optional',PCLZIP_OPT_PREPEND_COMMENT => 'optional',PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional',PCLZIP_OPT_TEMP_FILE_ON => 'optional',PCLZIP_OPT_TEMP_FILE_OFF => 'optional'//, PCLZIP_OPT_CRYPT => 'optional'));if ($v_result != 1) {return 0;}}// ----- Look for 2 args// Here we need to support the first historic synopsis of the// method.else {// ----- Get the first argument$v_options[PCLZIP_OPT_ADD_PATH] = $v_add_path = $v_arg_list[0];// ----- Look for the optional second argumentif ($v_size == 2) {$v_options[PCLZIP_OPT_REMOVE_PATH] = $v_arg_list[1];}else if ($v_size > 2) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid number / type of arguments");// ----- Returnreturn 0;}}}// ----- Look for default option values$this->privOptionDefaultThreshold($v_options);// ----- Init$v_string_list = array();$v_att_list = array();$v_filedescr_list = array();$p_result_list = array();// ----- Look if the $p_filelist is really an arrayif (is_array($p_filelist)) {// ----- Look if the first element is also an array// This will mean that this is a file description entryif (isset($p_filelist[0]) && is_array($p_filelist[0])) {$v_att_list = $p_filelist;}// ----- The list is a list of string nameselse {$v_string_list = $p_filelist;}}// ----- Look if the $p_filelist is a stringelse if (is_string($p_filelist)) {// ----- Create a list from the string$v_string_list = explode(PCLZIP_SEPARATOR, $p_filelist);}// ----- Invalid variable type for $p_filelistelse {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type '".gettype($p_filelist)."' for p_filelist");return 0;}// ----- Reformat the string listif (sizeof($v_string_list) != 0) {foreach ($v_string_list as $v_string) {$v_att_list[][PCLZIP_ATT_FILE_NAME] = $v_string;}}// ----- For each file in the list check the attributes$v_supported_attributes= array ( PCLZIP_ATT_FILE_NAME => 'mandatory',PCLZIP_ATT_FILE_NEW_SHORT_NAME => 'optional',PCLZIP_ATT_FILE_NEW_FULL_NAME => 'optional',PCLZIP_ATT_FILE_MTIME => 'optional',PCLZIP_ATT_FILE_CONTENT => 'optional',PCLZIP_ATT_FILE_COMMENT => 'optional');foreach ($v_att_list as $v_entry) {$v_result = $this->privFileDescrParseAtt($v_entry,$v_filedescr_list[],$v_options,$v_supported_attributes);if ($v_result != 1) {return 0;}}// ----- Expand the filelist (expand directories)$v_result = $this->privFileDescrExpand($v_filedescr_list, $v_options);if ($v_result != 1) {return 0;}// ----- Call the create fct$v_result = $this->privAdd($v_filedescr_list, $p_result_list, $v_options);if ($v_result != 1) {return 0;}// ----- Returnreturn $p_result_list;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : listContent()// Description :// This public method, gives the list of the files and directories, with their// properties.// The properties of each entries in the list are (used also in other functions) :// filename : Name of the file. For a create or add action it is the filename// given by the user. For an extract function it is the filename// of the extracted file.// stored_filename : Name of the file / directory stored in the archive.// size : Size of the stored file.// compressed_size : Size of the file's data compressed in the archive// (without the headers overhead)// mtime : Last known modification date of the file (UNIX timestamp)// comment : Comment associated with the file// folder : true | false// index : index of the file in the archive// status : status of the action (depending of the action) :// Values are :// ok : OK !// filtered : the file / dir is not extracted (filtered by user)// already_a_directory : the file can not be extracted because a// directory with the same name already exists// write_protected : the file can not be extracted because a file// with the same name already exists and is// write protected// newer_exist : the file was not extracted because a newer file exists// path_creation_fail : the file is not extracted because the folder// does not exist and can not be created// write_error : the file was not extracted because there was a// error while writing the file// read_error : the file was not extracted because there was a error// while reading the file// invalid_header : the file was not extracted because of an archive// format error (bad file header)// Note that each time a method can continue operating when there// is an action error on a file, the error is only logged in the file status.// Return Values :// 0 on an unrecoverable failure,// The list of the files in the archive.// --------------------------------------------------------------------------------function listContent(){$v_result=1;// ----- Reset the error handler$this->privErrorReset();// ----- Check archiveif (!$this->privCheckFormat()) {return(0);}// ----- Call the extracting fct$p_list = array();if (($v_result = $this->privList($p_list)) != 1){unset($p_list);return(0);}// ----- Returnreturn $p_list;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function :// extract($p_path="./", $p_remove_path="")// extract([$p_option, $p_option_value, ...])// Description :// This method supports two synopsis. The first one is historical.// This method extract all the files / directories from the archive to the// folder indicated in $p_path.// If you want to ignore the 'root' part of path of the memorized files// you can indicate this in the optional $p_remove_path parameter.// By default, if a newer file with the same name already exists, the// file is not extracted.//// If both PCLZIP_OPT_PATH and PCLZIP_OPT_ADD_PATH aoptions// are used, the path indicated in PCLZIP_OPT_ADD_PATH is append// at the end of the path value of PCLZIP_OPT_PATH.// Parameters :// $p_path : Path where the files and directories are to be extracted// $p_remove_path : First part ('root' part) of the memorized path// (if any similar) to remove while extracting.// Options :// PCLZIP_OPT_PATH :// PCLZIP_OPT_ADD_PATH :// PCLZIP_OPT_REMOVE_PATH :// PCLZIP_OPT_REMOVE_ALL_PATH :// PCLZIP_CB_PRE_EXTRACT :// PCLZIP_CB_POST_EXTRACT :// Return Values :// 0 or a negative value on failure,// The list of the extracted files, with a status of the action.// (see PclZip::listContent() for list entry format)// --------------------------------------------------------------------------------function extract(){$v_result=1;// ----- Reset the error handler$this->privErrorReset();// ----- Check archiveif (!$this->privCheckFormat()) {return(0);}// ----- Set default values$v_options = array();// $v_path = "./";$v_path = '';$v_remove_path = "";$v_remove_all_path = false;// ----- Look for variable options arguments$v_size = func_num_args();// ----- Default values for option$v_options[PCLZIP_OPT_EXTRACT_AS_STRING] = FALSE;// ----- Look for argumentsif ($v_size > 0) {// ----- Get the arguments$v_arg_list = func_get_args();// ----- Look for first argif ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) {// ----- Parse the options$v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options,array (PCLZIP_OPT_PATH => 'optional',PCLZIP_OPT_REMOVE_PATH => 'optional',PCLZIP_OPT_REMOVE_ALL_PATH => 'optional',PCLZIP_OPT_ADD_PATH => 'optional',PCLZIP_CB_PRE_EXTRACT => 'optional',PCLZIP_CB_POST_EXTRACT => 'optional',PCLZIP_OPT_SET_CHMOD => 'optional',PCLZIP_OPT_BY_NAME => 'optional',PCLZIP_OPT_BY_EREG => 'optional',PCLZIP_OPT_BY_PREG => 'optional',PCLZIP_OPT_BY_INDEX => 'optional',PCLZIP_OPT_EXTRACT_AS_STRING => 'optional',PCLZIP_OPT_EXTRACT_IN_OUTPUT => 'optional',PCLZIP_OPT_REPLACE_NEWER => 'optional',PCLZIP_OPT_STOP_ON_ERROR => 'optional',PCLZIP_OPT_EXTRACT_DIR_RESTRICTION => 'optional',PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional',PCLZIP_OPT_TEMP_FILE_ON => 'optional',PCLZIP_OPT_TEMP_FILE_OFF => 'optional'));if ($v_result != 1) {return 0;}// ----- Set the argumentsif (isset($v_options[PCLZIP_OPT_PATH])) {$v_path = $v_options[PCLZIP_OPT_PATH];}if (isset($v_options[PCLZIP_OPT_REMOVE_PATH])) {$v_remove_path = $v_options[PCLZIP_OPT_REMOVE_PATH];}if (isset($v_options[PCLZIP_OPT_REMOVE_ALL_PATH])) {$v_remove_all_path = $v_options[PCLZIP_OPT_REMOVE_ALL_PATH];}if (isset($v_options[PCLZIP_OPT_ADD_PATH])) {// ----- Check for '/' in last path charif ((strlen($v_path) > 0) && (substr($v_path, -1) != '/')) {$v_path .= '/';}$v_path .= $v_options[PCLZIP_OPT_ADD_PATH];}}// ----- Look for 2 args// Here we need to support the first historic synopsis of the// method.else {// ----- Get the first argument$v_path = $v_arg_list[0];// ----- Look for the optional second argumentif ($v_size == 2) {$v_remove_path = $v_arg_list[1];}else if ($v_size > 2) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid number / type of arguments");// ----- Returnreturn 0;}}}// ----- Look for default option values$this->privOptionDefaultThreshold($v_options);// ----- Trace// ----- Call the extracting fct$p_list = array();$v_result = $this->privExtractByRule($p_list, $v_path, $v_remove_path,$v_remove_all_path, $v_options);if ($v_result < 1) {unset($p_list);return(0);}// ----- Returnreturn $p_list;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function :// extractByIndex($p_index, $p_path="./", $p_remove_path="")// extractByIndex($p_index, [$p_option, $p_option_value, ...])// Description :// This method supports two synopsis. The first one is historical.// This method is doing a partial extract of the archive.// The extracted files or folders are identified by their index in the// archive (from 0 to n).// Note that if the index identify a folder, only the folder entry is// extracted, not all the files included in the archive.// Parameters :// $p_index : A single index (integer) or a string of indexes of files to// extract. The form of the string is "0,4-6,8-12" with only numbers// and '-' for range or ',' to separate ranges. No spaces or ';'// are allowed.// $p_path : Path where the files and directories are to be extracted// $p_remove_path : First part ('root' part) of the memorized path// (if any similar) to remove while extracting.// Options :// PCLZIP_OPT_PATH :// PCLZIP_OPT_ADD_PATH :// PCLZIP_OPT_REMOVE_PATH :// PCLZIP_OPT_REMOVE_ALL_PATH :// PCLZIP_OPT_EXTRACT_AS_STRING : The files are extracted as strings and// not as files.// The resulting content is in a new field 'content' in the file// structure.// This option must be used alone (any other options are ignored).// PCLZIP_CB_PRE_EXTRACT :// PCLZIP_CB_POST_EXTRACT :// Return Values :// 0 on failure,// The list of the extracted files, with a status of the action.// (see PclZip::listContent() for list entry format)// --------------------------------------------------------------------------------//function extractByIndex($p_index, options...)function extractByIndex($p_index){$v_result=1;// ----- Reset the error handler$this->privErrorReset();// ----- Check archiveif (!$this->privCheckFormat()) {return(0);}// ----- Set default values$v_options = array();// $v_path = "./";$v_path = '';$v_remove_path = "";$v_remove_all_path = false;// ----- Look for variable options arguments$v_size = func_num_args();// ----- Default values for option$v_options[PCLZIP_OPT_EXTRACT_AS_STRING] = FALSE;// ----- Look for argumentsif ($v_size > 1) {// ----- Get the arguments$v_arg_list = func_get_args();// ----- Remove form the options list the first argumentarray_shift($v_arg_list);$v_size--;// ----- Look for first argif ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) {// ----- Parse the options$v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options,array (PCLZIP_OPT_PATH => 'optional',PCLZIP_OPT_REMOVE_PATH => 'optional',PCLZIP_OPT_REMOVE_ALL_PATH => 'optional',PCLZIP_OPT_EXTRACT_AS_STRING => 'optional',PCLZIP_OPT_ADD_PATH => 'optional',PCLZIP_CB_PRE_EXTRACT => 'optional',PCLZIP_CB_POST_EXTRACT => 'optional',PCLZIP_OPT_SET_CHMOD => 'optional',PCLZIP_OPT_REPLACE_NEWER => 'optional',PCLZIP_OPT_STOP_ON_ERROR => 'optional',PCLZIP_OPT_EXTRACT_DIR_RESTRICTION => 'optional',PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional',PCLZIP_OPT_TEMP_FILE_ON => 'optional',PCLZIP_OPT_TEMP_FILE_OFF => 'optional'));if ($v_result != 1) {return 0;}// ----- Set the argumentsif (isset($v_options[PCLZIP_OPT_PATH])) {$v_path = $v_options[PCLZIP_OPT_PATH];}if (isset($v_options[PCLZIP_OPT_REMOVE_PATH])) {$v_remove_path = $v_options[PCLZIP_OPT_REMOVE_PATH];}if (isset($v_options[PCLZIP_OPT_REMOVE_ALL_PATH])) {$v_remove_all_path = $v_options[PCLZIP_OPT_REMOVE_ALL_PATH];}if (isset($v_options[PCLZIP_OPT_ADD_PATH])) {// ----- Check for '/' in last path charif ((strlen($v_path) > 0) && (substr($v_path, -1) != '/')) {$v_path .= '/';}$v_path .= $v_options[PCLZIP_OPT_ADD_PATH];}if (!isset($v_options[PCLZIP_OPT_EXTRACT_AS_STRING])) {$v_options[PCLZIP_OPT_EXTRACT_AS_STRING] = FALSE;}else {}}// ----- Look for 2 args// Here we need to support the first historic synopsis of the// method.else {// ----- Get the first argument$v_path = $v_arg_list[0];// ----- Look for the optional second argumentif ($v_size == 2) {$v_remove_path = $v_arg_list[1];}else if ($v_size > 2) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid number / type of arguments");// ----- Returnreturn 0;}}}// ----- Trace// ----- Trick// Here I want to reuse extractByRule(), so I need to parse the $p_index// with privParseOptions()$v_arg_trick = array (PCLZIP_OPT_BY_INDEX, $p_index);$v_options_trick = array();$v_result = $this->privParseOptions($v_arg_trick, sizeof($v_arg_trick), $v_options_trick,array (PCLZIP_OPT_BY_INDEX => 'optional' ));if ($v_result != 1) {return 0;}$v_options[PCLZIP_OPT_BY_INDEX] = $v_options_trick[PCLZIP_OPT_BY_INDEX];// ----- Look for default option values$this->privOptionDefaultThreshold($v_options);// ----- Call the extracting fctif (($v_result = $this->privExtractByRule($p_list, $v_path, $v_remove_path, $v_remove_all_path, $v_options)) < 1) {return(0);}// ----- Returnreturn $p_list;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function :// delete([$p_option, $p_option_value, ...])// Description :// This method removes files from the archive.// If no parameters are given, then all the archive is emptied.// Parameters :// None or optional arguments.// Options :// PCLZIP_OPT_BY_INDEX :// PCLZIP_OPT_BY_NAME :// PCLZIP_OPT_BY_EREG :// PCLZIP_OPT_BY_PREG :// Return Values :// 0 on failure,// The list of the files which are still present in the archive.// (see PclZip::listContent() for list entry format)// --------------------------------------------------------------------------------function delete(){$v_result=1;// ----- Reset the error handler$this->privErrorReset();// ----- Check archiveif (!$this->privCheckFormat()) {return(0);}// ----- Set default values$v_options = array();// ----- Look for variable options arguments$v_size = func_num_args();// ----- Look for argumentsif ($v_size > 0) {// ----- Get the arguments$v_arg_list = func_get_args();// ----- Parse the options$v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options,array (PCLZIP_OPT_BY_NAME => 'optional',PCLZIP_OPT_BY_EREG => 'optional',PCLZIP_OPT_BY_PREG => 'optional',PCLZIP_OPT_BY_INDEX => 'optional' ));if ($v_result != 1) {return 0;}}// ----- Magic quotes trick$this->privDisableMagicQuotes();// ----- Call the delete fct$v_list = array();if (($v_result = $this->privDeleteByRule($v_list, $v_options)) != 1) {$this->privSwapBackMagicQuotes();unset($v_list);return(0);}// ----- Magic quotes trick$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_list;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : deleteByIndex()// Description :// ***** Deprecated *****// delete(PCLZIP_OPT_BY_INDEX, $p_index) should be prefered.// --------------------------------------------------------------------------------function deleteByIndex($p_index){$p_list = $this->delete(PCLZIP_OPT_BY_INDEX, $p_index);// ----- Returnreturn $p_list;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : properties()// Description :// This method gives the properties of the archive.// The properties are :// nb : Number of files in the archive// comment : Comment associated with the archive file// status : not_exist, ok// Parameters :// None// Return Values :// 0 on failure,// An array with the archive properties.// --------------------------------------------------------------------------------function properties(){// ----- Reset the error handler$this->privErrorReset();// ----- Magic quotes trick$this->privDisableMagicQuotes();// ----- Check archiveif (!$this->privCheckFormat()) {$this->privSwapBackMagicQuotes();return(0);}// ----- Default properties$v_prop = array();$v_prop['comment'] = '';$v_prop['nb'] = 0;$v_prop['status'] = 'not_exist';// ----- Look if file existsif (@is_file($this->zipname)){// ----- Open the zip fileif (($this->zip_fd = @fopen($this->zipname, 'rb')) == 0){$this->privSwapBackMagicQuotes();// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive \''.$this->zipname.'\' in binary read mode');// ----- Returnreturn 0;}// ----- Read the central directory informations$v_central_dir = array();if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1){$this->privSwapBackMagicQuotes();return 0;}// ----- Close the zip file$this->privCloseFd();// ----- Set the user attributes$v_prop['comment'] = $v_central_dir['comment'];$v_prop['nb'] = $v_central_dir['entries'];$v_prop['status'] = 'ok';}// ----- Magic quotes trick$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_prop;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : duplicate()// Description :// This method creates an archive by copying the content of an other one. If// the archive already exist, it is replaced by the new one without any warning.// Parameters :// $p_archive : The filename of a valid archive, or// a valid PclZip object.// Return Values :// 1 on success.// 0 or a negative value on error (error code).// --------------------------------------------------------------------------------function duplicate($p_archive){$v_result = 1;// ----- Reset the error handler$this->privErrorReset();// ----- Look if the $p_archive is a PclZip objectif ((is_object($p_archive)) && (get_class($p_archive) == 'pclzip')){// ----- Duplicate the archive$v_result = $this->privDuplicate($p_archive->zipname);}// ----- Look if the $p_archive is a string (so a filename)else if (is_string($p_archive)){// ----- Check that $p_archive is a valid zip file// TBC : Should also check the archive formatif (!is_file($p_archive)) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "No file with filename '".$p_archive."'");$v_result = PCLZIP_ERR_MISSING_FILE;}else {// ----- Duplicate the archive$v_result = $this->privDuplicate($p_archive);}}// ----- Invalid variableelse{// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type p_archive_to_add");$v_result = PCLZIP_ERR_INVALID_PARAMETER;}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : merge()// Description :// This method merge the $p_archive_to_add archive at the end of the current// one ($this).// If the archive ($this) does not exist, the merge becomes a duplicate.// If the $p_archive_to_add archive does not exist, the merge is a success.// Parameters :// $p_archive_to_add : It can be directly the filename of a valid zip archive,// or a PclZip object archive.// Return Values :// 1 on success,// 0 or negative values on error (see below).// --------------------------------------------------------------------------------function merge($p_archive_to_add){$v_result = 1;// ----- Reset the error handler$this->privErrorReset();// ----- Check archiveif (!$this->privCheckFormat()) {return(0);}// ----- Look if the $p_archive_to_add is a PclZip objectif ((is_object($p_archive_to_add)) && (get_class($p_archive_to_add) == 'pclzip')){// ----- Merge the archive$v_result = $this->privMerge($p_archive_to_add);}// ----- Look if the $p_archive_to_add is a string (so a filename)else if (is_string($p_archive_to_add)){// ----- Create a temporary archive$v_object_archive = new PclZip($p_archive_to_add);// ----- Merge the archive$v_result = $this->privMerge($v_object_archive);}// ----- Invalid variableelse{// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type p_archive_to_add");$v_result = PCLZIP_ERR_INVALID_PARAMETER;}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : errorCode()// Description :// Parameters :// --------------------------------------------------------------------------------function errorCode(){if (PCLZIP_ERROR_EXTERNAL == 1) {return(PclErrorCode());}else {return($this->error_code);}}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : errorName()// Description :// Parameters :// --------------------------------------------------------------------------------function errorName($p_with_code=false){$v_name = array ( PCLZIP_ERR_NO_ERROR => 'PCLZIP_ERR_NO_ERROR',PCLZIP_ERR_WRITE_OPEN_FAIL => 'PCLZIP_ERR_WRITE_OPEN_FAIL',PCLZIP_ERR_READ_OPEN_FAIL => 'PCLZIP_ERR_READ_OPEN_FAIL',PCLZIP_ERR_INVALID_PARAMETER => 'PCLZIP_ERR_INVALID_PARAMETER',PCLZIP_ERR_MISSING_FILE => 'PCLZIP_ERR_MISSING_FILE',PCLZIP_ERR_FILENAME_TOO_LONG => 'PCLZIP_ERR_FILENAME_TOO_LONG',PCLZIP_ERR_INVALID_ZIP => 'PCLZIP_ERR_INVALID_ZIP',PCLZIP_ERR_BAD_EXTRACTED_FILE => 'PCLZIP_ERR_BAD_EXTRACTED_FILE',PCLZIP_ERR_DIR_CREATE_FAIL => 'PCLZIP_ERR_DIR_CREATE_FAIL',PCLZIP_ERR_BAD_EXTENSION => 'PCLZIP_ERR_BAD_EXTENSION',PCLZIP_ERR_BAD_FORMAT => 'PCLZIP_ERR_BAD_FORMAT',PCLZIP_ERR_DELETE_FILE_FAIL => 'PCLZIP_ERR_DELETE_FILE_FAIL',PCLZIP_ERR_RENAME_FILE_FAIL => 'PCLZIP_ERR_RENAME_FILE_FAIL',PCLZIP_ERR_BAD_CHECKSUM => 'PCLZIP_ERR_BAD_CHECKSUM',PCLZIP_ERR_INVALID_ARCHIVE_ZIP => 'PCLZIP_ERR_INVALID_ARCHIVE_ZIP',PCLZIP_ERR_MISSING_OPTION_VALUE => 'PCLZIP_ERR_MISSING_OPTION_VALUE',PCLZIP_ERR_INVALID_OPTION_VALUE => 'PCLZIP_ERR_INVALID_OPTION_VALUE',PCLZIP_ERR_UNSUPPORTED_COMPRESSION => 'PCLZIP_ERR_UNSUPPORTED_COMPRESSION',PCLZIP_ERR_UNSUPPORTED_ENCRYPTION => 'PCLZIP_ERR_UNSUPPORTED_ENCRYPTION',PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE => 'PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE',PCLZIP_ERR_DIRECTORY_RESTRICTION => 'PCLZIP_ERR_DIRECTORY_RESTRICTION');if (isset($v_name[$this->error_code])) {$v_value = $v_name[$this->error_code];}else {$v_value = 'NoName';}if ($p_with_code) {return($v_value.' ('.$this->error_code.')');}else {return($v_value);}}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : errorInfo()// Description :// Parameters :// --------------------------------------------------------------------------------function errorInfo($p_full=false){if (PCLZIP_ERROR_EXTERNAL == 1) {return(PclErrorString());}else {if ($p_full) {return($this->errorName(true)." : ".$this->error_string);}else {return($this->error_string." [code ".$this->error_code."]");}}}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// ***** UNDER THIS LINE ARE DEFINED PRIVATE INTERNAL FUNCTIONS *****// ***** *****// ***** THESES FUNCTIONS MUST NOT BE USED DIRECTLY *****// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privCheckFormat()// Description :// This method check that the archive exists and is a valid zip archive.// Several level of check exists. (futur)// Parameters :// $p_level : Level of check. Default 0.// 0 : Check the first bytes (magic codes) (default value))// 1 : 0 + Check the central directory (futur)// 2 : 1 + Check each file header (futur)// Return Values :// true on success,// false on error, the error code is set.// --------------------------------------------------------------------------------function privCheckFormat($p_level=0){$v_result = true;// ----- Reset the file system cacheclearstatcache();// ----- Reset the error handler$this->privErrorReset();// ----- Look if the file exitsif (!is_file($this->zipname)) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "Missing archive file '".$this->zipname."'");return(false);}// ----- Check that the file is readeableif (!is_readable($this->zipname)) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, "Unable to read archive '".$this->zipname."'");return(false);}// ----- Check the magic code// TBC// ----- Check the central header// TBC// ----- Check each file header// TBC// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privParseOptions()// Description :// This internal methods reads the variable list of arguments ($p_options_list,// $p_size) and generate an array with the options and values ($v_result_list).// $v_requested_options contains the options that can be present and those that// must be present.// $v_requested_options is an array, with the option value as key, and 'optional',// or 'mandatory' as value.// Parameters :// See above.// Return Values :// 1 on success.// 0 on failure.// --------------------------------------------------------------------------------function privParseOptions(&$p_options_list, $p_size, &$v_result_list, $v_requested_options=false){$v_result=1;// ----- Read the options$i=0;while ($i<$p_size) {// ----- Check if the option is supportedif (!isset($v_requested_options[$p_options_list[$i]])) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid optional parameter '".$p_options_list[$i]."' for this method");// ----- Returnreturn PclZip::errorCode();}// ----- Look for next optionswitch ($p_options_list[$i]) {// ----- Look for options that request a path valuecase PCLZIP_OPT_PATH :case PCLZIP_OPT_REMOVE_PATH :case PCLZIP_OPT_ADD_PATH :// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the value$v_result_list[$p_options_list[$i]] = PclZipUtilTranslateWinPath($p_options_list[$i+1], FALSE);$i++;break;case PCLZIP_OPT_TEMP_FILE_THRESHOLD :// ----- Check the number of parametersif (($i+1) >= $p_size) {PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");return PclZip::errorCode();}// ----- Check for incompatible optionsif (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_OFF])) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '".PclZipUtilOptionText($p_options_list[$i])."' can not be used with option 'PCLZIP_OPT_TEMP_FILE_OFF'");return PclZip::errorCode();}// ----- Check the value$v_value = $p_options_list[$i+1];if ((!is_integer($v_value)) || ($v_value<0)) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Integer expected for option '".PclZipUtilOptionText($p_options_list[$i])."'");return PclZip::errorCode();}// ----- Get the value (and convert it in bytes)$v_result_list[$p_options_list[$i]] = $v_value*1048576;$i++;break;case PCLZIP_OPT_TEMP_FILE_ON :// ----- Check for incompatible optionsif (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_OFF])) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '".PclZipUtilOptionText($p_options_list[$i])."' can not be used with option 'PCLZIP_OPT_TEMP_FILE_OFF'");return PclZip::errorCode();}$v_result_list[$p_options_list[$i]] = true;break;case PCLZIP_OPT_TEMP_FILE_OFF :// ----- Check for incompatible optionsif (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_ON])) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '".PclZipUtilOptionText($p_options_list[$i])."' can not be used with option 'PCLZIP_OPT_TEMP_FILE_ON'");return PclZip::errorCode();}// ----- Check for incompatible optionsif (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_THRESHOLD])) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '".PclZipUtilOptionText($p_options_list[$i])."' can not be used with option 'PCLZIP_OPT_TEMP_FILE_THRESHOLD'");return PclZip::errorCode();}$v_result_list[$p_options_list[$i]] = true;break;case PCLZIP_OPT_EXTRACT_DIR_RESTRICTION :// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the valueif ( is_string($p_options_list[$i+1])&& ($p_options_list[$i+1] != '')) {$v_result_list[$p_options_list[$i]] = PclZipUtilTranslateWinPath($p_options_list[$i+1], FALSE);$i++;}else {}break;// ----- Look for options that request an array of string for valuecase PCLZIP_OPT_BY_NAME :// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the valueif (is_string($p_options_list[$i+1])) {$v_result_list[$p_options_list[$i]][0] = $p_options_list[$i+1];}else if (is_array($p_options_list[$i+1])) {$v_result_list[$p_options_list[$i]] = $p_options_list[$i+1];}else {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Wrong parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}$i++;break;// ----- Look for options that request an EREG or PREG expressioncase PCLZIP_OPT_BY_EREG :// ereg() is deprecated starting with PHP 5.3. Move PCLZIP_OPT_BY_EREG// to PCLZIP_OPT_BY_PREG$p_options_list[$i] = PCLZIP_OPT_BY_PREG;case PCLZIP_OPT_BY_PREG ://case PCLZIP_OPT_CRYPT :// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the valueif (is_string($p_options_list[$i+1])) {$v_result_list[$p_options_list[$i]] = $p_options_list[$i+1];}else {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Wrong parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}$i++;break;// ----- Look for options that takes a stringcase PCLZIP_OPT_COMMENT :case PCLZIP_OPT_ADD_COMMENT :case PCLZIP_OPT_PREPEND_COMMENT :// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE,"Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the valueif (is_string($p_options_list[$i+1])) {$v_result_list[$p_options_list[$i]] = $p_options_list[$i+1];}else {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE,"Wrong parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}$i++;break;// ----- Look for options that request an array of indexcase PCLZIP_OPT_BY_INDEX :// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the value$v_work_list = array();if (is_string($p_options_list[$i+1])) {// ----- Remove spaces$p_options_list[$i+1] = strtr($p_options_list[$i+1], ' ', '');// ----- Parse items$v_work_list = explode(",", $p_options_list[$i+1]);}else if (is_integer($p_options_list[$i+1])) {$v_work_list[0] = $p_options_list[$i+1].'-'.$p_options_list[$i+1];}else if (is_array($p_options_list[$i+1])) {$v_work_list = $p_options_list[$i+1];}else {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Value must be integer, string or array for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Reduce the index list// each index item in the list must be a couple with a start and// an end value : [0,3], [5-5], [8-10], ...// ----- Check the format of each item$v_sort_flag=false;$v_sort_value=0;for ($j=0; $j<sizeof($v_work_list); $j++) {// ----- Explode the item$v_item_list = explode("-", $v_work_list[$j]);$v_size_item_list = sizeof($v_item_list);// ----- TBC : Here we might check that each item is a// real integer ...// ----- Look for single valueif ($v_size_item_list == 1) {// ----- Set the option value$v_result_list[$p_options_list[$i]][$j]['start'] = $v_item_list[0];$v_result_list[$p_options_list[$i]][$j]['end'] = $v_item_list[0];}elseif ($v_size_item_list == 2) {// ----- Set the option value$v_result_list[$p_options_list[$i]][$j]['start'] = $v_item_list[0];$v_result_list[$p_options_list[$i]][$j]['end'] = $v_item_list[1];}else {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Too many values in index range for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Look for list sortif ($v_result_list[$p_options_list[$i]][$j]['start'] < $v_sort_value) {$v_sort_flag=true;// ----- TBC : An automatic sort should be writen ...// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Invalid order of index range for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}$v_sort_value = $v_result_list[$p_options_list[$i]][$j]['start'];}// ----- Sort the itemsif ($v_sort_flag) {// TBC : To Be Completed}// ----- Next option$i++;break;// ----- Look for options that request no valuecase PCLZIP_OPT_REMOVE_ALL_PATH :case PCLZIP_OPT_EXTRACT_AS_STRING :case PCLZIP_OPT_NO_COMPRESSION :case PCLZIP_OPT_EXTRACT_IN_OUTPUT :case PCLZIP_OPT_REPLACE_NEWER :case PCLZIP_OPT_STOP_ON_ERROR :$v_result_list[$p_options_list[$i]] = true;break;// ----- Look for options that request an octal valuecase PCLZIP_OPT_SET_CHMOD :// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the value$v_result_list[$p_options_list[$i]] = $p_options_list[$i+1];$i++;break;// ----- Look for options that request a call-backcase PCLZIP_CB_PRE_EXTRACT :case PCLZIP_CB_POST_EXTRACT :case PCLZIP_CB_PRE_ADD :case PCLZIP_CB_POST_ADD :/* for futur usecase PCLZIP_CB_PRE_DELETE :case PCLZIP_CB_POST_DELETE :case PCLZIP_CB_PRE_LIST :case PCLZIP_CB_POST_LIST :*/// ----- Check the number of parametersif (($i+1) >= $p_size) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Get the value$v_function_name = $p_options_list[$i+1];// ----- Check that the value is a valid existing functionif (!function_exists($v_function_name)) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Function '".$v_function_name."()' is not an existing function for option '".PclZipUtilOptionText($p_options_list[$i])."'");// ----- Returnreturn PclZip::errorCode();}// ----- Set the attribute$v_result_list[$p_options_list[$i]] = $v_function_name;$i++;break;default :// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER,"Unknown parameter '".$p_options_list[$i]."'");// ----- Returnreturn PclZip::errorCode();}// ----- Next options$i++;}// ----- Look for mandatory optionsif ($v_requested_options !== false) {for ($key=reset($v_requested_options); $key=key($v_requested_options); $key=next($v_requested_options)) {// ----- Look for mandatory optionif ($v_requested_options[$key] == 'mandatory') {// ----- Look if presentif (!isset($v_result_list[$key])) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Missing mandatory parameter ".PclZipUtilOptionText($key)."(".$key.")");// ----- Returnreturn PclZip::errorCode();}}}}// ----- Look for default valuesif (!isset($v_result_list[PCLZIP_OPT_TEMP_FILE_THRESHOLD])) {}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privOptionDefaultThreshold()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privOptionDefaultThreshold(&$p_options){$v_result=1;if (isset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD])|| isset($p_options[PCLZIP_OPT_TEMP_FILE_OFF])) {return $v_result;}// ----- Get 'memory_limit' configuration value$v_memory_limit = ini_get('memory_limit');$v_memory_limit = trim($v_memory_limit);$last = strtolower(substr($v_memory_limit, -1));if($last == 'g')//$v_memory_limit = $v_memory_limit*1024*1024*1024;$v_memory_limit = $v_memory_limit*1073741824;if($last == 'm')//$v_memory_limit = $v_memory_limit*1024*1024;$v_memory_limit = $v_memory_limit*1048576;if($last == 'k')$v_memory_limit = $v_memory_limit*1024;$p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] = floor($v_memory_limit*PCLZIP_TEMPORARY_FILE_RATIO);// ----- Sanity check : No threshold if value lower than 1Mif ($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] < 1048576) {unset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD]);}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privFileDescrParseAtt()// Description :// Parameters :// Return Values :// 1 on success.// 0 on failure.// --------------------------------------------------------------------------------function privFileDescrParseAtt(&$p_file_list, &$p_filedescr, $v_options, $v_requested_options=false){$v_result=1;// ----- For each file in the list check the attributesforeach ($p_file_list as $v_key => $v_value) {// ----- Check if the option is supportedif (!isset($v_requested_options[$v_key])) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid file attribute '".$v_key."' for this file");// ----- Returnreturn PclZip::errorCode();}// ----- Look for attributeswitch ($v_key) {case PCLZIP_ATT_FILE_NAME :if (!is_string($v_value)) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type ".gettype($v_value).". String expected for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}$p_filedescr['filename'] = PclZipUtilPathReduction($v_value);if ($p_filedescr['filename'] == '') {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid empty filename for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}break;case PCLZIP_ATT_FILE_NEW_SHORT_NAME :if (!is_string($v_value)) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type ".gettype($v_value).". String expected for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}$p_filedescr['new_short_name'] = PclZipUtilPathReduction($v_value);if ($p_filedescr['new_short_name'] == '') {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid empty short filename for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}break;case PCLZIP_ATT_FILE_NEW_FULL_NAME :if (!is_string($v_value)) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type ".gettype($v_value).". String expected for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}$p_filedescr['new_full_name'] = PclZipUtilPathReduction($v_value);if ($p_filedescr['new_full_name'] == '') {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid empty full filename for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}break;// ----- Look for options that takes a stringcase PCLZIP_ATT_FILE_COMMENT :if (!is_string($v_value)) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type ".gettype($v_value).". String expected for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}$p_filedescr['comment'] = $v_value;break;case PCLZIP_ATT_FILE_MTIME :if (!is_integer($v_value)) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type ".gettype($v_value).". Integer expected for attribute '".PclZipUtilOptionText($v_key)."'");return PclZip::errorCode();}$p_filedescr['mtime'] = $v_value;break;case PCLZIP_ATT_FILE_CONTENT :$p_filedescr['content'] = $v_value;break;default :// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER,"Unknown parameter '".$v_key."'");// ----- Returnreturn PclZip::errorCode();}// ----- Look for mandatory optionsif ($v_requested_options !== false) {for ($key=reset($v_requested_options); $key=key($v_requested_options); $key=next($v_requested_options)) {// ----- Look for mandatory optionif ($v_requested_options[$key] == 'mandatory') {// ----- Look if presentif (!isset($p_file_list[$key])) {PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Missing mandatory parameter ".PclZipUtilOptionText($key)."(".$key.")");return PclZip::errorCode();}}}}// end foreach}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privFileDescrExpand()// Description :// This method look for each item of the list to see if its a file, a folder// or a string to be added as file. For any other type of files (link, other)// just ignore the item.// Then prepare the information that will be stored for that file.// When its a folder, expand the folder with all the files that are in that// folder (recursively).// Parameters :// Return Values :// 1 on success.// 0 on failure.// --------------------------------------------------------------------------------function privFileDescrExpand(&$p_filedescr_list, &$p_options){$v_result=1;// ----- Create a result list$v_result_list = array();// ----- Look each entryfor ($i=0; $i<sizeof($p_filedescr_list); $i++) {// ----- Get filedescr$v_descr = $p_filedescr_list[$i];// ----- Reduce the filename$v_descr['filename'] = PclZipUtilTranslateWinPath($v_descr['filename'], false);$v_descr['filename'] = PclZipUtilPathReduction($v_descr['filename']);// ----- Look for real file or folderif (file_exists($v_descr['filename'])) {if (@is_file($v_descr['filename'])) {$v_descr['type'] = 'file';}else if (@is_dir($v_descr['filename'])) {$v_descr['type'] = 'folder';}else if (@is_link($v_descr['filename'])) {// skipcontinue;}else {// skipcontinue;}}// ----- Look for string added as fileelse if (isset($v_descr['content'])) {$v_descr['type'] = 'virtual_file';}// ----- Missing fileelse {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "File '".$v_descr['filename']."' does not exist");// ----- Returnreturn PclZip::errorCode();}// ----- Calculate the stored filename$this->privCalculateStoredFilename($v_descr, $p_options);// ----- Add the descriptor in result list$v_result_list[sizeof($v_result_list)] = $v_descr;// ----- Look for folderif ($v_descr['type'] == 'folder') {// ----- List of items in folder$v_dirlist_descr = array();$v_dirlist_nb = 0;if ($v_folder_handler = @opendir($v_descr['filename'])) {while (($v_item_handler = @readdir($v_folder_handler)) !== false) {// ----- Skip '.' and '..'if (($v_item_handler == '.') || ($v_item_handler == '..')) {continue;}// ----- Compose the full filename$v_dirlist_descr[$v_dirlist_nb]['filename'] = $v_descr['filename'].'/'.$v_item_handler;// ----- Look for different stored filename// Because the name of the folder was changed, the name of the// files/sub-folders also changeif (($v_descr['stored_filename'] != $v_descr['filename'])&& (!isset($p_options[PCLZIP_OPT_REMOVE_ALL_PATH]))) {if ($v_descr['stored_filename'] != '') {$v_dirlist_descr[$v_dirlist_nb]['new_full_name'] = $v_descr['stored_filename'].'/'.$v_item_handler;}else {$v_dirlist_descr[$v_dirlist_nb]['new_full_name'] = $v_item_handler;}}$v_dirlist_nb++;}@closedir($v_folder_handler);}else {// TBC : unable to open folder in read mode}// ----- Expand each element of the listif ($v_dirlist_nb != 0) {// ----- Expandif (($v_result = $this->privFileDescrExpand($v_dirlist_descr, $p_options)) != 1) {return $v_result;}// ----- Concat the resulting list$v_result_list = array_merge($v_result_list, $v_dirlist_descr);}else {}// ----- Free local arrayunset($v_dirlist_descr);}}// ----- Get the result list$p_filedescr_list = $v_result_list;// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privCreate()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privCreate($p_filedescr_list, &$p_result_list, &$p_options){$v_result=1;$v_list_detail = array();// ----- Magic quotes trick$this->privDisableMagicQuotes();// ----- Open the file in write modeif (($v_result = $this->privOpenFd('wb')) != 1){// ----- Returnreturn $v_result;}// ----- Add the list of files$v_result = $this->privAddList($p_filedescr_list, $p_result_list, $p_options);// ----- Close$this->privCloseFd();// ----- Magic quotes trick$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privAdd()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privAdd($p_filedescr_list, &$p_result_list, &$p_options){$v_result=1;$v_list_detail = array();// ----- Look if the archive exists or is emptyif ((!is_file($this->zipname)) || (filesize($this->zipname) == 0)){// ----- Do a create$v_result = $this->privCreate($p_filedescr_list, $p_result_list, $p_options);// ----- Returnreturn $v_result;}// ----- Magic quotes trick$this->privDisableMagicQuotes();// ----- Open the zip fileif (($v_result=$this->privOpenFd('rb')) != 1){// ----- Magic quotes trick$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_result;}// ----- Read the central directory informations$v_central_dir = array();if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1){$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result;}// ----- Go to beginning of File@rewind($this->zip_fd);// ----- Creates a temporay file$v_zip_temp_name = PCLZIP_TEMPORARY_DIR.uniqid('pclzip-').'.tmp';// ----- Open the temporary file in write modeif (($v_zip_temp_fd = @fopen($v_zip_temp_name, 'wb')) == 0){$this->privCloseFd();$this->privSwapBackMagicQuotes();PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \''.$v_zip_temp_name.'\' in binary write mode');// ----- Returnreturn PclZip::errorCode();}// ----- Copy the files from the archive to the temporary file// TBC : Here I should better append the file and go back to erase the central dir$v_size = $v_central_dir['offset'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = fread($this->zip_fd, $v_read_size);@fwrite($v_zip_temp_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Swap the file descriptor// Here is a trick : I swap the temporary fd with the zip fd, in order to use// the following methods on the temporary fil and not the real archive$v_swap = $this->zip_fd;$this->zip_fd = $v_zip_temp_fd;$v_zip_temp_fd = $v_swap;// ----- Add the files$v_header_list = array();if (($v_result = $this->privAddFileList($p_filedescr_list, $v_header_list, $p_options)) != 1){fclose($v_zip_temp_fd);$this->privCloseFd();@unlink($v_zip_temp_name);$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_result;}// ----- Store the offset of the central dir$v_offset = @ftell($this->zip_fd);// ----- Copy the block of file headers from the old archive$v_size = $v_central_dir['size'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($v_zip_temp_fd, $v_read_size);@fwrite($this->zip_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Create the Central Dir files headerfor ($i=0, $v_count=0; $i<sizeof($v_header_list); $i++){// ----- Create the file headerif ($v_header_list[$i]['status'] == 'ok') {if (($v_result = $this->privWriteCentralFileHeader($v_header_list[$i])) != 1) {fclose($v_zip_temp_fd);$this->privCloseFd();@unlink($v_zip_temp_name);$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_result;}$v_count++;}// ----- Transform the header to a 'usable' info$this->privConvertHeader2FileInfo($v_header_list[$i], $p_result_list[$i]);}// ----- Zip file comment$v_comment = $v_central_dir['comment'];if (isset($p_options[PCLZIP_OPT_COMMENT])) {$v_comment = $p_options[PCLZIP_OPT_COMMENT];}if (isset($p_options[PCLZIP_OPT_ADD_COMMENT])) {$v_comment = $v_comment.$p_options[PCLZIP_OPT_ADD_COMMENT];}if (isset($p_options[PCLZIP_OPT_PREPEND_COMMENT])) {$v_comment = $p_options[PCLZIP_OPT_PREPEND_COMMENT].$v_comment;}// ----- Calculate the size of the central header$v_size = @ftell($this->zip_fd)-$v_offset;// ----- Create the central dir footerif (($v_result = $this->privWriteCentralHeader($v_count+$v_central_dir['entries'], $v_size, $v_offset, $v_comment)) != 1){// ----- Reset the file listunset($v_header_list);$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_result;}// ----- Swap back the file descriptor$v_swap = $this->zip_fd;$this->zip_fd = $v_zip_temp_fd;$v_zip_temp_fd = $v_swap;// ----- Close$this->privCloseFd();// ----- Close the temporary file@fclose($v_zip_temp_fd);// ----- Magic quotes trick$this->privSwapBackMagicQuotes();// ----- Delete the zip file// TBC : I should test the result ...@unlink($this->zipname);// ----- Rename the temporary file// TBC : I should test the result ...//@rename($v_zip_temp_name, $this->zipname);PclZipUtilRename($v_zip_temp_name, $this->zipname);// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privOpenFd()// Description :// Parameters :// --------------------------------------------------------------------------------function privOpenFd($p_mode){$v_result=1;// ----- Look if already openif ($this->zip_fd != 0){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Zip file \''.$this->zipname.'\' already open');// ----- Returnreturn PclZip::errorCode();}// ----- Open the zip fileif (($this->zip_fd = @fopen($this->zipname, $p_mode)) == 0){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive \''.$this->zipname.'\' in '.$p_mode.' mode');// ----- Returnreturn PclZip::errorCode();}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privCloseFd()// Description :// Parameters :// --------------------------------------------------------------------------------function privCloseFd(){$v_result=1;if ($this->zip_fd != 0)@fclose($this->zip_fd);$this->zip_fd = 0;// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privAddList()// Description :// $p_add_dir and $p_remove_dir will give the ability to memorize a path which is// different from the real path of the file. This is usefull if you want to have PclTar// running in any directory, and memorize relative path from an other directory.// Parameters :// $p_list : An array containing the file or directory names to add in the tar// $p_result_list : list of added files with their properties (specially the status field)// $p_add_dir : Path to add in the filename path archived// $p_remove_dir : Path to remove in the filename path archived// Return Values :// --------------------------------------------------------------------------------// function privAddList($p_list, &$p_result_list, $p_add_dir, $p_remove_dir, $p_remove_all_dir, &$p_options)function privAddList($p_filedescr_list, &$p_result_list, &$p_options){$v_result=1;// ----- Add the files$v_header_list = array();if (($v_result = $this->privAddFileList($p_filedescr_list, $v_header_list, $p_options)) != 1){// ----- Returnreturn $v_result;}// ----- Store the offset of the central dir$v_offset = @ftell($this->zip_fd);// ----- Create the Central Dir files headerfor ($i=0,$v_count=0; $i<sizeof($v_header_list); $i++){// ----- Create the file headerif ($v_header_list[$i]['status'] == 'ok') {if (($v_result = $this->privWriteCentralFileHeader($v_header_list[$i])) != 1) {// ----- Returnreturn $v_result;}$v_count++;}// ----- Transform the header to a 'usable' info$this->privConvertHeader2FileInfo($v_header_list[$i], $p_result_list[$i]);}// ----- Zip file comment$v_comment = '';if (isset($p_options[PCLZIP_OPT_COMMENT])) {$v_comment = $p_options[PCLZIP_OPT_COMMENT];}// ----- Calculate the size of the central header$v_size = @ftell($this->zip_fd)-$v_offset;// ----- Create the central dir footerif (($v_result = $this->privWriteCentralHeader($v_count, $v_size, $v_offset, $v_comment)) != 1){// ----- Reset the file listunset($v_header_list);// ----- Returnreturn $v_result;}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privAddFileList()// Description :// Parameters :// $p_filedescr_list : An array containing the file description// or directory names to add in the zip// $p_result_list : list of added files with their properties (specially the status field)// Return Values :// --------------------------------------------------------------------------------function privAddFileList($p_filedescr_list, &$p_result_list, &$p_options){$v_result=1;$v_header = array();// ----- Recuperate the current number of elt in list$v_nb = sizeof($p_result_list);// ----- Loop on the filesfor ($j=0; ($j<sizeof($p_filedescr_list)) && ($v_result==1); $j++) {// ----- Format the filename$p_filedescr_list[$j]['filename']= PclZipUtilTranslateWinPath($p_filedescr_list[$j]['filename'], false);// ----- Skip empty file names// TBC : Can this be possible ? not checked in DescrParseAtt ?if ($p_filedescr_list[$j]['filename'] == "") {continue;}// ----- Check the filenameif ( ($p_filedescr_list[$j]['type'] != 'virtual_file')&& (!file_exists($p_filedescr_list[$j]['filename']))) {PclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "File '".$p_filedescr_list[$j]['filename']."' does not exist");return PclZip::errorCode();}// ----- Look if it is a file or a dir with no all path remove option// or a dir with all its path removed// if ( (is_file($p_filedescr_list[$j]['filename']))// || ( is_dir($p_filedescr_list[$j]['filename'])if ( ($p_filedescr_list[$j]['type'] == 'file')|| ($p_filedescr_list[$j]['type'] == 'virtual_file')|| ( ($p_filedescr_list[$j]['type'] == 'folder')&& ( !isset($p_options[PCLZIP_OPT_REMOVE_ALL_PATH])|| !$p_options[PCLZIP_OPT_REMOVE_ALL_PATH]))) {// ----- Add the file$v_result = $this->privAddFile($p_filedescr_list[$j], $v_header,$p_options);if ($v_result != 1) {return $v_result;}// ----- Store the file infos$p_result_list[$v_nb++] = $v_header;}}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privAddFile()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privAddFile($p_filedescr, &$p_header, &$p_options){$v_result=1;// ----- Working variable$p_filename = $p_filedescr['filename'];// TBC : Already done in the fileAtt check ... ?if ($p_filename == "") {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid file list parameter (invalid or empty list)");// ----- Returnreturn PclZip::errorCode();}// ----- Look for a stored different filename/* TBC : Removedif (isset($p_filedescr['stored_filename'])) {$v_stored_filename = $p_filedescr['stored_filename'];}else {$v_stored_filename = $p_filedescr['stored_filename'];}*/// ----- Set the file propertiesclearstatcache();$p_header['version'] = 20;$p_header['version_extracted'] = 10;$p_header['flag'] = 0;$p_header['compression'] = 0;$p_header['crc'] = 0;$p_header['compressed_size'] = 0;$p_header['filename_len'] = strlen($p_filename);$p_header['extra_len'] = 0;$p_header['disk'] = 0;$p_header['internal'] = 0;$p_header['offset'] = 0;$p_header['filename'] = $p_filename;// TBC : Removed $p_header['stored_filename'] = $v_stored_filename;$p_header['stored_filename'] = $p_filedescr['stored_filename'];$p_header['extra'] = '';$p_header['status'] = 'ok';$p_header['index'] = -1;// ----- Look for regular fileif ($p_filedescr['type']=='file') {$p_header['external'] = 0x00000000;$p_header['size'] = filesize($p_filename);}// ----- Look for regular folderelse if ($p_filedescr['type']=='folder') {$p_header['external'] = 0x00000010;$p_header['mtime'] = filemtime($p_filename);$p_header['size'] = filesize($p_filename);}// ----- Look for virtual fileelse if ($p_filedescr['type'] == 'virtual_file') {$p_header['external'] = 0x00000000;$p_header['size'] = strlen($p_filedescr['content']);}// ----- Look for filetimeif (isset($p_filedescr['mtime'])) {$p_header['mtime'] = $p_filedescr['mtime'];}else if ($p_filedescr['type'] == 'virtual_file') {$p_header['mtime'] = time();}else {$p_header['mtime'] = filemtime($p_filename);}// ------ Look for file commentif (isset($p_filedescr['comment'])) {$p_header['comment_len'] = strlen($p_filedescr['comment']);$p_header['comment'] = $p_filedescr['comment'];}else {$p_header['comment_len'] = 0;$p_header['comment'] = '';}// ----- Look for pre-add callbackif (isset($p_options[PCLZIP_CB_PRE_ADD])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_header, $v_local_header);// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_PRE_ADD].'(PCLZIP_CB_PRE_ADD, $v_local_header);');$v_result = $p_options[PCLZIP_CB_PRE_ADD](PCLZIP_CB_PRE_ADD, $v_local_header);if ($v_result == 0) {// ----- Change the file status$p_header['status'] = "skipped";$v_result = 1;}// ----- Update the informations// Only some fields can be modifiedif ($p_header['stored_filename'] != $v_local_header['stored_filename']) {$p_header['stored_filename'] = PclZipUtilPathReduction($v_local_header['stored_filename']);}}// ----- Look for empty stored filenameif ($p_header['stored_filename'] == "") {$p_header['status'] = "filtered";}// ----- Check the path lengthif (strlen($p_header['stored_filename']) > 0xFF) {$p_header['status'] = 'filename_too_long';}// ----- Look if no error, or file not skippedif ($p_header['status'] == 'ok') {// ----- Look for a fileif ($p_filedescr['type'] == 'file') {// ----- Look for using temporary file to zipif ( (!isset($p_options[PCLZIP_OPT_TEMP_FILE_OFF]))&& (isset($p_options[PCLZIP_OPT_TEMP_FILE_ON])|| (isset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD])&& ($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] <= $p_header['size'])) ) ) {$v_result = $this->privAddFileUsingTempFile($p_filedescr, $p_header, $p_options);if ($v_result < PCLZIP_ERR_NO_ERROR) {return $v_result;}}// ----- Use "in memory" zip algoelse {// ----- Open the source fileif (($v_file = @fopen($p_filename, "rb")) == 0) {PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, "Unable to open file '$p_filename' in binary read mode");return PclZip::errorCode();}// ----- Read the file content$v_content = @fread($v_file, $p_header['size']);// ----- Close the file@fclose($v_file);// ----- Calculate the CRC$p_header['crc'] = @crc32($v_content);// ----- Look for no compressionif ($p_options[PCLZIP_OPT_NO_COMPRESSION]) {// ----- Set header parameters$p_header['compressed_size'] = $p_header['size'];$p_header['compression'] = 0;}// ----- Look for normal compressionelse {// ----- Compress the content$v_content = @gzdeflate($v_content);// ----- Set header parameters$p_header['compressed_size'] = strlen($v_content);$p_header['compression'] = 8;}// ----- Call the header generationif (($v_result = $this->privWriteFileHeader($p_header)) != 1) {@fclose($v_file);return $v_result;}// ----- Write the compressed (or not) content@fwrite($this->zip_fd, $v_content, $p_header['compressed_size']);}}// ----- Look for a virtual file (a file from string)else if ($p_filedescr['type'] == 'virtual_file') {$v_content = $p_filedescr['content'];// ----- Calculate the CRC$p_header['crc'] = @crc32($v_content);// ----- Look for no compressionif ($p_options[PCLZIP_OPT_NO_COMPRESSION]) {// ----- Set header parameters$p_header['compressed_size'] = $p_header['size'];$p_header['compression'] = 0;}// ----- Look for normal compressionelse {// ----- Compress the content$v_content = @gzdeflate($v_content);// ----- Set header parameters$p_header['compressed_size'] = strlen($v_content);$p_header['compression'] = 8;}// ----- Call the header generationif (($v_result = $this->privWriteFileHeader($p_header)) != 1) {@fclose($v_file);return $v_result;}// ----- Write the compressed (or not) content@fwrite($this->zip_fd, $v_content, $p_header['compressed_size']);}// ----- Look for a directoryelse if ($p_filedescr['type'] == 'folder') {// ----- Look for directory last '/'if (@substr($p_header['stored_filename'], -1) != '/') {$p_header['stored_filename'] .= '/';}// ----- Set the file properties$p_header['size'] = 0;//$p_header['external'] = 0x41FF0010; // Value for a folder : to be checked$p_header['external'] = 0x00000010; // Value for a folder : to be checked// ----- Call the header generationif (($v_result = $this->privWriteFileHeader($p_header)) != 1){return $v_result;}}}// ----- Look for post-add callbackif (isset($p_options[PCLZIP_CB_POST_ADD])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_header, $v_local_header);// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_POST_ADD].'(PCLZIP_CB_POST_ADD, $v_local_header);');$v_result = $p_options[PCLZIP_CB_POST_ADD](PCLZIP_CB_POST_ADD, $v_local_header);if ($v_result == 0) {// ----- Ignored$v_result = 1;}// ----- Update the informations// Nothing can be modified}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privAddFileUsingTempFile()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privAddFileUsingTempFile($p_filedescr, &$p_header, &$p_options){$v_result=PCLZIP_ERR_NO_ERROR;// ----- Working variable$p_filename = $p_filedescr['filename'];// ----- Open the source fileif (($v_file = @fopen($p_filename, "rb")) == 0) {PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, "Unable to open file '$p_filename' in binary read mode");return PclZip::errorCode();}// ----- Creates a compressed temporary file$v_gzip_temp_name = PCLZIP_TEMPORARY_DIR.uniqid('pclzip-').'.gz';if (($v_file_compressed = @gzopen($v_gzip_temp_name, "wb")) == 0) {fclose($v_file);PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL, 'Unable to open temporary file \''.$v_gzip_temp_name.'\' in binary write mode');return PclZip::errorCode();}// ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks$v_size = filesize($p_filename);while ($v_size != 0) {$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($v_file, $v_read_size);//$v_binary_data = pack('a'.$v_read_size, $v_buffer);@gzputs($v_file_compressed, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Close the file@fclose($v_file);@gzclose($v_file_compressed);// ----- Check the minimum file sizeif (filesize($v_gzip_temp_name) < 18) {PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'gzip temporary file \''.$v_gzip_temp_name.'\' has invalid filesize - should be minimum 18 bytes');return PclZip::errorCode();}// ----- Extract the compressed attributesif (($v_file_compressed = @fopen($v_gzip_temp_name, "rb")) == 0) {PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \''.$v_gzip_temp_name.'\' in binary read mode');return PclZip::errorCode();}// ----- Read the gzip file header$v_binary_data = @fread($v_file_compressed, 10);$v_data_header = unpack('a1id1/a1id2/a1cm/a1flag/Vmtime/a1xfl/a1os', $v_binary_data);// ----- Check some parameters$v_data_header['os'] = bin2hex($v_data_header['os']);// ----- Read the gzip file footer@fseek($v_file_compressed, filesize($v_gzip_temp_name)-8);$v_binary_data = @fread($v_file_compressed, 8);$v_data_footer = unpack('Vcrc/Vcompressed_size', $v_binary_data);// ----- Set the attributes$p_header['compression'] = ord($v_data_header['cm']);//$p_header['mtime'] = $v_data_header['mtime'];$p_header['crc'] = $v_data_footer['crc'];$p_header['compressed_size'] = filesize($v_gzip_temp_name)-18;// ----- Close the file@fclose($v_file_compressed);// ----- Call the header generationif (($v_result = $this->privWriteFileHeader($p_header)) != 1) {return $v_result;}// ----- Add the compressed dataif (($v_file_compressed = @fopen($v_gzip_temp_name, "rb")) == 0){PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \''.$v_gzip_temp_name.'\' in binary read mode');return PclZip::errorCode();}// ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocksfseek($v_file_compressed, 10);$v_size = $p_header['compressed_size'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($v_file_compressed, $v_read_size);//$v_binary_data = pack('a'.$v_read_size, $v_buffer);@fwrite($this->zip_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Close the file@fclose($v_file_compressed);// ----- Unlink the temporary file@unlink($v_gzip_temp_name);// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privCalculateStoredFilename()// Description :// Based on file descriptor properties and global options, this method// calculate the filename that will be stored in the archive.// Parameters :// Return Values :// --------------------------------------------------------------------------------function privCalculateStoredFilename(&$p_filedescr, &$p_options){$v_result=1;// ----- Working variables$p_filename = $p_filedescr['filename'];if (isset($p_options[PCLZIP_OPT_ADD_PATH])) {$p_add_dir = $p_options[PCLZIP_OPT_ADD_PATH];}else {$p_add_dir = '';}if (isset($p_options[PCLZIP_OPT_REMOVE_PATH])) {$p_remove_dir = $p_options[PCLZIP_OPT_REMOVE_PATH];}else {$p_remove_dir = '';}if (isset($p_options[PCLZIP_OPT_REMOVE_ALL_PATH])) {$p_remove_all_dir = $p_options[PCLZIP_OPT_REMOVE_ALL_PATH];}else {$p_remove_all_dir = 0;}// ----- Look for full name changeif (isset($p_filedescr['new_full_name'])) {// ----- Remove drive letter if any$v_stored_filename = PclZipUtilTranslateWinPath($p_filedescr['new_full_name']);}// ----- Look for path and/or short name changeelse {// ----- Look for short name change// Its when we cahnge just the filename but not the pathif (isset($p_filedescr['new_short_name'])) {$v_path_info = pathinfo($p_filename);$v_dir = '';if ($v_path_info['dirname'] != '') {$v_dir = $v_path_info['dirname'].'/';}$v_stored_filename = $v_dir.$p_filedescr['new_short_name'];}else {// ----- Calculate the stored filename$v_stored_filename = $p_filename;}// ----- Look for all path to removeif ($p_remove_all_dir) {$v_stored_filename = basename($p_filename);}// ----- Look for partial path removeelse if ($p_remove_dir != "") {if (substr($p_remove_dir, -1) != '/')$p_remove_dir .= "/";if ( (substr($p_filename, 0, 2) == "./")|| (substr($p_remove_dir, 0, 2) == "./")) {if ( (substr($p_filename, 0, 2) == "./")&& (substr($p_remove_dir, 0, 2) != "./")) {$p_remove_dir = "./".$p_remove_dir;}if ( (substr($p_filename, 0, 2) != "./")&& (substr($p_remove_dir, 0, 2) == "./")) {$p_remove_dir = substr($p_remove_dir, 2);}}$v_compare = PclZipUtilPathInclusion($p_remove_dir,$v_stored_filename);if ($v_compare > 0) {if ($v_compare == 2) {$v_stored_filename = "";}else {$v_stored_filename = substr($v_stored_filename,strlen($p_remove_dir));}}}// ----- Remove drive letter if any$v_stored_filename = PclZipUtilTranslateWinPath($v_stored_filename);// ----- Look for path to addif ($p_add_dir != "") {if (substr($p_add_dir, -1) == "/")$v_stored_filename = $p_add_dir.$v_stored_filename;else$v_stored_filename = $p_add_dir."/".$v_stored_filename;}}// ----- Filename (reduce the path of stored name)$v_stored_filename = PclZipUtilPathReduction($v_stored_filename);$p_filedescr['stored_filename'] = $v_stored_filename;// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privWriteFileHeader()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privWriteFileHeader(&$p_header){$v_result=1;// ----- Store the offset position of the file$p_header['offset'] = ftell($this->zip_fd);// ----- Transform UNIX mtime to DOS format mdate/mtime$v_date = getdate($p_header['mtime']);$v_mtime = ($v_date['hours']<<11) + ($v_date['minutes']<<5) + $v_date['seconds']/2;$v_mdate = (($v_date['year']-1980)<<9) + ($v_date['mon']<<5) + $v_date['mday'];// ----- Packed data$v_binary_data = pack("VvvvvvVVVvv", 0x04034b50,$p_header['version_extracted'], $p_header['flag'],$p_header['compression'], $v_mtime, $v_mdate,$p_header['crc'], $p_header['compressed_size'],$p_header['size'],strlen($p_header['stored_filename']),$p_header['extra_len']);// ----- Write the first 148 bytes of the header in the archivefputs($this->zip_fd, $v_binary_data, 30);// ----- Write the variable fieldsif (strlen($p_header['stored_filename']) != 0){fputs($this->zip_fd, $p_header['stored_filename'], strlen($p_header['stored_filename']));}if ($p_header['extra_len'] != 0){fputs($this->zip_fd, $p_header['extra'], $p_header['extra_len']);}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privWriteCentralFileHeader()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privWriteCentralFileHeader(&$p_header){$v_result=1;// TBC//for(reset($p_header); $key = key($p_header); next($p_header)) {//}// ----- Transform UNIX mtime to DOS format mdate/mtime$v_date = getdate($p_header['mtime']);$v_mtime = ($v_date['hours']<<11) + ($v_date['minutes']<<5) + $v_date['seconds']/2;$v_mdate = (($v_date['year']-1980)<<9) + ($v_date['mon']<<5) + $v_date['mday'];// ----- Packed data$v_binary_data = pack("VvvvvvvVVVvvvvvVV", 0x02014b50,$p_header['version'], $p_header['version_extracted'],$p_header['flag'], $p_header['compression'],$v_mtime, $v_mdate, $p_header['crc'],$p_header['compressed_size'], $p_header['size'],strlen($p_header['stored_filename']),$p_header['extra_len'], $p_header['comment_len'],$p_header['disk'], $p_header['internal'],$p_header['external'], $p_header['offset']);// ----- Write the 42 bytes of the header in the zip filefputs($this->zip_fd, $v_binary_data, 46);// ----- Write the variable fieldsif (strlen($p_header['stored_filename']) != 0){fputs($this->zip_fd, $p_header['stored_filename'], strlen($p_header['stored_filename']));}if ($p_header['extra_len'] != 0){fputs($this->zip_fd, $p_header['extra'], $p_header['extra_len']);}if ($p_header['comment_len'] != 0){fputs($this->zip_fd, $p_header['comment'], $p_header['comment_len']);}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privWriteCentralHeader()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privWriteCentralHeader($p_nb_entries, $p_size, $p_offset, $p_comment){$v_result=1;// ----- Packed data$v_binary_data = pack("VvvvvVVv", 0x06054b50, 0, 0, $p_nb_entries,$p_nb_entries, $p_size,$p_offset, strlen($p_comment));// ----- Write the 22 bytes of the header in the zip filefputs($this->zip_fd, $v_binary_data, 22);// ----- Write the variable fieldsif (strlen($p_comment) != 0){fputs($this->zip_fd, $p_comment, strlen($p_comment));}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privList()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privList(&$p_list){$v_result=1;// ----- Magic quotes trick$this->privDisableMagicQuotes();// ----- Open the zip fileif (($this->zip_fd = @fopen($this->zipname, 'rb')) == 0){// ----- Magic quotes trick$this->privSwapBackMagicQuotes();// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive \''.$this->zipname.'\' in binary read mode');// ----- Returnreturn PclZip::errorCode();}// ----- Read the central directory informations$v_central_dir = array();if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1){$this->privSwapBackMagicQuotes();return $v_result;}// ----- Go to beginning of Central Dir@rewind($this->zip_fd);if (@fseek($this->zip_fd, $v_central_dir['offset'])){$this->privSwapBackMagicQuotes();// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size');// ----- Returnreturn PclZip::errorCode();}// ----- Read each entryfor ($i=0; $i<$v_central_dir['entries']; $i++){// ----- Read the file headerif (($v_result = $this->privReadCentralFileHeader($v_header)) != 1){$this->privSwapBackMagicQuotes();return $v_result;}$v_header['index'] = $i;// ----- Get the only interesting attributes$this->privConvertHeader2FileInfo($v_header, $p_list[$i]);unset($v_header);}// ----- Close the zip file$this->privCloseFd();// ----- Magic quotes trick$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privConvertHeader2FileInfo()// Description :// This function takes the file informations from the central directory// entries and extract the interesting parameters that will be given back.// The resulting file infos are set in the array $p_info// $p_info['filename'] : Filename with full path. Given by user (add),// extracted in the filesystem (extract).// $p_info['stored_filename'] : Stored filename in the archive.// $p_info['size'] = Size of the file.// $p_info['compressed_size'] = Compressed size of the file.// $p_info['mtime'] = Last modification date of the file.// $p_info['comment'] = Comment associated with the file.// $p_info['folder'] = true/false : indicates if the entry is a folder or not.// $p_info['status'] = status of the action on the file.// $p_info['crc'] = CRC of the file content.// Parameters :// Return Values :// --------------------------------------------------------------------------------function privConvertHeader2FileInfo($p_header, &$p_info){$v_result=1;// ----- Get the interesting attributes$v_temp_path = PclZipUtilPathReduction($p_header['filename']);$p_info['filename'] = $v_temp_path;$v_temp_path = PclZipUtilPathReduction($p_header['stored_filename']);$p_info['stored_filename'] = $v_temp_path;$p_info['size'] = $p_header['size'];$p_info['compressed_size'] = $p_header['compressed_size'];$p_info['mtime'] = $p_header['mtime'];$p_info['comment'] = $p_header['comment'];$p_info['folder'] = (($p_header['external']&0x00000010)==0x00000010);$p_info['index'] = $p_header['index'];$p_info['status'] = $p_header['status'];$p_info['crc'] = $p_header['crc'];// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privExtractByRule()// Description :// Extract a file or directory depending of rules (by index, by name, ...)// Parameters :// $p_file_list : An array where will be placed the properties of each// extracted file// $p_path : Path to add while writing the extracted files// $p_remove_path : Path to remove (from the file memorized path) while writing the// extracted files. If the path does not match the file path,// the file is extracted with its memorized path.// $p_remove_path does not apply to 'list' mode.// $p_path and $p_remove_path are commulative.// Return Values :// 1 on success,0 or less on error (see error code list)// --------------------------------------------------------------------------------function privExtractByRule(&$p_file_list, $p_path, $p_remove_path, $p_remove_all_path, &$p_options){$v_result=1;// ----- Magic quotes trick$this->privDisableMagicQuotes();// ----- Check the pathif ( ($p_path == "")|| ( (substr($p_path, 0, 1) != "/")&& (substr($p_path, 0, 3) != "../")&& (substr($p_path,1,2)!=":/")))$p_path = "./".$p_path;// ----- Reduce the path last (and duplicated) '/'if (($p_path != "./") && ($p_path != "/")){// ----- Look for the path end '/'while (substr($p_path, -1) == "/"){$p_path = substr($p_path, 0, strlen($p_path)-1);}}// ----- Look for path to remove format (should end by /)if (($p_remove_path != "") && (substr($p_remove_path, -1) != '/')){$p_remove_path .= '/';}$p_remove_path_size = strlen($p_remove_path);// ----- Open the zip fileif (($v_result = $this->privOpenFd('rb')) != 1){$this->privSwapBackMagicQuotes();return $v_result;}// ----- Read the central directory informations$v_central_dir = array();if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1){// ----- Close the zip file$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result;}// ----- Start at beginning of Central Dir$v_pos_entry = $v_central_dir['offset'];// ----- Read each entry$j_start = 0;for ($i=0, $v_nb_extracted=0; $i<$v_central_dir['entries']; $i++){// ----- Read next Central dir entry@rewind($this->zip_fd);if (@fseek($this->zip_fd, $v_pos_entry)){// ----- Close the zip file$this->privCloseFd();$this->privSwapBackMagicQuotes();// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size');// ----- Returnreturn PclZip::errorCode();}// ----- Read the file header$v_header = array();if (($v_result = $this->privReadCentralFileHeader($v_header)) != 1){// ----- Close the zip file$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result;}// ----- Store the index$v_header['index'] = $i;// ----- Store the file position$v_pos_entry = ftell($this->zip_fd);// ----- Look for the specific extract rules$v_extract = false;// ----- Look for extract by name ruleif ( (isset($p_options[PCLZIP_OPT_BY_NAME]))&& ($p_options[PCLZIP_OPT_BY_NAME] != 0)) {// ----- Look if the filename is in the listfor ($j=0; ($j<sizeof($p_options[PCLZIP_OPT_BY_NAME])) && (!$v_extract); $j++) {// ----- Look for a directoryif (substr($p_options[PCLZIP_OPT_BY_NAME][$j], -1) == "/") {// ----- Look if the directory is in the filename pathif ( (strlen($v_header['stored_filename']) > strlen($p_options[PCLZIP_OPT_BY_NAME][$j]))&& (substr($v_header['stored_filename'], 0, strlen($p_options[PCLZIP_OPT_BY_NAME][$j])) == $p_options[PCLZIP_OPT_BY_NAME][$j])) {$v_extract = true;}}// ----- Look for a filenameelseif ($v_header['stored_filename'] == $p_options[PCLZIP_OPT_BY_NAME][$j]) {$v_extract = true;}}}// ----- Look for extract by ereg rule// ereg() is deprecated with PHP 5.3/*else if ( (isset($p_options[PCLZIP_OPT_BY_EREG]))&& ($p_options[PCLZIP_OPT_BY_EREG] != "")) {if (ereg($p_options[PCLZIP_OPT_BY_EREG], $v_header['stored_filename'])) {$v_extract = true;}}*/// ----- Look for extract by preg ruleelse if ( (isset($p_options[PCLZIP_OPT_BY_PREG]))&& ($p_options[PCLZIP_OPT_BY_PREG] != "")) {if (preg_match($p_options[PCLZIP_OPT_BY_PREG], $v_header['stored_filename'])) {$v_extract = true;}}// ----- Look for extract by index ruleelse if ( (isset($p_options[PCLZIP_OPT_BY_INDEX]))&& ($p_options[PCLZIP_OPT_BY_INDEX] != 0)) {// ----- Look if the index is in the listfor ($j=$j_start; ($j<sizeof($p_options[PCLZIP_OPT_BY_INDEX])) && (!$v_extract); $j++) {if (($i>=$p_options[PCLZIP_OPT_BY_INDEX][$j]['start']) && ($i<=$p_options[PCLZIP_OPT_BY_INDEX][$j]['end'])) {$v_extract = true;}if ($i>=$p_options[PCLZIP_OPT_BY_INDEX][$j]['end']) {$j_start = $j+1;}if ($p_options[PCLZIP_OPT_BY_INDEX][$j]['start']>$i) {break;}}}// ----- Look for no rule, which means extract all the archiveelse {$v_extract = true;}// ----- Check compression methodif ( ($v_extract)&& ( ($v_header['compression'] != 8)&& ($v_header['compression'] != 0))) {$v_header['status'] = 'unsupported_compression';// ----- Look for PCLZIP_OPT_STOP_ON_ERRORif ( (isset($p_options[PCLZIP_OPT_STOP_ON_ERROR]))&& ($p_options[PCLZIP_OPT_STOP_ON_ERROR]===true)) {$this->privSwapBackMagicQuotes();PclZip::privErrorLog(PCLZIP_ERR_UNSUPPORTED_COMPRESSION,"Filename '".$v_header['stored_filename']."' is "."compressed by an unsupported compression "."method (".$v_header['compression'].") ");return PclZip::errorCode();}}// ----- Check encrypted filesif (($v_extract) && (($v_header['flag'] & 1) == 1)) {$v_header['status'] = 'unsupported_encryption';// ----- Look for PCLZIP_OPT_STOP_ON_ERRORif ( (isset($p_options[PCLZIP_OPT_STOP_ON_ERROR]))&& ($p_options[PCLZIP_OPT_STOP_ON_ERROR]===true)) {$this->privSwapBackMagicQuotes();PclZip::privErrorLog(PCLZIP_ERR_UNSUPPORTED_ENCRYPTION,"Unsupported encryption for "." filename '".$v_header['stored_filename']."'");return PclZip::errorCode();}}// ----- Look for real extractionif (($v_extract) && ($v_header['status'] != 'ok')) {$v_result = $this->privConvertHeader2FileInfo($v_header,$p_file_list[$v_nb_extracted++]);if ($v_result != 1) {$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result;}$v_extract = false;}// ----- Look for real extractionif ($v_extract){// ----- Go to the file position@rewind($this->zip_fd);if (@fseek($this->zip_fd, $v_header['offset'])){// ----- Close the zip file$this->privCloseFd();$this->privSwapBackMagicQuotes();// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size');// ----- Returnreturn PclZip::errorCode();}// ----- Look for extraction as stringif ($p_options[PCLZIP_OPT_EXTRACT_AS_STRING]) {$v_string = '';// ----- Extracting the file$v_result1 = $this->privExtractFileAsString($v_header, $v_string, $p_options);if ($v_result1 < 1) {$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result1;}// ----- Get the only interesting attributesif (($v_result = $this->privConvertHeader2FileInfo($v_header, $p_file_list[$v_nb_extracted])) != 1){// ----- Close the zip file$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result;}// ----- Set the file content$p_file_list[$v_nb_extracted]['content'] = $v_string;// ----- Next extracted file$v_nb_extracted++;// ----- Look for user callback abortif ($v_result1 == 2) {break;}}// ----- Look for extraction in standard outputelseif ( (isset($p_options[PCLZIP_OPT_EXTRACT_IN_OUTPUT]))&& ($p_options[PCLZIP_OPT_EXTRACT_IN_OUTPUT])) {// ----- Extracting the file in standard output$v_result1 = $this->privExtractFileInOutput($v_header, $p_options);if ($v_result1 < 1) {$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result1;}// ----- Get the only interesting attributesif (($v_result = $this->privConvertHeader2FileInfo($v_header, $p_file_list[$v_nb_extracted++])) != 1) {$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result;}// ----- Look for user callback abortif ($v_result1 == 2) {break;}}// ----- Look for normal extractionelse {// ----- Extracting the file$v_result1 = $this->privExtractFile($v_header,$p_path, $p_remove_path,$p_remove_all_path,$p_options);if ($v_result1 < 1) {$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result1;}// ----- Get the only interesting attributesif (($v_result = $this->privConvertHeader2FileInfo($v_header, $p_file_list[$v_nb_extracted++])) != 1){// ----- Close the zip file$this->privCloseFd();$this->privSwapBackMagicQuotes();return $v_result;}// ----- Look for user callback abortif ($v_result1 == 2) {break;}}}}// ----- Close the zip file$this->privCloseFd();$this->privSwapBackMagicQuotes();// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privExtractFile()// Description :// Parameters :// Return Values ://// 1 : ... ?// PCLZIP_ERR_USER_ABORTED(2) : User ask for extraction stop in callback// --------------------------------------------------------------------------------function privExtractFile(&$p_entry, $p_path, $p_remove_path, $p_remove_all_path, &$p_options){$v_result=1;// ----- Read the file headerif (($v_result = $this->privReadFileHeader($v_header)) != 1){// ----- Returnreturn $v_result;}// ----- Check that the file header is coherent with $p_entry infoif ($this->privCheckFileHeaders($v_header, $p_entry) != 1) {// TBC}// ----- Look for all path to removeif ($p_remove_all_path == true) {// ----- Look for folder entry that not need to be extractedif (($p_entry['external']&0x00000010)==0x00000010) {$p_entry['status'] = "filtered";return $v_result;}// ----- Get the basename of the path$p_entry['filename'] = basename($p_entry['filename']);}// ----- Look for path to removeelse if ($p_remove_path != ""){if (PclZipUtilPathInclusion($p_remove_path, $p_entry['filename']) == 2){// ----- Change the file status$p_entry['status'] = "filtered";// ----- Returnreturn $v_result;}$p_remove_path_size = strlen($p_remove_path);if (substr($p_entry['filename'], 0, $p_remove_path_size) == $p_remove_path){// ----- Remove the path$p_entry['filename'] = substr($p_entry['filename'], $p_remove_path_size);}}// ----- Add the pathif ($p_path != '') {$p_entry['filename'] = $p_path."/".$p_entry['filename'];}// ----- Check a base_dir_restrictionif (isset($p_options[PCLZIP_OPT_EXTRACT_DIR_RESTRICTION])) {$v_inclusion= PclZipUtilPathInclusion($p_options[PCLZIP_OPT_EXTRACT_DIR_RESTRICTION],$p_entry['filename']);if ($v_inclusion == 0) {PclZip::privErrorLog(PCLZIP_ERR_DIRECTORY_RESTRICTION,"Filename '".$p_entry['filename']."' is "."outside PCLZIP_OPT_EXTRACT_DIR_RESTRICTION");return PclZip::errorCode();}}// ----- Look for pre-extract callbackif (isset($p_options[PCLZIP_CB_PRE_EXTRACT])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_entry, $v_local_header);// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_PRE_EXTRACT].'(PCLZIP_CB_PRE_EXTRACT, $v_local_header);');$v_result = $p_options[PCLZIP_CB_PRE_EXTRACT](PCLZIP_CB_PRE_EXTRACT, $v_local_header);if ($v_result == 0) {// ----- Change the file status$p_entry['status'] = "skipped";$v_result = 1;}// ----- Look for abort resultif ($v_result == 2) {// ----- This status is internal and will be changed in 'skipped'$p_entry['status'] = "aborted";$v_result = PCLZIP_ERR_USER_ABORTED;}// ----- Update the informations// Only some fields can be modified$p_entry['filename'] = $v_local_header['filename'];}// ----- Look if extraction should be doneif ($p_entry['status'] == 'ok') {// ----- Look for specific actions while the file existif (file_exists($p_entry['filename'])){// ----- Look if file is a directoryif (is_dir($p_entry['filename'])){// ----- Change the file status$p_entry['status'] = "already_a_directory";// ----- Look for PCLZIP_OPT_STOP_ON_ERROR// For historical reason first PclZip implementation does not stop// when this kind of error occurs.if ( (isset($p_options[PCLZIP_OPT_STOP_ON_ERROR]))&& ($p_options[PCLZIP_OPT_STOP_ON_ERROR]===true)) {PclZip::privErrorLog(PCLZIP_ERR_ALREADY_A_DIRECTORY,"Filename '".$p_entry['filename']."' is "."already used by an existing directory");return PclZip::errorCode();}}// ----- Look if file is write protectedelse if (!is_writeable($p_entry['filename'])){// ----- Change the file status$p_entry['status'] = "write_protected";// ----- Look for PCLZIP_OPT_STOP_ON_ERROR// For historical reason first PclZip implementation does not stop// when this kind of error occurs.if ( (isset($p_options[PCLZIP_OPT_STOP_ON_ERROR]))&& ($p_options[PCLZIP_OPT_STOP_ON_ERROR]===true)) {PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL,"Filename '".$p_entry['filename']."' exists "."and is write protected");return PclZip::errorCode();}}// ----- Look if the extracted file is olderelse if (filemtime($p_entry['filename']) > $p_entry['mtime']){// ----- Change the file statusif ( (isset($p_options[PCLZIP_OPT_REPLACE_NEWER]))&& ($p_options[PCLZIP_OPT_REPLACE_NEWER]===true)) {}else {$p_entry['status'] = "newer_exist";// ----- Look for PCLZIP_OPT_STOP_ON_ERROR// For historical reason first PclZip implementation does not stop// when this kind of error occurs.if ( (isset($p_options[PCLZIP_OPT_STOP_ON_ERROR]))&& ($p_options[PCLZIP_OPT_STOP_ON_ERROR]===true)) {PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL,"Newer version of '".$p_entry['filename']."' exists "."and option PCLZIP_OPT_REPLACE_NEWER is not selected");return PclZip::errorCode();}}}else {}}// ----- Check the directory availability and create it if necessaryelse {if ((($p_entry['external']&0x00000010)==0x00000010) || (substr($p_entry['filename'], -1) == '/'))$v_dir_to_check = $p_entry['filename'];else if (!strstr($p_entry['filename'], "/"))$v_dir_to_check = "";else$v_dir_to_check = dirname($p_entry['filename']);if (($v_result = $this->privDirCheck($v_dir_to_check, (($p_entry['external']&0x00000010)==0x00000010))) != 1) {// ----- Change the file status$p_entry['status'] = "path_creation_fail";// ----- Return//return $v_result;$v_result = 1;}}}// ----- Look if extraction should be doneif ($p_entry['status'] == 'ok') {// ----- Do the extraction (if not a folder)if (!(($p_entry['external']&0x00000010)==0x00000010)){// ----- Look for not compressed fileif ($p_entry['compression'] == 0) {// ----- Opening destination fileif (($v_dest_file = @fopen($p_entry['filename'], 'wb')) == 0){// ----- Change the file status$p_entry['status'] = "write_error";// ----- Returnreturn $v_result;}// ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks$v_size = $p_entry['compressed_size'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($this->zip_fd, $v_read_size);/* Try to speed up the code$v_binary_data = pack('a'.$v_read_size, $v_buffer);@fwrite($v_dest_file, $v_binary_data, $v_read_size);*/@fwrite($v_dest_file, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Closing the destination filefclose($v_dest_file);// ----- Change the file mtimetouch($p_entry['filename'], $p_entry['mtime']);}else {// ----- TBC// Need to be finishedif (($p_entry['flag'] & 1) == 1) {PclZip::privErrorLog(PCLZIP_ERR_UNSUPPORTED_ENCRYPTION, 'File \''.$p_entry['filename'].'\' is encrypted. Encrypted files are not supported.');return PclZip::errorCode();}// ----- Look for using temporary file to unzipif ( (!isset($p_options[PCLZIP_OPT_TEMP_FILE_OFF]))&& (isset($p_options[PCLZIP_OPT_TEMP_FILE_ON])|| (isset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD])&& ($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] <= $p_entry['size'])) ) ) {$v_result = $this->privExtractFileUsingTempFile($p_entry, $p_options);if ($v_result < PCLZIP_ERR_NO_ERROR) {return $v_result;}}// ----- Look for extract in memoryelse {// ----- Read the compressed file in a buffer (one shot)$v_buffer = @fread($this->zip_fd, $p_entry['compressed_size']);// ----- Decompress the file$v_file_content = @gzinflate($v_buffer);unset($v_buffer);if ($v_file_content === FALSE) {// ----- Change the file status// TBC$p_entry['status'] = "error";return $v_result;}// ----- Opening destination fileif (($v_dest_file = @fopen($p_entry['filename'], 'wb')) == 0) {// ----- Change the file status$p_entry['status'] = "write_error";return $v_result;}// ----- Write the uncompressed data@fwrite($v_dest_file, $v_file_content, $p_entry['size']);unset($v_file_content);// ----- Closing the destination file@fclose($v_dest_file);}// ----- Change the file mtime@touch($p_entry['filename'], $p_entry['mtime']);}// ----- Look for chmod optionif (isset($p_options[PCLZIP_OPT_SET_CHMOD])) {// ----- Change the mode of the file@chmod($p_entry['filename'], $p_options[PCLZIP_OPT_SET_CHMOD]);}}}// ----- Change abort statusif ($p_entry['status'] == "aborted") {$p_entry['status'] = "skipped";}// ----- Look for post-extract callbackelseif (isset($p_options[PCLZIP_CB_POST_EXTRACT])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_entry, $v_local_header);// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_POST_EXTRACT].'(PCLZIP_CB_POST_EXTRACT, $v_local_header);');$v_result = $p_options[PCLZIP_CB_POST_EXTRACT](PCLZIP_CB_POST_EXTRACT, $v_local_header);// ----- Look for abort resultif ($v_result == 2) {$v_result = PCLZIP_ERR_USER_ABORTED;}}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privExtractFileUsingTempFile()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privExtractFileUsingTempFile(&$p_entry, &$p_options){$v_result=1;// ----- Creates a temporary file$v_gzip_temp_name = PCLZIP_TEMPORARY_DIR.uniqid('pclzip-').'.gz';if (($v_dest_file = @fopen($v_gzip_temp_name, "wb")) == 0) {fclose($v_file);PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL, 'Unable to open temporary file \''.$v_gzip_temp_name.'\' in binary write mode');return PclZip::errorCode();}// ----- Write gz file format header$v_binary_data = pack('va1a1Va1a1', 0x8b1f, Chr($p_entry['compression']), Chr(0x00), time(), Chr(0x00), Chr(3));@fwrite($v_dest_file, $v_binary_data, 10);// ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks$v_size = $p_entry['compressed_size'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($this->zip_fd, $v_read_size);//$v_binary_data = pack('a'.$v_read_size, $v_buffer);@fwrite($v_dest_file, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Write gz file format footer$v_binary_data = pack('VV', $p_entry['crc'], $p_entry['size']);@fwrite($v_dest_file, $v_binary_data, 8);// ----- Close the temporary file@fclose($v_dest_file);// ----- Opening destination fileif (($v_dest_file = @fopen($p_entry['filename'], 'wb')) == 0) {$p_entry['status'] = "write_error";return $v_result;}// ----- Open the temporary gz fileif (($v_src_file = @gzopen($v_gzip_temp_name, 'rb')) == 0) {@fclose($v_dest_file);$p_entry['status'] = "read_error";PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \''.$v_gzip_temp_name.'\' in binary read mode');return PclZip::errorCode();}// ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks$v_size = $p_entry['size'];while ($v_size != 0) {$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @gzread($v_src_file, $v_read_size);//$v_binary_data = pack('a'.$v_read_size, $v_buffer);@fwrite($v_dest_file, $v_buffer, $v_read_size);$v_size -= $v_read_size;}@fclose($v_dest_file);@gzclose($v_src_file);// ----- Delete the temporary file@unlink($v_gzip_temp_name);// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privExtractFileInOutput()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privExtractFileInOutput(&$p_entry, &$p_options){$v_result=1;// ----- Read the file headerif (($v_result = $this->privReadFileHeader($v_header)) != 1) {return $v_result;}// ----- Check that the file header is coherent with $p_entry infoif ($this->privCheckFileHeaders($v_header, $p_entry) != 1) {// TBC}// ----- Look for pre-extract callbackif (isset($p_options[PCLZIP_CB_PRE_EXTRACT])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_entry, $v_local_header);// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_PRE_EXTRACT].'(PCLZIP_CB_PRE_EXTRACT, $v_local_header);');$v_result = $p_options[PCLZIP_CB_PRE_EXTRACT](PCLZIP_CB_PRE_EXTRACT, $v_local_header);if ($v_result == 0) {// ----- Change the file status$p_entry['status'] = "skipped";$v_result = 1;}// ----- Look for abort resultif ($v_result == 2) {// ----- This status is internal and will be changed in 'skipped'$p_entry['status'] = "aborted";$v_result = PCLZIP_ERR_USER_ABORTED;}// ----- Update the informations// Only some fields can be modified$p_entry['filename'] = $v_local_header['filename'];}// ----- Trace// ----- Look if extraction should be doneif ($p_entry['status'] == 'ok') {// ----- Do the extraction (if not a folder)if (!(($p_entry['external']&0x00000010)==0x00000010)) {// ----- Look for not compressed fileif ($p_entry['compressed_size'] == $p_entry['size']) {// ----- Read the file in a buffer (one shot)$v_buffer = @fread($this->zip_fd, $p_entry['compressed_size']);// ----- Send the file to the outputecho $v_buffer;unset($v_buffer);}else {// ----- Read the compressed file in a buffer (one shot)$v_buffer = @fread($this->zip_fd, $p_entry['compressed_size']);// ----- Decompress the file$v_file_content = gzinflate($v_buffer);unset($v_buffer);// ----- Send the file to the outputecho $v_file_content;unset($v_file_content);}}}// ----- Change abort statusif ($p_entry['status'] == "aborted") {$p_entry['status'] = "skipped";}// ----- Look for post-extract callbackelseif (isset($p_options[PCLZIP_CB_POST_EXTRACT])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_entry, $v_local_header);// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_POST_EXTRACT].'(PCLZIP_CB_POST_EXTRACT, $v_local_header);');$v_result = $p_options[PCLZIP_CB_POST_EXTRACT](PCLZIP_CB_POST_EXTRACT, $v_local_header);// ----- Look for abort resultif ($v_result == 2) {$v_result = PCLZIP_ERR_USER_ABORTED;}}return $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privExtractFileAsString()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privExtractFileAsString(&$p_entry, &$p_string, &$p_options){$v_result=1;// ----- Read the file header$v_header = array();if (($v_result = $this->privReadFileHeader($v_header)) != 1){// ----- Returnreturn $v_result;}// ----- Check that the file header is coherent with $p_entry infoif ($this->privCheckFileHeaders($v_header, $p_entry) != 1) {// TBC}// ----- Look for pre-extract callbackif (isset($p_options[PCLZIP_CB_PRE_EXTRACT])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_entry, $v_local_header);// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_PRE_EXTRACT].'(PCLZIP_CB_PRE_EXTRACT, $v_local_header);');$v_result = $p_options[PCLZIP_CB_PRE_EXTRACT](PCLZIP_CB_PRE_EXTRACT, $v_local_header);if ($v_result == 0) {// ----- Change the file status$p_entry['status'] = "skipped";$v_result = 1;}// ----- Look for abort resultif ($v_result == 2) {// ----- This status is internal and will be changed in 'skipped'$p_entry['status'] = "aborted";$v_result = PCLZIP_ERR_USER_ABORTED;}// ----- Update the informations// Only some fields can be modified$p_entry['filename'] = $v_local_header['filename'];}// ----- Look if extraction should be doneif ($p_entry['status'] == 'ok') {// ----- Do the extraction (if not a folder)if (!(($p_entry['external']&0x00000010)==0x00000010)) {// ----- Look for not compressed file// if ($p_entry['compressed_size'] == $p_entry['size'])if ($p_entry['compression'] == 0) {// ----- Reading the file$p_string = @fread($this->zip_fd, $p_entry['compressed_size']);}else {// ----- Reading the file$v_data = @fread($this->zip_fd, $p_entry['compressed_size']);// ----- Decompress the fileif (($p_string = @gzinflate($v_data)) === FALSE) {// TBC}}// ----- Trace}else {// TBC : error : can not extract a folder in a string}}// ----- Change abort statusif ($p_entry['status'] == "aborted") {$p_entry['status'] = "skipped";}// ----- Look for post-extract callbackelseif (isset($p_options[PCLZIP_CB_POST_EXTRACT])) {// ----- Generate a local information$v_local_header = array();$this->privConvertHeader2FileInfo($p_entry, $v_local_header);// ----- Swap the content to header$v_local_header['content'] = $p_string;$p_string = '';// ----- Call the callback// Here I do not use call_user_func() because I need to send a reference to the// header.// eval('$v_result = '.$p_options[PCLZIP_CB_POST_EXTRACT].'(PCLZIP_CB_POST_EXTRACT, $v_local_header);');$v_result = $p_options[PCLZIP_CB_POST_EXTRACT](PCLZIP_CB_POST_EXTRACT, $v_local_header);// ----- Swap back the content to header$p_string = $v_local_header['content'];unset($v_local_header['content']);// ----- Look for abort resultif ($v_result == 2) {$v_result = PCLZIP_ERR_USER_ABORTED;}}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privReadFileHeader()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privReadFileHeader(&$p_header){$v_result=1;// ----- Read the 4 bytes signature$v_binary_data = @fread($this->zip_fd, 4);$v_data = unpack('Vid', $v_binary_data);// ----- Check signatureif ($v_data['id'] != 0x04034b50){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Invalid archive structure');// ----- Returnreturn PclZip::errorCode();}// ----- Read the first 42 bytes of the header$v_binary_data = fread($this->zip_fd, 26);// ----- Look for invalid block sizeif (strlen($v_binary_data) != 26){$p_header['filename'] = "";$p_header['status'] = "invalid_header";// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Invalid block size : ".strlen($v_binary_data));// ----- Returnreturn PclZip::errorCode();}// ----- Extract the values$v_data = unpack('vversion/vflag/vcompression/vmtime/vmdate/Vcrc/Vcompressed_size/Vsize/vfilename_len/vextra_len', $v_binary_data);// ----- Get filename$p_header['filename'] = fread($this->zip_fd, $v_data['filename_len']);// ----- Get extra_fieldsif ($v_data['extra_len'] != 0) {$p_header['extra'] = fread($this->zip_fd, $v_data['extra_len']);}else {$p_header['extra'] = '';}// ----- Extract properties$p_header['version_extracted'] = $v_data['version'];$p_header['compression'] = $v_data['compression'];$p_header['size'] = $v_data['size'];$p_header['compressed_size'] = $v_data['compressed_size'];$p_header['crc'] = $v_data['crc'];$p_header['flag'] = $v_data['flag'];$p_header['filename_len'] = $v_data['filename_len'];// ----- Recuperate date in UNIX format$p_header['mdate'] = $v_data['mdate'];$p_header['mtime'] = $v_data['mtime'];if ($p_header['mdate'] && $p_header['mtime']){// ----- Extract time$v_hour = ($p_header['mtime'] & 0xF800) >> 11;$v_minute = ($p_header['mtime'] & 0x07E0) >> 5;$v_seconde = ($p_header['mtime'] & 0x001F)*2;// ----- Extract date$v_year = (($p_header['mdate'] & 0xFE00) >> 9) + 1980;$v_month = ($p_header['mdate'] & 0x01E0) >> 5;$v_day = $p_header['mdate'] & 0x001F;// ----- Get UNIX date format$p_header['mtime'] = @mktime($v_hour, $v_minute, $v_seconde, $v_month, $v_day, $v_year);}else{$p_header['mtime'] = time();}// TBC//for(reset($v_data); $key = key($v_data); next($v_data)) {//}// ----- Set the stored filename$p_header['stored_filename'] = $p_header['filename'];// ----- Set the status field$p_header['status'] = "ok";// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privReadCentralFileHeader()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privReadCentralFileHeader(&$p_header){$v_result=1;// ----- Read the 4 bytes signature$v_binary_data = @fread($this->zip_fd, 4);$v_data = unpack('Vid', $v_binary_data);// ----- Check signatureif ($v_data['id'] != 0x02014b50){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Invalid archive structure');// ----- Returnreturn PclZip::errorCode();}// ----- Read the first 42 bytes of the header$v_binary_data = fread($this->zip_fd, 42);// ----- Look for invalid block sizeif (strlen($v_binary_data) != 42){$p_header['filename'] = "";$p_header['status'] = "invalid_header";// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Invalid block size : ".strlen($v_binary_data));// ----- Returnreturn PclZip::errorCode();}// ----- Extract the values$p_header = unpack('vversion/vversion_extracted/vflag/vcompression/vmtime/vmdate/Vcrc/Vcompressed_size/Vsize/vfilename_len/vextra_len/vcomment_len/vdisk/vinternal/Vexternal/Voffset', $v_binary_data);// ----- Get filenameif ($p_header['filename_len'] != 0)$p_header['filename'] = fread($this->zip_fd, $p_header['filename_len']);else$p_header['filename'] = '';// ----- Get extraif ($p_header['extra_len'] != 0)$p_header['extra'] = fread($this->zip_fd, $p_header['extra_len']);else$p_header['extra'] = '';// ----- Get commentif ($p_header['comment_len'] != 0)$p_header['comment'] = fread($this->zip_fd, $p_header['comment_len']);else$p_header['comment'] = '';// ----- Extract properties// ----- Recuperate date in UNIX format//if ($p_header['mdate'] && $p_header['mtime'])// TBC : bug : this was ignoring time with 0/0/0if (1){// ----- Extract time$v_hour = ($p_header['mtime'] & 0xF800) >> 11;$v_minute = ($p_header['mtime'] & 0x07E0) >> 5;$v_seconde = ($p_header['mtime'] & 0x001F)*2;// ----- Extract date$v_year = (($p_header['mdate'] & 0xFE00) >> 9) + 1980;$v_month = ($p_header['mdate'] & 0x01E0) >> 5;$v_day = $p_header['mdate'] & 0x001F;// ----- Get UNIX date format$p_header['mtime'] = @mktime($v_hour, $v_minute, $v_seconde, $v_month, $v_day, $v_year);}else{$p_header['mtime'] = time();}// ----- Set the stored filename$p_header['stored_filename'] = $p_header['filename'];// ----- Set default status to ok$p_header['status'] = 'ok';// ----- Look if it is a directoryif (substr($p_header['filename'], -1) == '/') {//$p_header['external'] = 0x41FF0010;$p_header['external'] = 0x00000010;}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privCheckFileHeaders()// Description :// Parameters :// Return Values :// 1 on success,// 0 on error;// --------------------------------------------------------------------------------function privCheckFileHeaders(&$p_local_header, &$p_central_header){$v_result=1;// ----- Check the static values// TBCif ($p_local_header['filename'] != $p_central_header['filename']) {}if ($p_local_header['version_extracted'] != $p_central_header['version_extracted']) {}if ($p_local_header['flag'] != $p_central_header['flag']) {}if ($p_local_header['compression'] != $p_central_header['compression']) {}if ($p_local_header['mtime'] != $p_central_header['mtime']) {}if ($p_local_header['filename_len'] != $p_central_header['filename_len']) {}// ----- Look for flag bit 3if (($p_local_header['flag'] & 8) == 8) {$p_local_header['size'] = $p_central_header['size'];$p_local_header['compressed_size'] = $p_central_header['compressed_size'];$p_local_header['crc'] = $p_central_header['crc'];}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privReadEndCentralDir()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privReadEndCentralDir(&$p_central_dir){$v_result=1;// ----- Go to the end of the zip file$v_size = filesize($this->zipname);@fseek($this->zip_fd, $v_size);if (@ftell($this->zip_fd) != $v_size){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Unable to go to the end of the archive \''.$this->zipname.'\'');// ----- Returnreturn PclZip::errorCode();}// ----- First try : look if this is an archive with no commentaries (most of the time)// in this case the end of central dir is at 22 bytes of the file end$v_found = 0;if ($v_size > 26) {@fseek($this->zip_fd, $v_size-22);if (($v_pos = @ftell($this->zip_fd)) != ($v_size-22)){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Unable to seek back to the middle of the archive \''.$this->zipname.'\'');// ----- Returnreturn PclZip::errorCode();}// ----- Read for bytes$v_binary_data = @fread($this->zip_fd, 4);$v_data = @unpack('Vid', $v_binary_data);// ----- Check signatureif ($v_data['id'] == 0x06054b50) {$v_found = 1;}$v_pos = ftell($this->zip_fd);}// ----- Go back to the maximum possible size of the Central Dir End Recordif (!$v_found) {$v_maximum_size = 65557; // 0xFFFF + 22;if ($v_maximum_size > $v_size)$v_maximum_size = $v_size;@fseek($this->zip_fd, $v_size-$v_maximum_size);if (@ftell($this->zip_fd) != ($v_size-$v_maximum_size)){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Unable to seek back to the middle of the archive \''.$this->zipname.'\'');// ----- Returnreturn PclZip::errorCode();}// ----- Read byte per byte in order to find the signature$v_pos = ftell($this->zip_fd);$v_bytes = 0x00000000;while ($v_pos < $v_size){// ----- Read a byte$v_byte = @fread($this->zip_fd, 1);// ----- Add the byte//$v_bytes = ($v_bytes << 8) | Ord($v_byte);// Note we mask the old value down such that once shifted we can never end up with more than a 32bit number// Otherwise on systems where we have 64bit integers the check below for the magic number will fail.$v_bytes = ( ($v_bytes & 0xFFFFFF) << 8) | Ord($v_byte);// ----- Compare the bytesif ($v_bytes == 0x504b0506){$v_pos++;break;}$v_pos++;}// ----- Look if not found end of central dirif ($v_pos == $v_size){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Unable to find End of Central Dir Record signature");// ----- Returnreturn PclZip::errorCode();}}// ----- Read the first 18 bytes of the header$v_binary_data = fread($this->zip_fd, 18);// ----- Look for invalid block sizeif (strlen($v_binary_data) != 18){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Invalid End of Central Dir Record size : ".strlen($v_binary_data));// ----- Returnreturn PclZip::errorCode();}// ----- Extract the values$v_data = unpack('vdisk/vdisk_start/vdisk_entries/ventries/Vsize/Voffset/vcomment_size', $v_binary_data);// ----- Check the global sizeif (($v_pos + $v_data['comment_size'] + 18) != $v_size) {// ----- Removed in release 2.2 see readme file// The check of the file size is a little too strict.// Some bugs where found when a zip is encrypted/decrypted with 'crypt'.// While decrypted, zip has training 0 bytesif (0) {// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT,'The central dir is not at the end of the archive.'.' Some trailing bytes exists after the archive.');// ----- Returnreturn PclZip::errorCode();}}// ----- Get commentif ($v_data['comment_size'] != 0) {$p_central_dir['comment'] = fread($this->zip_fd, $v_data['comment_size']);}else$p_central_dir['comment'] = '';$p_central_dir['entries'] = $v_data['entries'];$p_central_dir['disk_entries'] = $v_data['disk_entries'];$p_central_dir['offset'] = $v_data['offset'];$p_central_dir['size'] = $v_data['size'];$p_central_dir['disk'] = $v_data['disk'];$p_central_dir['disk_start'] = $v_data['disk_start'];// TBC//for(reset($p_central_dir); $key = key($p_central_dir); next($p_central_dir)) {//}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privDeleteByRule()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privDeleteByRule(&$p_result_list, &$p_options){$v_result=1;$v_list_detail = array();// ----- Open the zip fileif (($v_result=$this->privOpenFd('rb')) != 1){// ----- Returnreturn $v_result;}// ----- Read the central directory informations$v_central_dir = array();if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1){$this->privCloseFd();return $v_result;}// ----- Go to beginning of File@rewind($this->zip_fd);// ----- Scan all the files// ----- Start at beginning of Central Dir$v_pos_entry = $v_central_dir['offset'];@rewind($this->zip_fd);if (@fseek($this->zip_fd, $v_pos_entry)){// ----- Close the zip file$this->privCloseFd();// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size');// ----- Returnreturn PclZip::errorCode();}// ----- Read each entry$v_header_list = array();$j_start = 0;for ($i=0, $v_nb_extracted=0; $i<$v_central_dir['entries']; $i++){// ----- Read the file header$v_header_list[$v_nb_extracted] = array();if (($v_result = $this->privReadCentralFileHeader($v_header_list[$v_nb_extracted])) != 1){// ----- Close the zip file$this->privCloseFd();return $v_result;}// ----- Store the index$v_header_list[$v_nb_extracted]['index'] = $i;// ----- Look for the specific extract rules$v_found = false;// ----- Look for extract by name ruleif ( (isset($p_options[PCLZIP_OPT_BY_NAME]))&& ($p_options[PCLZIP_OPT_BY_NAME] != 0)) {// ----- Look if the filename is in the listfor ($j=0; ($j<sizeof($p_options[PCLZIP_OPT_BY_NAME])) && (!$v_found); $j++) {// ----- Look for a directoryif (substr($p_options[PCLZIP_OPT_BY_NAME][$j], -1) == "/") {// ----- Look if the directory is in the filename pathif ( (strlen($v_header_list[$v_nb_extracted]['stored_filename']) > strlen($p_options[PCLZIP_OPT_BY_NAME][$j]))&& (substr($v_header_list[$v_nb_extracted]['stored_filename'], 0, strlen($p_options[PCLZIP_OPT_BY_NAME][$j])) == $p_options[PCLZIP_OPT_BY_NAME][$j])) {$v_found = true;}elseif ( (($v_header_list[$v_nb_extracted]['external']&0x00000010)==0x00000010) /* Indicates a folder */&& ($v_header_list[$v_nb_extracted]['stored_filename'].'/' == $p_options[PCLZIP_OPT_BY_NAME][$j])) {$v_found = true;}}// ----- Look for a filenameelseif ($v_header_list[$v_nb_extracted]['stored_filename'] == $p_options[PCLZIP_OPT_BY_NAME][$j]) {$v_found = true;}}}// ----- Look for extract by ereg rule// ereg() is deprecated with PHP 5.3/*else if ( (isset($p_options[PCLZIP_OPT_BY_EREG]))&& ($p_options[PCLZIP_OPT_BY_EREG] != "")) {if (ereg($p_options[PCLZIP_OPT_BY_EREG], $v_header_list[$v_nb_extracted]['stored_filename'])) {$v_found = true;}}*/// ----- Look for extract by preg ruleelse if ( (isset($p_options[PCLZIP_OPT_BY_PREG]))&& ($p_options[PCLZIP_OPT_BY_PREG] != "")) {if (preg_match($p_options[PCLZIP_OPT_BY_PREG], $v_header_list[$v_nb_extracted]['stored_filename'])) {$v_found = true;}}// ----- Look for extract by index ruleelse if ( (isset($p_options[PCLZIP_OPT_BY_INDEX]))&& ($p_options[PCLZIP_OPT_BY_INDEX] != 0)) {// ----- Look if the index is in the listfor ($j=$j_start; ($j<sizeof($p_options[PCLZIP_OPT_BY_INDEX])) && (!$v_found); $j++) {if (($i>=$p_options[PCLZIP_OPT_BY_INDEX][$j]['start']) && ($i<=$p_options[PCLZIP_OPT_BY_INDEX][$j]['end'])) {$v_found = true;}if ($i>=$p_options[PCLZIP_OPT_BY_INDEX][$j]['end']) {$j_start = $j+1;}if ($p_options[PCLZIP_OPT_BY_INDEX][$j]['start']>$i) {break;}}}else {$v_found = true;}// ----- Look for deletionif ($v_found){unset($v_header_list[$v_nb_extracted]);}else{$v_nb_extracted++;}}// ----- Look if something need to be deletedif ($v_nb_extracted > 0) {// ----- Creates a temporay file$v_zip_temp_name = PCLZIP_TEMPORARY_DIR.uniqid('pclzip-').'.tmp';// ----- Creates a temporary zip archive$v_temp_zip = new PclZip($v_zip_temp_name);// ----- Open the temporary zip file in write modeif (($v_result = $v_temp_zip->privOpenFd('wb')) != 1) {$this->privCloseFd();// ----- Returnreturn $v_result;}// ----- Look which file need to be keptfor ($i=0; $i<sizeof($v_header_list); $i++) {// ----- Calculate the position of the header@rewind($this->zip_fd);if (@fseek($this->zip_fd, $v_header_list[$i]['offset'])) {// ----- Close the zip file$this->privCloseFd();$v_temp_zip->privCloseFd();@unlink($v_zip_temp_name);// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size');// ----- Returnreturn PclZip::errorCode();}// ----- Read the file header$v_local_header = array();if (($v_result = $this->privReadFileHeader($v_local_header)) != 1) {// ----- Close the zip file$this->privCloseFd();$v_temp_zip->privCloseFd();@unlink($v_zip_temp_name);// ----- Returnreturn $v_result;}// ----- Check that local file header is same as central file headerif ($this->privCheckFileHeaders($v_local_header,$v_header_list[$i]) != 1) {// TBC}unset($v_local_header);// ----- Write the file headerif (($v_result = $v_temp_zip->privWriteFileHeader($v_header_list[$i])) != 1) {// ----- Close the zip file$this->privCloseFd();$v_temp_zip->privCloseFd();@unlink($v_zip_temp_name);// ----- Returnreturn $v_result;}// ----- Read/write the data blockif (($v_result = PclZipUtilCopyBlock($this->zip_fd, $v_temp_zip->zip_fd, $v_header_list[$i]['compressed_size'])) != 1) {// ----- Close the zip file$this->privCloseFd();$v_temp_zip->privCloseFd();@unlink($v_zip_temp_name);// ----- Returnreturn $v_result;}}// ----- Store the offset of the central dir$v_offset = @ftell($v_temp_zip->zip_fd);// ----- Re-Create the Central Dir files headerfor ($i=0; $i<sizeof($v_header_list); $i++) {// ----- Create the file headerif (($v_result = $v_temp_zip->privWriteCentralFileHeader($v_header_list[$i])) != 1) {$v_temp_zip->privCloseFd();$this->privCloseFd();@unlink($v_zip_temp_name);// ----- Returnreturn $v_result;}// ----- Transform the header to a 'usable' info$v_temp_zip->privConvertHeader2FileInfo($v_header_list[$i], $p_result_list[$i]);}// ----- Zip file comment$v_comment = '';if (isset($p_options[PCLZIP_OPT_COMMENT])) {$v_comment = $p_options[PCLZIP_OPT_COMMENT];}// ----- Calculate the size of the central header$v_size = @ftell($v_temp_zip->zip_fd)-$v_offset;// ----- Create the central dir footerif (($v_result = $v_temp_zip->privWriteCentralHeader(sizeof($v_header_list), $v_size, $v_offset, $v_comment)) != 1) {// ----- Reset the file listunset($v_header_list);$v_temp_zip->privCloseFd();$this->privCloseFd();@unlink($v_zip_temp_name);// ----- Returnreturn $v_result;}// ----- Close$v_temp_zip->privCloseFd();$this->privCloseFd();// ----- Delete the zip file// TBC : I should test the result ...@unlink($this->zipname);// ----- Rename the temporary file// TBC : I should test the result ...//@rename($v_zip_temp_name, $this->zipname);PclZipUtilRename($v_zip_temp_name, $this->zipname);// ----- Destroy the temporary archiveunset($v_temp_zip);}// ----- Remove every files : reset the fileelse if ($v_central_dir['entries'] != 0) {$this->privCloseFd();if (($v_result = $this->privOpenFd('wb')) != 1) {return $v_result;}if (($v_result = $this->privWriteCentralHeader(0, 0, 0, '')) != 1) {return $v_result;}$this->privCloseFd();}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privDirCheck()// Description :// Check if a directory exists, if not it creates it and all the parents directory// which may be useful.// Parameters :// $p_dir : Directory path to check.// Return Values :// 1 : OK// -1 : Unable to create directory// --------------------------------------------------------------------------------function privDirCheck($p_dir, $p_is_dir=false){$v_result = 1;// ----- Remove the final '/'if (($p_is_dir) && (substr($p_dir, -1)=='/')){$p_dir = substr($p_dir, 0, strlen($p_dir)-1);}// ----- Check the directory availabilityif ((is_dir($p_dir)) || ($p_dir == "")){return 1;}// ----- Extract parent directory$p_parent_dir = dirname($p_dir);// ----- Just a checkif ($p_parent_dir != $p_dir){// ----- Look for parent directoryif ($p_parent_dir != ""){if (($v_result = $this->privDirCheck($p_parent_dir)) != 1){return $v_result;}}}// ----- Create the directoryif (!@mkdir($p_dir, 0777)){// ----- Error logPclZip::privErrorLog(PCLZIP_ERR_DIR_CREATE_FAIL, "Unable to create directory '$p_dir'");// ----- Returnreturn PclZip::errorCode();}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privMerge()// Description :// If $p_archive_to_add does not exist, the function exit with a success result.// Parameters :// Return Values :// --------------------------------------------------------------------------------function privMerge(&$p_archive_to_add){$v_result=1;// ----- Look if the archive_to_add existsif (!is_file($p_archive_to_add->zipname)){// ----- Nothing to merge, so merge is a success$v_result = 1;// ----- Returnreturn $v_result;}// ----- Look if the archive existsif (!is_file($this->zipname)){// ----- Do a duplicate$v_result = $this->privDuplicate($p_archive_to_add->zipname);// ----- Returnreturn $v_result;}// ----- Open the zip fileif (($v_result=$this->privOpenFd('rb')) != 1){// ----- Returnreturn $v_result;}// ----- Read the central directory informations$v_central_dir = array();if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1){$this->privCloseFd();return $v_result;}// ----- Go to beginning of File@rewind($this->zip_fd);// ----- Open the archive_to_add fileif (($v_result=$p_archive_to_add->privOpenFd('rb')) != 1){$this->privCloseFd();// ----- Returnreturn $v_result;}// ----- Read the central directory informations$v_central_dir_to_add = array();if (($v_result = $p_archive_to_add->privReadEndCentralDir($v_central_dir_to_add)) != 1){$this->privCloseFd();$p_archive_to_add->privCloseFd();return $v_result;}// ----- Go to beginning of File@rewind($p_archive_to_add->zip_fd);// ----- Creates a temporay file$v_zip_temp_name = PCLZIP_TEMPORARY_DIR.uniqid('pclzip-').'.tmp';// ----- Open the temporary file in write modeif (($v_zip_temp_fd = @fopen($v_zip_temp_name, 'wb')) == 0){$this->privCloseFd();$p_archive_to_add->privCloseFd();PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \''.$v_zip_temp_name.'\' in binary write mode');// ----- Returnreturn PclZip::errorCode();}// ----- Copy the files from the archive to the temporary file// TBC : Here I should better append the file and go back to erase the central dir$v_size = $v_central_dir['offset'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = fread($this->zip_fd, $v_read_size);@fwrite($v_zip_temp_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Copy the files from the archive_to_add into the temporary file$v_size = $v_central_dir_to_add['offset'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = fread($p_archive_to_add->zip_fd, $v_read_size);@fwrite($v_zip_temp_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Store the offset of the central dir$v_offset = @ftell($v_zip_temp_fd);// ----- Copy the block of file headers from the old archive$v_size = $v_central_dir['size'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($this->zip_fd, $v_read_size);@fwrite($v_zip_temp_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Copy the block of file headers from the archive_to_add$v_size = $v_central_dir_to_add['size'];while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($p_archive_to_add->zip_fd, $v_read_size);@fwrite($v_zip_temp_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Merge the file comments$v_comment = $v_central_dir['comment'].' '.$v_central_dir_to_add['comment'];// ----- Calculate the size of the (new) central header$v_size = @ftell($v_zip_temp_fd)-$v_offset;// ----- Swap the file descriptor// Here is a trick : I swap the temporary fd with the zip fd, in order to use// the following methods on the temporary fil and not the real archive fd$v_swap = $this->zip_fd;$this->zip_fd = $v_zip_temp_fd;$v_zip_temp_fd = $v_swap;// ----- Create the central dir footerif (($v_result = $this->privWriteCentralHeader($v_central_dir['entries']+$v_central_dir_to_add['entries'], $v_size, $v_offset, $v_comment)) != 1){$this->privCloseFd();$p_archive_to_add->privCloseFd();@fclose($v_zip_temp_fd);$this->zip_fd = null;// ----- Reset the file listunset($v_header_list);// ----- Returnreturn $v_result;}// ----- Swap back the file descriptor$v_swap = $this->zip_fd;$this->zip_fd = $v_zip_temp_fd;$v_zip_temp_fd = $v_swap;// ----- Close$this->privCloseFd();$p_archive_to_add->privCloseFd();// ----- Close the temporary file@fclose($v_zip_temp_fd);// ----- Delete the zip file// TBC : I should test the result ...@unlink($this->zipname);// ----- Rename the temporary file// TBC : I should test the result ...//@rename($v_zip_temp_name, $this->zipname);PclZipUtilRename($v_zip_temp_name, $this->zipname);// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privDuplicate()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privDuplicate($p_archive_filename){$v_result=1;// ----- Look if the $p_archive_filename existsif (!is_file($p_archive_filename)){// ----- Nothing to duplicate, so duplicate is a success.$v_result = 1;// ----- Returnreturn $v_result;}// ----- Open the zip fileif (($v_result=$this->privOpenFd('wb')) != 1){// ----- Returnreturn $v_result;}// ----- Open the temporary file in write modeif (($v_zip_temp_fd = @fopen($p_archive_filename, 'rb')) == 0){$this->privCloseFd();PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive file \''.$p_archive_filename.'\' in binary write mode');// ----- Returnreturn PclZip::errorCode();}// ----- Copy the files from the archive to the temporary file// TBC : Here I should better append the file and go back to erase the central dir$v_size = filesize($p_archive_filename);while ($v_size != 0){$v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = fread($v_zip_temp_fd, $v_read_size);@fwrite($this->zip_fd, $v_buffer, $v_read_size);$v_size -= $v_read_size;}// ----- Close$this->privCloseFd();// ----- Close the temporary file@fclose($v_zip_temp_fd);// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privErrorLog()// Description :// Parameters :// --------------------------------------------------------------------------------function privErrorLog($p_error_code=0, $p_error_string=''){if (PCLZIP_ERROR_EXTERNAL == 1) {PclError($p_error_code, $p_error_string);}else {$this->error_code = $p_error_code;$this->error_string = $p_error_string;}}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privErrorReset()// Description :// Parameters :// --------------------------------------------------------------------------------function privErrorReset(){if (PCLZIP_ERROR_EXTERNAL == 1) {PclErrorReset();}else {$this->error_code = 0;$this->error_string = '';}}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privDisableMagicQuotes()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privDisableMagicQuotes(){$v_result=1;// ----- Look if function existsif ( (!function_exists("get_magic_quotes_runtime"))|| (!function_exists("set_magic_quotes_runtime"))) {return $v_result;}// ----- Look if already doneif ($this->magic_quotes_status != -1) {return $v_result;}// ----- Get and memorize the magic_quote value$this->magic_quotes_status = @get_magic_quotes_runtime();// ----- Disable magic_quotesif ($this->magic_quotes_status == 1) {@set_magic_quotes_runtime(0);}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : privSwapBackMagicQuotes()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function privSwapBackMagicQuotes(){$v_result=1;// ----- Look if function existsif ( (!function_exists("get_magic_quotes_runtime"))|| (!function_exists("set_magic_quotes_runtime"))) {return $v_result;}// ----- Look if something to doif ($this->magic_quotes_status != -1) {return $v_result;}// ----- Swap back magic_quotesif ($this->magic_quotes_status == 1) {@set_magic_quotes_runtime($this->magic_quotes_status);}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------}// End of class// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : PclZipUtilPathReduction()// Description :// Parameters :// Return Values :// --------------------------------------------------------------------------------function PclZipUtilPathReduction($p_dir){$v_result = "";// ----- Look for not empty pathif ($p_dir != "") {// ----- Explode path by directory names$v_list = explode("/", $p_dir);// ----- Study directories from last to first$v_skip = 0;for ($i=sizeof($v_list)-1; $i>=0; $i--) {// ----- Look for current pathif ($v_list[$i] == ".") {// ----- Ignore this directory// Should be the first $i=0, but no check is done}else if ($v_list[$i] == "..") {$v_skip++;}else if ($v_list[$i] == "") {// ----- First '/' i.e. root slashif ($i == 0) {$v_result = "/".$v_result;if ($v_skip > 0) {// ----- It is an invalid path, so the path is not modified// TBC$v_result = $p_dir;$v_skip = 0;}}// ----- Last '/' i.e. indicates a directoryelse if ($i == (sizeof($v_list)-1)) {$v_result = $v_list[$i];}// ----- Double '/' inside the pathelse {// ----- Ignore only the double '//' in path,// but not the first and last '/'}}else {// ----- Look for item to skipif ($v_skip > 0) {$v_skip--;}else {$v_result = $v_list[$i].($i!=(sizeof($v_list)-1)?"/".$v_result:"");}}}// ----- Look for skipif ($v_skip > 0) {while ($v_skip > 0) {$v_result = '../'.$v_result;$v_skip--;}}}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : PclZipUtilPathInclusion()// Description :// This function indicates if the path $p_path is under the $p_dir tree. Or,// said in an other way, if the file or sub-dir $p_path is inside the dir// $p_dir.// The function indicates also if the path is exactly the same as the dir.// This function supports path with duplicated '/' like '//', but does not// support '.' or '..' statements.// Parameters :// Return Values :// 0 if $p_path is not inside directory $p_dir// 1 if $p_path is inside directory $p_dir// 2 if $p_path is exactly the same as $p_dir// --------------------------------------------------------------------------------function PclZipUtilPathInclusion($p_dir, $p_path){$v_result = 1;// ----- Look for path beginning by ./if ( ($p_dir == '.')|| ((strlen($p_dir) >=2) && (substr($p_dir, 0, 2) == './'))) {$p_dir = PclZipUtilTranslateWinPath(getcwd(), FALSE).'/'.substr($p_dir, 1);}if ( ($p_path == '.')|| ((strlen($p_path) >=2) && (substr($p_path, 0, 2) == './'))) {$p_path = PclZipUtilTranslateWinPath(getcwd(), FALSE).'/'.substr($p_path, 1);}// ----- Explode dir and path by directory separator$v_list_dir = explode("/", $p_dir);$v_list_dir_size = sizeof($v_list_dir);$v_list_path = explode("/", $p_path);$v_list_path_size = sizeof($v_list_path);// ----- Study directories paths$i = 0;$j = 0;while (($i < $v_list_dir_size) && ($j < $v_list_path_size) && ($v_result)) {// ----- Look for empty dir (path reduction)if ($v_list_dir[$i] == '') {$i++;continue;}if ($v_list_path[$j] == '') {$j++;continue;}// ----- Compare the itemsif (($v_list_dir[$i] != $v_list_path[$j]) && ($v_list_dir[$i] != '') && ( $v_list_path[$j] != '')) {$v_result = 0;}// ----- Next items$i++;$j++;}// ----- Look if everything seems to be the sameif ($v_result) {// ----- Skip all the empty itemswhile (($j < $v_list_path_size) && ($v_list_path[$j] == '')) $j++;while (($i < $v_list_dir_size) && ($v_list_dir[$i] == '')) $i++;if (($i >= $v_list_dir_size) && ($j >= $v_list_path_size)) {// ----- There are exactly the same$v_result = 2;}else if ($i < $v_list_dir_size) {// ----- The path is shorter than the dir$v_result = 0;}}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : PclZipUtilCopyBlock()// Description :// Parameters :// $p_mode : read/write compression mode// 0 : src & dest normal// 1 : src gzip, dest normal// 2 : src normal, dest gzip// 3 : src & dest gzip// Return Values :// --------------------------------------------------------------------------------function PclZipUtilCopyBlock($p_src, $p_dest, $p_size, $p_mode=0){$v_result = 1;if ($p_mode==0){while ($p_size != 0){$v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($p_src, $v_read_size);@fwrite($p_dest, $v_buffer, $v_read_size);$p_size -= $v_read_size;}}else if ($p_mode==1){while ($p_size != 0){$v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @gzread($p_src, $v_read_size);@fwrite($p_dest, $v_buffer, $v_read_size);$p_size -= $v_read_size;}}else if ($p_mode==2){while ($p_size != 0){$v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @fread($p_src, $v_read_size);@gzwrite($p_dest, $v_buffer, $v_read_size);$p_size -= $v_read_size;}}else if ($p_mode==3){while ($p_size != 0){$v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE);$v_buffer = @gzread($p_src, $v_read_size);@gzwrite($p_dest, $v_buffer, $v_read_size);$p_size -= $v_read_size;}}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : PclZipUtilRename()// Description :// This function tries to do a simple rename() function. If it fails, it// tries to copy the $p_src file in a new $p_dest file and then unlink the// first one.// Parameters :// $p_src : Old filename// $p_dest : New filename// Return Values :// 1 on success, 0 on failure.// --------------------------------------------------------------------------------function PclZipUtilRename($p_src, $p_dest){$v_result = 1;// ----- Try to rename the filesif (!@rename($p_src, $p_dest)) {// ----- Try to copy & unlink the srcif (!@copy($p_src, $p_dest)) {$v_result = 0;}else if (!@unlink($p_src)) {$v_result = 0;}}// ----- Returnreturn $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : PclZipUtilOptionText()// Description :// Translate option value in text. Mainly for debug purpose.// Parameters :// $p_option : the option value.// Return Values :// The option text value.// --------------------------------------------------------------------------------function PclZipUtilOptionText($p_option){$v_list = get_defined_constants();for (reset($v_list); $v_key = key($v_list); next($v_list)) {$v_prefix = substr($v_key, 0, 10);if (( ($v_prefix == 'PCLZIP_OPT')|| ($v_prefix == 'PCLZIP_CB_')|| ($v_prefix == 'PCLZIP_ATT'))&& ($v_list[$v_key] == $p_option)) {return $v_key;}}$v_result = 'Unknown';return $v_result;}// --------------------------------------------------------------------------------// --------------------------------------------------------------------------------// Function : PclZipUtilTranslateWinPath()// Description :// Translate windows path by replacing '\' by '/' and optionally removing// drive letter.// Parameters :// $p_path : path to translate.// $p_remove_disk_letter : true | false// Return Values :// The path translated.// --------------------------------------------------------------------------------function PclZipUtilTranslateWinPath($p_path, $p_remove_disk_letter=true){if (stristr(php_uname(), 'windows')) {// ----- Look for potential disk letterif (($p_remove_disk_letter) && (($v_position = strpos($p_path, ':')) != false)) {$p_path = substr($p_path, $v_position+1);}// ----- Change potential windows directory separatorif ((strpos($p_path, '\\') > 0) || (substr($p_path, 0,1) == '\\')) {$p_path = strtr($p_path, '\\', '/');}}return $p_path;}// --------------------------------------------------------------------------------?>