mirror of
https://github.com/lubuntu-team/lubuntu.me.git
synced 2025-10-24 05:04:03 +00:00
373 lines
11 KiB
PHP
373 lines
11 KiB
PHP
<?php
|
|
|
|
if (!defined('ABSPATH')) die('No direct access allowed');
|
|
|
|
if (class_exists('ZipArchive')) :
|
|
/**
|
|
* We just add a last_error variable for comaptibility with our UpdraftPlus_PclZip object
|
|
*/
|
|
class UpdraftPlus_ZipArchive extends ZipArchive {
|
|
|
|
public $last_error = 'Unknown: ZipArchive does not return error messages';
|
|
}
|
|
endif;
|
|
|
|
/**
|
|
* A ZipArchive compatibility layer, with behaviour sufficient for our usage of ZipArchive
|
|
*/
|
|
class UpdraftPlus_PclZip {
|
|
|
|
protected $pclzip;
|
|
|
|
protected $path;
|
|
|
|
protected $addfiles;
|
|
|
|
protected $adddirs;
|
|
|
|
private $statindex;
|
|
|
|
private $include_mtime = false;
|
|
|
|
public $last_error;
|
|
|
|
public function __construct() {
|
|
$this->addfiles = array();
|
|
$this->adddirs = array();
|
|
// Put this in a non-backed-up, writeable location, to make sure that huge temporary files aren't created and then added to the backup - and that we have somewhere writable
|
|
global $updraftplus;
|
|
if (!defined('PCLZIP_TEMPORARY_DIR')) define('PCLZIP_TEMPORARY_DIR', trailingslashit($updraftplus->backups_dir_location()));
|
|
}
|
|
|
|
/**
|
|
* Used to include mtime in statindex (by default, not done - to save memory; probably a bit paranoid)
|
|
*
|
|
* @return null
|
|
*/
|
|
public function ud_include_mtime() {
|
|
$this->include_mtime = true;
|
|
}
|
|
|
|
public function __get($name) {
|
|
if ('numFiles' == $name || 'numAll' == $name) {
|
|
|
|
if (empty($this->pclzip)) return false;
|
|
|
|
$statindex = $this->pclzip->listContent();
|
|
|
|
if (empty($statindex)) {
|
|
$this->statindex = array();
|
|
// We return a value that is == 0, but allowing a PclZip error to be detected (PclZip returns 0 in the case of an error).
|
|
if (0 === $statindex) $this->last_error = $this->pclzip->errorInfo(true);
|
|
return (0 === $statindex) ? false : 0;
|
|
}
|
|
|
|
if ('numFiles' == $name) {
|
|
|
|
$result = array();
|
|
foreach ($statindex as $i => $file) {
|
|
if (!isset($statindex[$i]['folder']) || 0 == $statindex[$i]['folder']) {
|
|
$result[] = $file;
|
|
}
|
|
unset($statindex[$i]);
|
|
}
|
|
|
|
$this->statindex = $result;
|
|
|
|
} else {
|
|
$this->statindex = $statindex;
|
|
}
|
|
|
|
return count($this->statindex);
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
public function statIndex($i) {
|
|
if (empty($this->statindex[$i])) return array('name' => null, 'size' => 0);
|
|
$v = array('name' => $this->statindex[$i]['filename'], 'size' => $this->statindex[$i]['size']);
|
|
if ($this->include_mtime) $v['mtime'] = $this->statindex[$i]['mtime'];
|
|
return $v;
|
|
}
|
|
|
|
public function open($path, $flags = 0) {
|
|
|
|
if (!class_exists('PclZip')) include_once(ABSPATH.'/wp-admin/includes/class-pclzip.php');
|
|
if (!class_exists('PclZip')) {
|
|
$this->last_error = "No PclZip class was found";
|
|
return false;
|
|
}
|
|
|
|
// Route around PHP bug (exact version with the problem not known)
|
|
$ziparchive_create_match = (version_compare(PHP_VERSION, '5.2.12', '>') && defined('ZIPARCHIVE::CREATE')) ? ZIPARCHIVE::CREATE : 1;
|
|
|
|
if ($flags == $ziparchive_create_match && file_exists($path)) @unlink($path);
|
|
|
|
$this->pclzip = new PclZip($path);
|
|
|
|
if (empty($this->pclzip)) {
|
|
$this->last_error = 'Could not get a PclZip object';
|
|
return false;
|
|
}
|
|
|
|
// Make the empty directory we need to implement add_empty_dir()
|
|
global $updraftplus;
|
|
$updraft_dir = $updraftplus->backups_dir_location();
|
|
if (!is_dir($updraft_dir.'/emptydir') && !mkdir($updraft_dir.'/emptydir')) {
|
|
$this->last_error = "Could not create empty directory ($updraft_dir/emptydir)";
|
|
return false;
|
|
}
|
|
|
|
$this->path = $path;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
/**
|
|
* Do the actual write-out - it is assumed that close() is where this is done. Needs to return true/false
|
|
*
|
|
* @return boolean
|
|
*/
|
|
public function close() {
|
|
if (empty($this->pclzip)) {
|
|
$this->last_error = 'Zip file was not opened';
|
|
return false;
|
|
}
|
|
|
|
global $updraftplus;
|
|
$updraft_dir = $updraftplus->backups_dir_location();
|
|
|
|
$activity = false;
|
|
|
|
// Add the empty directories
|
|
foreach ($this->adddirs as $dir) {
|
|
if (false == $this->pclzip->add($updraft_dir.'/emptydir', PCLZIP_OPT_REMOVE_PATH, $updraft_dir.'/emptydir', PCLZIP_OPT_ADD_PATH, $dir)) {
|
|
$this->last_error = $this->pclzip->errorInfo(true);
|
|
return false;
|
|
}
|
|
$activity = true;
|
|
}
|
|
|
|
foreach ($this->addfiles as $rdirname => $adirnames) {
|
|
foreach ($adirnames as $adirname => $files) {
|
|
if (false == $this->pclzip->add($files, PCLZIP_OPT_REMOVE_PATH, $rdirname, PCLZIP_OPT_ADD_PATH, $adirname)) {
|
|
$this->last_error = $this->pclzip->errorInfo(true);
|
|
return false;
|
|
}
|
|
$activity = true;
|
|
}
|
|
unset($this->addfiles[$rdirname]);
|
|
}
|
|
|
|
$this->pclzip = false;
|
|
$this->addfiles = array();
|
|
$this->adddirs = array();
|
|
|
|
clearstatcache();
|
|
if ($activity && filesize($this->path) < 50) {
|
|
$this->last_error = "Write failed - unknown cause (check your file permissions)";
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Note: basename($add_as) is irrelevant; that is, it is actually basename($file) that will be used. But these are always identical in our usage.
|
|
*
|
|
* @param string $file Specific file to add
|
|
* @param string $add_as This is the name of the file that it is added as but it is usually the same as $file
|
|
*/
|
|
public function addFile($file, $add_as) {
|
|
// Add the files. PclZip appears to do the whole (copy zip to temporary file, add file, move file) cycle for each file - so batch them as much as possible. We have to batch by dirname(). On a test with 1000 files of 25KB each in the same directory, this reduced the time needed on that directory from 120s to 15s (or 5s with primed caches).
|
|
$rdirname = dirname($file);
|
|
$adirname = dirname($add_as);
|
|
$this->addfiles[$rdirname][$adirname][] = $file;
|
|
}
|
|
|
|
/**
|
|
* PclZip doesn't have a direct way to do this
|
|
*
|
|
* @param string $dir Specific Directory to empty
|
|
*/
|
|
public function addEmptyDir($dir) {
|
|
$this->adddirs[] = $dir;
|
|
}
|
|
|
|
public function extract($path_to_extract, $path) {
|
|
return $this->pclzip->extract(PCLZIP_OPT_PATH, $path_to_extract, PCLZIP_OPT_BY_NAME, $path);
|
|
}
|
|
}
|
|
|
|
class UpdraftPlus_BinZip extends UpdraftPlus_PclZip {
|
|
|
|
private $binzip;
|
|
|
|
public function __construct() {
|
|
global $updraftplus_backup;
|
|
$this->binzip = $updraftplus_backup->binzip;
|
|
if (!is_string($this->binzip)) {
|
|
$this->last_error = "No binary zip was found";
|
|
return false;
|
|
}
|
|
return parent::__construct();
|
|
}
|
|
|
|
public function addFile($file, $add_as) {
|
|
|
|
global $updraftplus;
|
|
// Get the directory that $add_as is relative to
|
|
$base = $updraftplus->str_lreplace($add_as, '', $file);
|
|
|
|
if ($file == $base) {
|
|
// Shouldn't happen; but see: https://bugs.php.net/bug.php?id=62119
|
|
$updraftplus->log("File skipped due to unexpected name mismatch (locale: ".setlocale(LC_CTYPE, "0")."): file=$file add_as=$add_as", 'notice', false, true);
|
|
} else {
|
|
$rdirname = untrailingslashit($base);
|
|
// Note: $file equals $rdirname/$add_as
|
|
$this->addfiles[$rdirname][] = $add_as;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* The standard zip binary cannot list; so we use PclZip for that
|
|
* Do the actual write-out - it is assumed that close() is where this is done. Needs to return true/false
|
|
*
|
|
* @return boolean - success or failure state
|
|
*/
|
|
public function close() {
|
|
|
|
if (empty($this->pclzip)) {
|
|
$this->last_error = 'Zip file was not opened';
|
|
return false;
|
|
}
|
|
|
|
global $updraftplus, $updraftplus_backup;
|
|
$updraft_dir = $updraftplus->backups_dir_location();
|
|
|
|
$activity = false;
|
|
|
|
// BinZip does not like zero-sized zip files
|
|
if (file_exists($this->path) && 0 == filesize($this->path)) @unlink($this->path);
|
|
|
|
$descriptorspec = array(
|
|
0 => array('pipe', 'r'),
|
|
1 => array('pipe', 'w'),
|
|
2 => array('pipe', 'w')
|
|
);
|
|
$exec = $this->binzip;
|
|
if (defined('UPDRAFTPLUS_BINZIP_OPTS') && UPDRAFTPLUS_BINZIP_OPTS) $exec .= ' '.UPDRAFTPLUS_BINZIP_OPTS;
|
|
$exec .= " -v -@ ".escapeshellarg($this->path);
|
|
|
|
$last_recorded_alive = time();
|
|
$something_useful_happened = $updraftplus->something_useful_happened;
|
|
$orig_size = file_exists($this->path) ? filesize($this->path) : 0;
|
|
$last_size = $orig_size;
|
|
clearstatcache();
|
|
|
|
$added_dirs_yet = false;
|
|
|
|
// If there are no files to add, but there are empty directories, then we need to make sure the directories actually get added
|
|
if (0 == count($this->addfiles) && 0 < count($this->adddirs)) {
|
|
$dir = realpath($updraftplus_backup->make_zipfile_source);
|
|
$this->addfiles[$dir] = '././.';
|
|
}
|
|
// Loop over each destination directory name
|
|
foreach ($this->addfiles as $rdirname => $files) {
|
|
|
|
$process = proc_open($exec, $descriptorspec, $pipes, $rdirname);
|
|
|
|
if (!is_resource($process)) {
|
|
$updraftplus->log('BinZip error: proc_open failed');
|
|
$this->last_error = 'BinZip error: proc_open failed';
|
|
return false;
|
|
}
|
|
|
|
if (!$added_dirs_yet) {
|
|
// Add the directories - (in fact, with binzip, non-empty directories automatically have their entries added; but it doesn't hurt to add them explicitly)
|
|
foreach ($this->adddirs as $dir) {
|
|
fwrite($pipes[0], $dir."/\n");
|
|
}
|
|
$added_dirs_yet = true;
|
|
}
|
|
|
|
$read = array($pipes[1], $pipes[2]);
|
|
$except = null;
|
|
|
|
if (!is_array($files) || 0 == count($files)) {
|
|
fclose($pipes[0]);
|
|
$write = array();
|
|
} else {
|
|
$write = array($pipes[0]);
|
|
}
|
|
|
|
while ((!feof($pipes[1]) || !feof($pipes[2]) || (is_array($files) && count($files)>0)) && false !== ($changes = @stream_select($read, $write, $except, 0, 200000))) {
|
|
|
|
if (is_array($write) && in_array($pipes[0], $write) && is_array($files) && count($files)>0) {
|
|
$file = array_pop($files);
|
|
// Send the list of files on stdin
|
|
fwrite($pipes[0], $file."\n");
|
|
if (0 == count($files)) fclose($pipes[0]);
|
|
}
|
|
|
|
if (is_array($read) && in_array($pipes[1], $read)) {
|
|
$w = fgets($pipes[1]);
|
|
// Logging all this really slows things down; use debug to mitigate
|
|
if ($w && $updraftplus_backup->debug) $updraftplus->log("Output from zip: ".trim($w), 'debug');
|
|
if (time() > $last_recorded_alive + 5) {
|
|
$updraftplus->record_still_alive();
|
|
$last_recorded_alive = time();
|
|
}
|
|
if (file_exists($this->path)) {
|
|
$new_size = @filesize($this->path);
|
|
if (!$something_useful_happened && $new_size > $orig_size + 20) {
|
|
$updraftplus->something_useful_happened();
|
|
$something_useful_happened = true;
|
|
}
|
|
clearstatcache();
|
|
// Log when 20% bigger or at least every 50MB
|
|
if ($new_size > $last_size*1.2 || $new_size > $last_size + 52428800) {
|
|
$updraftplus->log(basename($this->path).sprintf(": size is now: %.2f MB", round($new_size/1048576, 1)));
|
|
$last_size = $new_size;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (is_array($read) && in_array($pipes[2], $read)) {
|
|
$last_error = fgets($pipes[2]);
|
|
if (!empty($last_error)) $this->last_error = rtrim($last_error);
|
|
}
|
|
|
|
// Re-set
|
|
$read = array($pipes[1], $pipes[2]);
|
|
$write = (is_array($files) && count($files) >0) ? array($pipes[0]) : array();
|
|
$except = null;
|
|
|
|
}
|
|
|
|
fclose($pipes[1]);
|
|
fclose($pipes[2]);
|
|
|
|
$ret = proc_close($process);
|
|
|
|
if (0 != $ret && 12 != $ret) {
|
|
if ($ret < 128) {
|
|
$updraftplus->log("Binary zip: error (code: $ret - look it up in the Diagnostics section of the zip manual at http://www.info-zip.org/mans/zip.html for interpretation... and also check that your hosting account quota is not full)");
|
|
} else {
|
|
$updraftplus->log("Binary zip: error (code: $ret - a code above 127 normally means that the zip process was deliberately killed ... and also check that your hosting account quota is not full)");
|
|
}
|
|
if (!empty($w) && !$updraftplus_backup->debug) $updraftplus->log("Last output from zip: ".trim($w), 'debug');
|
|
return false;
|
|
}
|
|
|
|
unset($this->addfiles[$rdirname]);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|