Save 12
Save 12
php
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/*
Akeeba Kickstart - The server-side archive extraction wizard
Copyright (C) 2008-2019 Nicholas K. Dionysopoulos / AkeebaBackup.com
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
//
===================================================================================
==================================
// DO NOT MODIFY BELOW THIS LINE
//
===================================================================================
==================================
define('KICKSTART', 1);
if (!defined('VERSION'))
{
define('VERSION', '6.0.1');
}
if (!defined('KICKSTARTPRO'))
{
define('KICKSTARTPRO', '0');
}
// Used during development
if (!defined('KSDEBUG') && isset($_SERVER) && isset($_SERVER['HTTP_HOST']) &&
(strpos($_SERVER['HTTP_HOST'], 'local.web') !== false))
{
define('KSDEBUG', 1);
}
if (!defined('KSROOTDIR'))
{
define('KSROOTDIR', dirname(__FILE__));
}
if (defined('KSDEBUG'))
{
ini_set('error_log', KSROOTDIR . '/kickstart_error_log');
if (file_exists(KSROOTDIR . '/kickstart_error_log'))
{
@unlink(KSROOTDIR . '/kickstart_error_log');
}
error_reporting(E_ALL | E_STRICT);
}
else
{
@error_reporting(E_NONE);
}
//
===================================================================================
=======
// IIS missing REQUEST_URI workaround
//
===================================================================================
=======
/*
* Based REQUEST_URI for IIS Servers 1.0 by NeoSmart Technologies
* The proper method to solve IIS problems is to take a look at this:
* http://neosmart.net/dl.php?id=7
*/
if (!isset($_SERVER['REQUEST_URI']))
{
if (isset($_SERVER['HTTP_REQUEST_URI']))
{
$_SERVER['REQUEST_URI'] = $_SERVER['HTTP_REQUEST_URI'];
//Good to go!
}
else
{
//Someone didn't follow the instructions!
if (isset($_SERVER['SCRIPT_NAME']))
{
$_SERVER['HTTP_REQUEST_URI'] = $_SERVER['SCRIPT_NAME'];
}
else
{
$_SERVER['HTTP_REQUEST_URI'] = $_SERVER['PHP_SELF'];
}
if (isset($_SERVER['QUERY_STRING']) && !
empty($_SERVER['QUERY_STRING']))
{
$_SERVER['HTTP_REQUEST_URI'] .= '?' . $_SERVER['QUERY_STRING'];
}
//WARNING: This is a workaround!
//For guaranteed compatibility, HTTP_REQUEST_URI *MUST* be defined!
//See product documentation for instructions!
$_SERVER['REQUEST_URI'] = $_SERVER['HTTP_REQUEST_URI'];
}
}
/**
* Loads other PHP files containing extra Kickstart features. You can do all sorts
of tricks such as injecting HTML
* and CSS code (see AKFeatureGeorgeWSpecialEdition), adding AJAX task handlers
(see AKFeatureURLImport) etc.
*
* Feature files must follow one of the following naming conventions:
*
* - kickstart.SOMETHING.php
* - script_basename.SOMETHING.php
*
* where script_basename is the base name of Kickstart's PHP file. If you have
renamed kickstart.php to foobar.php this
* means that feature files must be named foobar.SOMETHING.php.
*
* The file must contain a class whose name starts with "AKFeature". The rest of
the name is irrelevant and does not
* have to follow a convention. It is, however, prudent to name the class using
something similar to the filename it is
* stored in to preserve your sanity and avoid potential conflicts.
*
* The class is instantiated ONLY ONCE, when the first call to callExtraFeature()
is made. Its methods are called using
* callExtraFeature() from Kickstart's (non-user-modifiable) code.
*/
function importKickstartFeatures($directory, $prefixes = array('kickstart'))
{
$dh = @opendir($directory);
if (!in_array($selfBasename, $prefixes))
{
$prefixes[] = $selfBasename;
}
if (!$found)
{
continue;
}
/**
* We have to ignore files which are just the prefix and a .php
extension (because one of these scripts is the
* currently executing script).
*/
foreach ($prefixes as $prefix)
{
if ($filename == ($prefix . '.php'))
{
continue 2;
}
}
// Op-code busting before loading the feature (in case it's self-
modifying)
if (function_exists('opcache_invalidate'))
{
opcache_invalidate($directory . '/' . $filename);
}
if (function_exists('apc_compile_file'))
{
apc_compile_file($directory . '/' . $filename);
}
if (function_exists('wincache_refresh_if_changed'))
{
wincache_refresh_if_changed(array($directory . '/' . $filename));
}
if (function_exists('xcache_asm'))
{
xcache_asm($directory . '/' . $filename);
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
define('_AKEEBA_RESTORATION', 1);
defined('DS') or define('DS', DIRECTORY_SEPARATOR);
if (!function_exists('aksubstr'))
{
if (function_exists('mb_strlen'))
{
function aksubstr($string, $start, $length = null)
{
return mb_substr($string, $start, $length, '8bit');
}
}
else
{
function aksubstr($string, $start, $length = null)
{
return substr($string, $start, $length);
}
}
}
/**
* Gets a query parameter from GET or POST data
*
* @param $key
* @param $default
*/
function getQueryParam($key, $default = null)
{
$value = $default;
if (array_key_exists($key, $_REQUEST))
{
$value = $_REQUEST[$key];
}
return $value;
}
// Debugging function
function debugMsg($msg)
{
if (!defined('KSDEBUG'))
{
return;
}
/**
* The base class of Akeeba Engine objects. Allows for error and warnings logging
* and propagation. Largely based on the Joomla! 1.5 JObject class.
*/
abstract class AKAbstractObject
{
/** @var array The queue size of the $_errors array. Set to 0 for
infinite size. */
protected $_errors_queue_size = 0;
/** @var array The queue size of the $_warnings array. Set to 0 for
infinite size. */
protected $_warnings_queue_size = 0;
/** @var array An array of errors */
private $_errors = array();
/** @var array An array of warnings */
private $_warnings = array();
/**
* Public constructor, makes sure we are instantiated only by the factory
class
*/
public function __construct()
{
/*
// Assisted Singleton pattern
if(function_exists('debug_backtrace'))
{
$caller=debug_backtrace();
if(
($caller[1]['class'] != 'AKFactory') &&
($caller[2]['class'] != 'AKFactory') &&
($caller[3]['class'] != 'AKFactory') &&
($caller[4]['class'] != 'AKFactory')
) {
var_dump(debug_backtrace());
trigger_error("You can't create direct descendants of
".__CLASS__, E_USER_ERROR);
}
}
*/
}
/**
* Get the most recent error message
*
* @param integer $i Optional error index
*
* @return string Error message
*/
public function getError($i = null)
{
return $this->getItemFromArray($this->_errors, $i);
}
/**
* Returns the last item of a LIFO string message queue, or a specific item
* if so specified.
*
* @param array $array An array of strings, holding messages
* @param int $i Optional message index
*
* @return mixed The message string, or false if the key doesn't exist
*/
private function getItemFromArray($array, $i = null)
{
// Find the item
if ($i === null)
{
// Default, return the last item
$item = end($array);
}
else if (!array_key_exists($i, $array))
{
// If $i has been specified but does not exist, return false
return false;
}
else
{
$item = $array[$i];
}
return $item;
}
/**
* Return all errors, if any
*
* @return array Array of error messages
*/
public function getErrors()
{
return $this->_errors;
}
/**
* Resets all error messages
*/
public function resetErrors()
{
$this->_errors = array();
}
/**
* Get the most recent warning message
*
* @param integer $i Optional warning index
*
* @return string Error message
*/
public function getWarning($i = null)
{
return $this->getItemFromArray($this->_warnings, $i);
}
/**
* Return all warnings, if any
*
* @return array Array of error messages
*/
public function getWarnings()
{
return $this->_warnings;
}
/**
* Resets all warning messages
*/
public function resetWarnings()
{
$this->_warnings = array();
}
/**
* Propagates errors and warnings to a foreign object. The foreign object
SHOULD
* implement the setError() and/or setWarning() methods but DOESN'T HAVE TO
be of
* AKAbstractObject type. For example, this can even be used to propagate to
a
* JObject instance in Joomla!. Propagated items will be removed from
ourselves.
*
* @param object $object The object to propagate errors and warnings to.
*/
public function propagateToObject(&$object)
{
// Skip non-objects
if (!is_object($object))
{
return;
}
if (method_exists($object, 'setError'))
{
if (!empty($this->_errors))
{
foreach ($this->_errors as $error)
{
$object->setError($error);
}
$this->_errors = array();
}
}
if (method_exists($object, 'setWarning'))
{
if (!empty($this->_warnings))
{
foreach ($this->_warnings as $warning)
{
$object->setWarning($warning);
}
$this->_warnings = array();
}
}
}
/**
* Propagates errors and warnings from a foreign object. Each propagated list
is
* then cleared on the foreign object, as long as it implements resetErrors()
and/or
* resetWarnings() methods.
*
* @param object $object The object to propagate errors and warnings from
*/
public function propagateFromObject(&$object)
{
if (method_exists($object, 'getErrors'))
{
$errors = $object->getErrors();
if (!empty($errors))
{
foreach ($errors as $error)
{
$this->setError($error);
}
}
if (method_exists($object, 'resetErrors'))
{
$object->resetErrors();
}
}
if (method_exists($object, 'getWarnings'))
{
$warnings = $object->getWarnings();
if (!empty($warnings))
{
foreach ($warnings as $warning)
{
$this->setWarning($warning);
}
}
if (method_exists($object, 'resetWarnings'))
{
$object->resetWarnings();
}
}
}
/**
* Add an error message
*
* @param string $error Error message
*/
public function setError($error)
{
if ($this->_errors_queue_size > 0)
{
if (count($this->_errors) >= $this->_errors_queue_size)
{
array_shift($this->_errors);
}
}
$this->_errors[] = $error;
}
/**
* Add an error message
*
* @param string $error Error message
*/
public function setWarning($warning)
{
if ($this->_warnings_queue_size > 0)
{
if (count($this->_warnings) >= $this->_warnings_queue_size)
{
array_shift($this->_warnings);
}
}
$this->_warnings[] = $warning;
}
/**
* Sets the size of the error queue (acts like a LIFO buffer)
*
* @param int $newSize The new queue size. Set to 0 for infinite length.
*/
protected function setErrorsQueueSize($newSize = 0)
{
$this->_errors_queue_size = (int) $newSize;
}
/**
* Sets the size of the warnings queue (acts like a LIFO buffer)
*
* @param int $newSize The new queue size. Set to 0 for infinite length.
*/
protected function setWarningsQueueSize($newSize = 0)
{
$this->_warnings_queue_size = (int) $newSize;
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* The superclass of all Akeeba Kickstart parts. The "parts" are intelligent
stateful
* classes which perform a single procedure and have preparation, running and
* finalization phases. The transition between phases is handled automatically by
* this superclass' tick() final public method, which should be the ONLY public API
* exposed to the rest of the Akeeba Engine.
*/
abstract class AKAbstractPart extends AKAbstractObject
{
/**
* Indicates whether this part has finished its initialisation cycle
*
* @var boolean
*/
protected $isPrepared = false;
/**
* Indicates whether this part has more work to do (it's in running state)
*
* @var boolean
*/
protected $isRunning = false;
/**
* Indicates whether this part has finished its finalization cycle
*
* @var boolean
*/
protected $isFinished = false;
/**
* Indicates whether this part has finished its run cycle
*
* @var boolean
*/
protected $hasRan = false;
/**
* The name of the engine part (a.k.a. Domain), used in return table
* generation.
*
* @var string
*/
protected $active_domain = "";
/**
* The step this engine part is in. Used verbatim in return table and
* should be set by the code in the _run() method.
*
* @var string
*/
protected $active_step = "";
/**
* A more detailed description of the step this engine part is in. Used
* verbatim in return table and should be set by the code in the _run()
* method.
*
* @var string
*/
protected $active_substep = "";
/**
* Any configuration variables, in the form of an array.
*
* @var array
*/
protected $_parametersArray = array();
/**
* The public interface to an engine part. This method takes care for
* calling the correct method in order to perform the initialisation -
* run - finalisation cycle of operation and return a proper response array.
*
* @return array A Response Array
*/
final public function tick()
{
// Call the right action method, depending on engine part state
switch ($this->getState())
{
case "init":
$this->_prepare();
break;
case "prepared":
$this->_run();
break;
case "running":
$this->_run();
break;
case "postrun":
$this->_finalize();
break;
}
return $out;
}
/**
* Returns the state of this engine part.
*
* @return string The state of this engine part. It can be one of
* error, init, prepared, running, postrun, finished.
*/
final public function getState()
{
if ($this->getError())
{
return "error";
}
if (!($this->isPrepared))
{
return "init";
}
if ($this->isFinished)
{
return "finished";
}
}
/**
* Runs the preparation for this part. Should set _isPrepared
* to true
*/
abstract protected function _prepare();
/**
* Runs the main functionality loop for this part. Upon calling,
* should set the _isRunning to true. When it finished, should set
* the _hasRan to true. If an error is encountered, setError should
* be used.
*/
abstract protected function _run();
/**
* Runs the finalisation process for this part. Should set
* _isFinished to true.
*/
abstract protected function _finalize();
/**
* Constructs a Response Array based on the engine part's state.
*
* @return array The Response Array for the current state
*/
final protected function _makeReturnTable()
{
// Get a list of warnings
$warnings = $this->getWarnings();
// Report only new warnings if there is no warnings queue size
if ($this->_warnings_queue_size == 0)
{
if (($this->warnings_pointer > 0) && ($this->warnings_pointer <
(count($warnings))))
{
$warnings = array_slice($warnings, $this->warnings_pointer
+ 1);
$this->warnings_pointer += count($warnings);
}
else
{
$this->warnings_pointer = count($warnings);
}
}
$out = array(
'HasRun' => (!($this->isFinished)),
'Domain' => $this->active_domain,
'Step' => $this->active_step,
'Substep' => $this->active_substep,
'Error' => $this->getError(),
'Warnings' => $warnings
);
return $out;
}
/**
* Returns a copy of the class's status array
*
* @return array
*/
public function getStatusArray()
{
return $this->_makeReturnTable();
}
/**
* Sends any kind of setup information to the engine part. Using this,
* we avoid passing parameters to the constructor of the class. These
* parameters should be passed as an indexed array and should be taken
* into account during the preparation process only. This function will
* set the error flag if it's called after the engine part is prepared.
*
* @param array $parametersArray The parameters to be passed to the
* engine part.
*/
final public function setup($parametersArray)
{
if ($this->isPrepared)
{
$this->setState('error', "Can't modify configuration after the
preparation of " . $this->active_domain);
}
else
{
$this->_parametersArray = $parametersArray;
if (array_key_exists('root', $parametersArray))
{
$this->databaseRoot = $parametersArray['root'];
}
}
}
/**
* Sets the engine part's internal state, in an easy to use manner
*
* @param string $state One of init, prepared, running, postrun,
finished, error
* @param string $errorMessage The reported error message, should the
state be set to error
*/
protected function setState($state = 'init', $errorMessage = 'Invalid
setState argument')
{
switch ($state)
{
case 'init':
$this->isPrepared = false;
$this->isRunning = false;
$this->isFinished = false;
$this->hasRun = false;
break;
case 'prepared':
$this->isPrepared = true;
$this->isRunning = false;
$this->isFinished = false;
$this->hasRun = false;
break;
case 'running':
$this->isPrepared = true;
$this->isRunning = true;
$this->isFinished = false;
$this->hasRun = false;
break;
case 'postrun':
$this->isPrepared = true;
$this->isRunning = false;
$this->isFinished = false;
$this->hasRun = true;
break;
case 'finished':
$this->isPrepared = true;
$this->isRunning = false;
$this->isFinished = true;
$this->hasRun = false;
break;
case 'error':
default:
$this->setError($errorMessage);
break;
}
}
/**
* Attaches an observer object
*
* @param AKAbstractPartObserver $obs
*/
function attach(AKAbstractPartObserver $obs)
{
$this->observers["$obs"] = $obs;
}
/**
* Detaches an observer object
*
* @param AKAbstractPartObserver $obs
*/
function detach(AKAbstractPartObserver $obs)
{
unset($this->observers["$obs"]);
}
/**
* Sets the BREAKFLAG, which instructs this engine part that the current step
must break immediately,
* in fear of timing out.
*/
protected function setBreakFlag()
{
AKFactory::set('volatile.breakflag', true);
}
/**
* Notifies observers each time something interesting happened to the part
*
* @param mixed $message The event object
*/
protected function notify($message)
{
foreach ($this->observers as $obs)
{
$obs->update($this, $message);
}
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* The base class of unarchiver classes
*/
abstract class AKAbstractUnarchiver extends AKAbstractPart
{
/** @var array List of the names of all archive parts */
public $archiveList = array();
/** @var int The total size of all archive parts */
public $totalSize = array();
/** @var array Which files to rename */
public $renameFiles = array();
/** @var array Which directories to rename */
public $renameDirs = array();
/** @var array Which files to skip */
public $skipFiles = array();
/** @var string Archive filename */
protected $filename = null;
/** @var integer Current archive part number */
protected $currentPartNumber = -1;
/** @var integer The offset inside the current part */
protected $currentPartOffset = 0;
/** @var bool Should I restore permissions? */
protected $flagRestorePermissions = false;
/** @var AKAbstractPostproc Post processing class */
protected $postProcEngine = null;
/** @var string Absolute path to prepend to extracted files */
protected $addPath = '';
/** @var string Absolute path to remove from extracted files */
protected $removePath = '';
/** @var integer Chunk size for processing */
protected $chunkSize = 524288;
/** @var resource File pointer to the current archive part file */
protected $fp = null;
/** @var int Run state when processing the current archive file */
protected $runState = null;
/** @var stdClass File header data, as read by the readFileHeader() method */
protected $fileHeader = null;
/** @var int How much of the uncompressed data we've read so far */
protected $dataReadLength = 0;
/** @var array Unwriteable files in these directories are always ignored and
do not cause errors when not extracted */
protected $ignoreDirectories = array();
/**
* Public constructor
*/
public function __construct()
{
parent::__construct();
}
/**
* Wakeup function, called whenever the class is unserialized
*/
public function __wakeup()
{
if ($this->currentPartNumber >= 0)
{
$this->fp = @fopen($this->archiveList[$this->currentPartNumber],
'rb');
if ((is_resource($this->fp)) && ($this->currentPartOffset > 0))
{
@fseek($this->fp, $this->currentPartOffset);
}
}
}
/**
* Sleep function, called whenever the class is serialized
*/
public function shutdown()
{
if (is_resource($this->fp))
{
$this->currentPartOffset = @ftell($this->fp);
@fclose($this->fp);
}
}
/**
* Is this file or directory contained in a directory we've decided to ignore
* write errors for? This is useful to let the extraction work despite write
* errors in the log, logs and tmp directories which MIGHT be used by the
system
* on some low quality hosts and Plesk-powered hosts.
*
* @param string $shortFilename The relative path of the file/directory in
the package
*
* @return boolean True if it belongs in an ignored directory
*/
public function isIgnoredDirectory($shortFilename)
{
// return false;
/**
* Implements the abstract _prepare() method
*/
final protected function _prepare()
{
parent::__construct();
if (count($this->_parametersArray) > 0)
{
foreach ($this->_parametersArray as $key => $value)
{
switch ($key)
{
// Archive's absolute filename
case 'filename':
$this->filename = $value;
// Sanity check
if (!empty($value))
{
$value = strtolower($value);
if (strlen($value) > 6)
{
if (
(substr($value, 0, 7) ==
'http://')
|| (substr($value, 0, 8) ==
'https://')
|| (substr($value, 0, 6) ==
'ftp://')
|| (substr($value, 0, 7) ==
'ssh2://')
|| (substr($value, 0, 6) ==
'ssl://')
)
{
$this->setState('error',
'Invalid archive location');
}
}
}
break;
$this->scanArchives();
$this->readArchiveHeader();
$errMessage = $this->getError();
if (!empty($errMessage))
{
$this->setState('error', $errMessage);
}
else
{
$this->runState = AK_STATE_NOFILE;
$this->setState('prepared');
}
}
/**
* Scans for archive parts
*/
private function scanArchives()
{
if (defined('KSDEBUG'))
{
@unlink('debug.txt');
}
debugMsg('Preparing to scan archives');
$privateArchiveList = array();
// Scan for multiple parts until we don't find any more of them
$count = 0;
$found = true;
$this->archiveList = array();
while ($found)
{
++$count;
$extension = substr($base_extension, 0, 2) . sprintf('%02d',
$count);
$filename = $dirname . DIRECTORY_SEPARATOR . $basename .
$extension;
$found = file_exists($filename);
if ($found)
{
debugMsg('- Found archive ' . $filename);
// Add yet another part, with a numeric-appended filename
$this->archiveList[] = $filename;
$filesize = @filesize($filename);
$this->totalSize += $filesize;
$filename = $this->filename;
$filesize = @filesize($filename);
$this->totalSize += $filesize;
$this->currentPartNumber = -1;
$this->currentPartOffset = 0;
$this->runState = AK_STATE_NOFILE;
/**
* Returns the base extension of the file, e.g. '.jpa'
*
* @return string
*/
private function getBaseExtension()
{
static $baseextension;
if (empty($baseextension))
{
$basename = basename($this->filename);
$lastdot = strrpos($basename, '.');
$baseextension = substr($basename, $lastdot);
}
return $baseextension;
}
/**
* Concrete classes are supposed to use this method in order to read the
archive's header and
* prepare themselves to the point of being ready to extract the first file.
*/
protected abstract function readArchiveHeader();
$this->setState('running');
$timer = AKFactory::getTimer();
$status = true;
while ($status && ($timer->getTimeLeft() > 0))
{
switch ($this->runState)
{
case AK_STATE_NOFILE:
debugMsg(__CLASS__ . '::_run() - Reading file
header');
$status = $this->readFileHeader();
if ($status)
{
// Send start of file notification
$message = new stdClass;
$message->type = 'startfile';
$message->content = new stdClass;
$message->content->realfile = $this-
>fileHeader->file;
$message->content->file = $this-
>fileHeader->file;
$message->content->uncompressed = $this-
>fileHeader->uncompressed;
if (array_key_exists('realfile',
get_object_vars($this->fileHeader)))
{
$message->content->realfile = $this-
>fileHeader->realFile;
}
if (array_key_exists('compressed',
get_object_vars($this->fileHeader)))
{
$message->content->compressed = $this-
>fileHeader->compressed;
}
else
{
$message->content->compressed = 0;
}
$this->notify($message);
}
else
{
debugMsg(__CLASS__ . '::_run() - Could not read
file header');
}
break;
case AK_STATE_HEADER:
case AK_STATE_DATA:
debugMsg(__CLASS__ . '::_run() - Processing file
data');
$status = $this->processFileData();
break;
case AK_STATE_DATAREAD:
case AK_STATE_POSTPROC:
debugMsg(__CLASS__ . '::_run() - Calling post-
processing class');
$this->postProcEngine->timestamp = $this->fileHeader-
>timestamp;
$status = $this-
>postProcEngine->process();
$this->propagateFromObject($this->postProcEngine);
$this->runState = AK_STATE_DONE;
break;
case AK_STATE_DONE:
default:
if ($status)
{
debugMsg(__CLASS__ . '::_run() - Finished
extracting file');
// Send end of file notification
$message = new stdClass;
$message->type = 'endfile';
$message->content = new stdClass;
if (array_key_exists('realfile',
get_object_vars($this->fileHeader)))
{
$message->content->realfile = $this-
>fileHeader->realFile;
}
else
{
$message->content->realfile = $this-
>fileHeader->file;
}
$message->content->file = $this->fileHeader-
>file;
if (array_key_exists('compressed',
get_object_vars($this->fileHeader)))
{
$message->content->compressed = $this-
>fileHeader->compressed;
}
else
{
$message->content->compressed = 0;
}
$message->content->uncompressed = $this-
>fileHeader->uncompressed;
$this->notify($message);
}
$this->runState = AK_STATE_NOFILE;
break;
}
}
$error = $this->getError();
if (!$status && ($this->runState == AK_STATE_NOFILE) && empty($error))
{
debugMsg(__CLASS__ . '::_run() - Just finished');
// We just finished
$this->setState('postrun');
}
elseif (!empty($error))
{
debugMsg(__CLASS__ . '::_run() - Halted with an error:');
debugMsg($error);
$this->setState('error', $error);
}
}
/**
* Concrete classes must use this method to read the file header
*
* @return bool True if reading the file was successful, false if an error
occurred or we reached end of archive
*/
protected abstract function readFileHeader();
/**
* Concrete classes must use this method to process file data. It must set
$runState to AK_STATE_DATAREAD when
* it's finished processing the file data.
*
* @return bool True if processing the file data was successful, false if an
error occurred
*/
protected abstract function processFileData();
/**
* Opens the next part file for reading
*/
protected function nextFile()
{
debugMsg('Current part is ' . $this->currentPartNumber . '; opening the
next part');
++$this->currentPartNumber;
if ($this->currentPartNumber > (count($this->archiveList) - 1))
{
$this->setState('postrun');
return false;
}
else
{
if (is_resource($this->fp))
{
@fclose($this->fp);
}
debugMsg('Opening file ' . $this->archiveList[$this-
>currentPartNumber]);
$this->fp = @fopen($this->archiveList[$this->currentPartNumber],
'rb');
if ($this->fp === false)
{
debugMsg('Could not open file - crash imminent');
$this-
>setError(AKText::sprintf('ERR_COULD_NOT_OPEN_ARCHIVE_PART', $this-
>archiveList[$this->currentPartNumber]));
}
fseek($this->fp, 0);
$this->currentPartOffset = 0;
return true;
}
}
/**
* Returns true if we have reached the end of file
*
* @param $local bool True to return EOF of the local file, false (default)
to return if we have reached the end of
* the archive set
*
* @return bool True if we have reached End Of File
*/
protected function isEOF($local = false)
{
$eof = @feof($this->fp);
if (!$eof)
{
// Border case: right at the part's end (eeeek!!!). For the life
of me, I don't understand why
// feof() doesn't report true. It expects the fp to be positioned
*beyond* the EOF to report
// true. Incredible! :(
$position = @ftell($this->fp);
$filesize = @filesize($this->archiveList[$this-
>currentPartNumber]);
if ($filesize <= 0)
{
// 2Gb or more files on a 32 bit version of PHP tend to get
screwed up. Meh.
$eof = false;
}
elseif ($position >= $filesize)
{
$eof = true;
}
}
if ($local)
{
return $eof;
}
else
{
return $eof && ($this->currentPartNumber >= (count($this-
>archiveList) - 1));
}
}
/**
* Tries to make a directory user-writable so that we can write a file to it
*
* @param $path string A path to a file
*/
protected function setCorrectPermissions($path)
{
static $rootDir = null;
if (is_null($rootDir))
{
$rootDir = rtrim(AKFactory::get('kickstart.setup.destdir', ''),
'/\\');
}
/**
* Reads data from the archive and notifies the observer with the 'reading'
message
*
* @param $fp
* @param $length
*/
protected function fread($fp, $length = null)
{
if (is_numeric($length))
{
if ($length > 0)
{
$data = fread($fp, $length);
}
else
{
$data = fread($fp, PHP_INT_MAX);
}
}
else
{
$data = fread($fp, PHP_INT_MAX);
}
if ($data === false)
{
$data = '';
}
return $data;
}
/**
* Removes the configured $removePath from the path $path
*
* @param string $path The path to reduce
*
* @return string The reduced path
*/
protected function removePath($path)
{
if (empty($this->removePath))
{
return $path;
}
return $path;
}
/**
* Am I supposed to skip the extraction of the current file? This depends on
*
* @return bool
*/
protected function mustSkip()
{
static $isDryRun = null;
// I am asked about the same file again. Return the cached result.
if ($this->fileHeader->file == $lastFileName)
{
return $mustSkip;
}
return $mustSkip;
}
/**
* Get the list of files / folders to extract. The list can contain filenames
or glob patterns.
*
* @return array
*/
private function getExtractList()
{
$rawList = AKFactory::get('kickstart.setup.extract_list', '');
if (empty($rawList))
{
return array();
}
return array_unique($list);
}
/**
* Tests whether the item $item matches the list of shell patterns $list.
*
* @param string $item The file name to test
* @param array $list The list of glob patterns to match
*
* @return bool
*/
private function matchesGlobPatterns($item, array $list)
{
if (empty($list))
{
return true;
}
return false;
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* File post processor engines base class
*/
abstract class AKAbstractPostproc extends AKAbstractObject
{
/** @var int The UNIX timestamp of the file's desired modification date */
public $timestamp = 0;
/** @var string The current (real) file path we'll have to process */
protected $filename = null;
/** @var int The requested permissions */
protected $perms = 0755;
/** @var string The temporary file path we gave to the unarchiver engine */
protected $tempFilename = null;
/** @var string The temporary directory where the data will be stored */
protected $tempDir = '';
/**
* Processes the current file, e.g. moves it from temp to final location by
FTP
*/
abstract public function process();
/**
* The unarchiver tells us the path to the filename it wants to extract and
we give it
* a different path instead.
*
* @param string $filename The path to the real file
* @param int $perms The permissions we need the file to have
*
* @return string The path to the temporary file
*/
abstract public function processFilename($filename, $perms = 0755);
/**
* Recursively creates a directory if it doesn't exist
*
* @param string $dirName The directory to create
* @param int $perms The permissions to give to that directory
*/
abstract public function createDirRecursive($dirName, $perms);
/**
* Returns the configured temporary directory
*
* @return string
*/
public function getTempDir()
{
return $this->tempDir;
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Descendants of this class can be used in the unarchiver's observer methods
(attach, detach and notify)
*
* @author Nicholas
*
*/
abstract class AKAbstractPartObserver
{
abstract public function update($object, $message);
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Direct file writer
*/
class AKPostprocDirect extends AKAbstractPostproc
{
public function process()
{
$restorePerms = AKFactory::get('kickstart.setup.restoreperms', false);
if ($restorePerms)
{
@chmod($this->filename, $this->perms);
}
else
{
if (@is_file($this->filename))
{
@chmod($this->filename, 0644);
}
else
{
@chmod($this->filename, 0755);
}
}
if ($this->timestamp > 0)
{
@touch($this->filename, $this->timestamp);
}
return true;
}
return $filename;
}
return true;
}
$root = AKFactory::get('kickstart.setup.destdir');
$root = rtrim(str_replace('\\', '/', $root), '/');
$dir = rtrim(str_replace('\\', '/', $dirName), '/');
if (strpos($dir, $root) === 0)
{
$dir = ltrim(substr($dir, strlen($root)), '/');
$root .= '/';
}
else
{
$root = '';
}
if (empty($dir))
{
return true;
}
return false;
}
}
// Try to set new directory permissions to 0755
@chmod($root . $path, $perms);
}
return true;
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* FTP file writer
*/
class AKPostprocFTP extends AKAbstractPostproc
{
/** @var bool Should I use FTP over implicit SSL? */
public $useSSL = false;
/** @var bool use Passive mode? */
public $passive = true;
/** @var string FTP host name */
public $host = '';
/** @var int FTP port */
public $port = 21;
/** @var string FTP user name */
public $user = '';
/** @var string FTP password */
public $pass = '';
/** @var string FTP initial directory */
public $dir = '';
/** @var resource The FTP handle */
private $handle = null;
$connected = $this->connect();
if ($connected)
{
if (!empty($this->tempDir))
{
$tempDir = rtrim($this->tempDir, '/\\') . '/';
$writable = $this->isDirWritable($tempDir);
}
else
{
$tempDir = '';
$writable = false;
}
if (!$writable)
{
// Default temporary directory is the current root
$tempDir = KSROOTDIR;
if (empty($tempDir))
{
// Oh, we have no directory reported!
$tempDir = '.';
}
$absoluteDirToHere = $tempDir;
$tempDir = rtrim(str_replace('\\', '/',
$tempDir), '/');
if (!empty($tempDir))
{
$tempDir .= '/';
}
$this->tempDir = $tempDir;
// Is this directory writable?
$writable = $this->isDirWritable($tempDir);
}
if (!$writable)
{
// Nope. Let's try creating a temporary directory in the
site's root.
$tempDir = $absoluteDirToHere .
'/kicktemp';
$trustMeIKnowWhatImDoing = 500 + 10 + 1; // working around
overzealous scanners written by bozos
$this->createDirRecursive($tempDir,
$trustMeIKnowWhatImDoing);
// Try making it writable...
$this->fixPermissions($tempDir);
$writable = $this->isDirWritable($tempDir);
}
return false;
}
// Login
if (!@ftp_login($this->handle, $this->user, $this->pass))
{
$this->setError(AKText::_('WRONG_FTP_USER'));
@ftp_close($this->handle);
return false;
}
return false;
}
return false;
}
fclose($tempHandle);
return true;
}
return true;
}
}
return false;
}
else
{
// Since the directory was built by PHP,
change its permissions
$trustMeIKnowWhatImDoing =
500 + 10 + 1; // working around
overzealous scanners written by bozos
@chmod($check, $trustMeIKnowWhatImDoing);
return true;
}
}
}
@ftp_chmod($this->handle, $perms, $check);
}
$previousDir = $check;
}
return true;
}
function __wakeup()
{
$this->connect();
}
$remotePath = dirname($this->filename);
$removePath = AKFactory::get('kickstart.setup.destdir', '');
if (!empty($removePath))
{
$removePath = ltrim($removePath, "/");
$remotePath = ltrim($remotePath, "/");
$left = substr($remotePath, 0, strlen($removePath));
if ($left == $removePath)
{
$remotePath = substr($remotePath, strlen($removePath));
}
}
$absoluteFSPath = dirname($this->filename);
$relativeFTPPath = trim($remotePath, '/');
$absoluteFTPPath = '/' . trim($this->dir, '/') . '/' .
trim($remotePath, '/');
$onlyFilename = basename($this->filename);
return false;
}
$ret = @ftp_chdir($this->handle, $absoluteFTPPath);
if ($ret === false)
{
$this->setError(AKText::sprintf('FTP_COULDNT_UPLOAD',
$this->filename));
return false;
}
}
return false;
}
$restorePerms = AKFactory::get('kickstart.setup.restoreperms', false);
if ($restorePerms)
{
@ftp_chmod($this->_handle, $this->perms, $remoteName);
}
else
{
@ftp_chmod($this->_handle, 0644, $remoteName);
}
return true;
}
/*
* Tries to fix directory/file permissions in the PHP level, so that
* the FTP operation doesn't fail.
* @param $path string The full path to a directory or file
*/
return null;
}
$this->filename = $filename;
$this->tempFilename = tempnam($this->tempDir, 'kickstart-');
$this->perms = $perms;
if (empty($this->tempFilename))
{
// Oops! Let's try something different
$this->tempFilename = $this->tempDir . '/kickstart-' . time() .
'.dat';
}
return $this->tempFilename;
}
if (!empty($removePath))
{
$left = substr($to, 0, strlen($removePath));
if ($left == $removePath)
{
$to = substr($to, strlen($removePath));
}
}
$to = '/' . trim($this->dir, '/') . '/' . trim($to, '/');
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* FTP file writer
*/
class AKPostprocSFTP extends AKAbstractPostproc
{
/** @var bool Should I use FTP over implicit SSL? */
public $useSSL = false;
/** @var bool use Passive mode? */
public $passive = true;
/** @var string FTP host name */
public $host = '';
/** @var int FTP port */
public $port = 21;
/** @var string FTP user name */
public $user = '';
/** @var string FTP password */
public $pass = '';
/** @var string FTP initial directory */
public $dir = '';
/** @var string Current remote directory, including the remote directory
string */
private $_currentdir;
if (trim($this->port) == '')
{
$this->port = 22;
}
$connected = $this->connect();
if ($connected)
{
if (!empty($this->tempDir))
{
$tempDir = rtrim($this->tempDir, '/\\') . '/';
$writable = $this->isDirWritable($tempDir);
}
else
{
$tempDir = '';
$writable = false;
}
if (!$writable)
{
// Default temporary directory is the current root
$tempDir = KSROOTDIR;
if (empty($tempDir))
{
// Oh, we have no directory reported!
$tempDir = '.';
}
$absoluteDirToHere = $tempDir;
$tempDir = rtrim(str_replace('\\', '/',
$tempDir), '/');
if (!empty($tempDir))
{
$tempDir .= '/';
}
$this->tempDir = $tempDir;
// Is this directory writable?
$writable = $this->isDirWritable($tempDir);
}
if (!$writable)
{
// Nope. Let's try creating a temporary directory in the
site's root.
$tempDir = $absoluteDirToHere .
'/kicktemp';
$trustMeIKnowWhatImDoing = 500 + 10 + 1; // working around
overzealous scanners written by bozos
$this->createDirRecursive($tempDir,
$trustMeIKnowWhatImDoing);
// Try making it writable...
$this->fixPermissions($tempDir);
$writable = $this->isDirWritable($tempDir);
}
if (!$writable)
{
// No writable directory found!!!
$this->setError(AKText::_('SFTP_TEMPDIR_NOT_WRITABLE'));
}
else
{
AKFactory::set('kickstart.ftp.tempdir', $tempDir);
$this->tempDir = $tempDir;
}
}
}
if (!function_exists('ssh2_connect'))
{
$this->setError(AKText::_('SFTP_NO_SSH2'));
return false;
}
$this->_connection = false;
return false;
}
$this->handle = @ssh2_sftp($this->_connection);
return false;
}
unset($this->_connection);
unset($this->handle);
return false;
}
if (@fopen("ssh2.sftp://{$this->handle}$basePath/$testFilename", 'r+')
=== false)
{
$this->setError(AKText::_('SFTP_WRONG_STARTING_DIR'));
unset($this->_connection);
unset($this->handle);
return false;
}
return true;
}
/**
* Changes to the requested directory in the remote server. You give only the
* path relative to the initial directory and it does all the rest by itself,
* including doing nothing if the remote directory is the one we want.
*
* @param string $dir The (realtive) remote directory
*
* @return bool True if successful, false otherwise.
*/
private function sftp_chdir($dir)
{
// Strip absolute filesystem path to website's root
$removePath = AKFactory::get('kickstart.setup.destdir', '');
if (!empty($removePath))
{
// UNIXize the paths
$removePath = str_replace('\\', '/', $removePath);
$dir = str_replace('\\', '/', $dir);
if ($left == $removePath)
{
$dir = substr($dir, strlen($removePath));
}
}
if (empty($dir))
{
// Because the substr() above may return FALSE.
$dir = '';
}
if ($this->_currentdir == $realdir)
{
// Already there, do nothing
return true;
}
return true;
}
}
return true;
}
}
if ($this->is_dir($check))
{
return true;
}
if (!$this->is_dir($check))
{
// Proactively try to delete a file by the same name
@ssh2_sftp_unlink($this->handle, $check);
return true;
}
}
}
$previousDir = $check;
}
return true;
}
if (!empty($basePath))
{
$basePath .= '/';
}
$oldPath = $pathBuilt;
$pathBuilt .= '/' . $dir;
function __wakeup()
{
$this->connect();
}
/*
* Tries to fix directory/file permissions in the PHP level, so that
* the FTP operation doesn't fail.
* @param $path string The full path to a directory or file
*/
$remotePath = dirname($this->filename);
$absoluteFSPath = dirname($this->filename);
$absoluteFTPPath = '/' . trim($this->dir, '/') . '/' .
trim($remotePath, '/');
$onlyFilename = basename($this->filename);
$ret = $this->sftp_chdir($absoluteFTPPath);
return false;
}
$ret = $this->sftp_chdir($absoluteFTPPath);
return false;
}
}
return false;
}
@unlink($this->tempFilename);
if ($restorePerms)
{
$this->chmod($remoteName, $this->perms);
}
else
{
$this->chmod($remoteName, 0644);
}
return true;
}
return -1;
}
$res = true;
@fclose($fp);
@fclose($localfp);
return $res;
}
return null;
}
$this->filename = $filename;
$this->tempFilename = tempnam($this->tempDir, 'kickstart-');
$this->perms = $perms;
if (empty($this->tempFilename))
{
// Oops! Let's try something different
$this->tempFilename = $this->tempDir . '/kickstart-' . time() .
'.dat';
}
return $this->tempFilename;
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Hybrid direct / FTP mode file writer
*/
class AKPostprocHybrid extends AKAbstractPostproc
{
/**
* Public constructor. Tries to connect to the FTP server.
*/
public function __construct()
{
parent::__construct();
$this->useFTP = true;
$this->useSSL = AKFactory::get('kickstart.ftp.ssl', false);
$this->passive = AKFactory::get('kickstart.ftp.passive', true);
$this->host = AKFactory::get('kickstart.ftp.host', '');
$this->port = AKFactory::get('kickstart.ftp.port', 21);
$this->user = AKFactory::get('kickstart.ftp.user', '');
$this->pass = AKFactory::get('kickstart.ftp.pass', '');
$this->dir = AKFactory::get('kickstart.ftp.dir', '');
$this->tempDir = AKFactory::get('kickstart.ftp.tempdir', '');
if (trim($this->port) == '')
{
$this->port = 21;
}
if ($connected)
{
if (!empty($this->tempDir))
{
$tempDir = rtrim($this->tempDir, '/\\') . '/';
$writable = $this->isDirWritable($tempDir);
}
else
{
$tempDir = '';
$writable = false;
}
if (!$writable)
{
// Default temporary directory is the current root
$tempDir = KSROOTDIR;
if (empty($tempDir))
{
// Oh, we have no directory reported!
$tempDir = '.';
}
$absoluteDirToHere = $tempDir;
$tempDir = rtrim(str_replace('\\', '/',
$tempDir), '/');
if (!empty($tempDir))
{
$tempDir .= '/';
}
$this->tempDir = $tempDir;
// Is this directory writable?
$writable = $this->isDirWritable($tempDir);
}
if (!$writable)
{
// Nope. Let's try creating a temporary directory in the
site's root.
$tempDir = $absoluteDirToHere .
'/kicktemp';
$trustMeIKnowWhatImDoing = 500 + 10 + 1; // working around
overzealous scanners written by bozos
$this->createDirRecursive($tempDir,
$trustMeIKnowWhatImDoing);
// Try making it writable...
$this->fixPermissions($tempDir);
$writable = $this->isDirWritable($tempDir);
}
if (!$writable)
{
// No writable directory found!!!
$this->setError(AKText::_('FTP_TEMPDIR_NOT_WRITABLE'));
}
else
{
AKFactory::set('kickstart.ftp.tempdir', $tempDir);
$this->tempDir = $tempDir;
}
}
}
/**
* Tries to connect to the FTP server
*
* @return bool
*/
public function connect()
{
if (!$this->useFTP)
{
return false;
}
return false;
}
// Login
if (!@ftp_login($this->handle, $this->user, $this->pass))
{
$this->setError(AKText::_('WRONG_FTP_USER'));
@ftp_close($this->handle);
return false;
}
// Change to initial directory
if (!@ftp_chdir($this->handle, $this->dir))
{
$this->setError(AKText::_('WRONG_FTP_PATH1'));
@ftp_close($this->handle);
return false;
}
return false;
}
fclose($tempHandle);
return true;
}
/**
* Is the directory writeable?
*
* @param string $dir The directory ti check
*
* @return bool
*/
private function isDirWritable($dir)
{
$fp = @fopen($dir . '/kickstart.dat', 'wb');
@fclose($fp);
unlink($dir . '/kickstart.dat');
return true;
}
/**
* Create a directory, recursively
*
* @param string $dirName The directory to create
* @param int $perms The permissions to give to the directory
*
* @return bool
*/
public function createDirRecursive($dirName, $perms)
{
// Strip absolute filesystem path to website's root
$removePath = AKFactory::get('kickstart.setup.destdir', '');
if (!empty($removePath))
{
// UNIXize the paths
$removePath = str_replace('\\', '/', $removePath);
$dirName = str_replace('\\', '/', $dirName);
// Make sure they both end in a slash
$removePath = rtrim($removePath, '/\\') . '/';
$dirName = rtrim($dirName, '/\\') . '/';
// Process the path removal
$left = substr($dirName, 0, strlen($removePath));
if ($left == $removePath)
{
$dirName = substr($dirName, strlen($removePath));
}
}
if ($this->is_dir($check))
{
return true;
}
return false;
}
}
$previousDir = $check;
$previousDirFS = $checkFS;
}
return true;
}
return false;
}
/**
* Tries to fix directory/file permissions in the PHP level, so that
* the FTP operation doesn't fail.
*
* @param $path string The full path to a directory or file
*/
private function fixPermissions($path)
{
// Turn off error reporting
if (!defined('KSDEBUG'))
{
$oldErrorReporting = error_reporting(0);
}
if (!empty($basePath))
{
$basePath .= '/';
}
$oldPath = $pathBuilt;
$pathBuilt .= '/' . $dir;
if (is_dir($oldPath . $dir))
{
$trustMeIKnowWhatImDoing = 500 + 10 + 1; // working around
overzealous scanners written by bozos
@chmod($oldPath . $dir, $trustMeIKnowWhatImDoing);
}
else
{
$trustMeIKnowWhatImDoing = 500 + 10 + 1; // working around
overzealous scanners written by bozos
if (@chmod($oldPath . $dir, $trustMeIKnowWhatImDoing) ===
false)
{
@unlink($oldPath . $dir);
}
}
}
// Restore error reporting
if (!defined('KSDEBUG'))
{
@error_reporting($oldErrorReporting);
}
}
/**
* Called after unserialisation, tries to reconnect to FTP
*/
function __wakeup()
{
if ($this->useFTP)
{
$this->connect();
}
}
function __destruct()
{
if (!$this->useFTP)
{
@ftp_close($this->handle);
}
}
/**
* Post-process an extracted file, using FTP or direct file writes to move it
*
* @return bool
*/
public function process()
{
if (is_null($this->tempFilename))
{
// If an empty filename is passed, it means that we shouldn't do
any post processing, i.e.
// the entity was a directory or symlink
return true;
}
$remotePath = dirname($this->filename);
$removePath = AKFactory::get('kickstart.setup.destdir', '');
$root = rtrim($removePath, '/\\');
if (!empty($removePath))
{
$removePath = ltrim($removePath, "/");
$remotePath = ltrim($remotePath, "/");
$left = substr($remotePath, 0, strlen($removePath));
if ($left == $removePath)
{
$remotePath = substr($remotePath, strlen($removePath));
}
}
$absoluteFSPath = dirname($this->filename);
$relativeFTPPath = trim($remotePath, '/');
$absoluteFTPPath = '/' . trim($this->dir, '/') . '/' .
trim($remotePath, '/');
$onlyFilename = basename($this->filename);
return false;
}
}
if ($this->useFTP)
{
$ret = @ftp_chdir($this->handle, $absoluteFTPPath);
}
@unlink($this->tempFilename);
return false;
}
return true;
}
return $ret;
}
/**
* Create a temporary filename
*
* @param string $filename The original filename
* @param int $perms The file permissions
*
* @return string
*/
public function processFilename($filename, $perms = 0755)
{
// Catch some error conditions...
if ($this->getError())
{
return false;
}
return null;
}
if (!empty($removePath))
{
$left = substr($filename, 0, strlen($removePath));
if ($left == $removePath)
{
$filename = substr($filename, strlen($removePath));
}
}
$this->filename = $filename;
$this->tempFilename = tempnam($this->tempDir, 'kickstart-');
$this->perms = $perms;
if (empty($this->tempFilename))
{
// Oops! Let's try something different
$this->tempFilename = $this->tempDir . '/kickstart-' . time() .
'.dat';
}
return $this->tempFilename;
}
/**
* Closes the FTP connection
*/
public function close()
{
if (!$this->useFTP)
{
@ftp_close($this->handle);
}
}
if (!empty($removePath))
{
$left = substr($file, 0, strlen($removePath));
if ($left == $removePath)
{
$file = substr($file, strlen($removePath));
}
}
return $ret;
}
if (!empty($removePath))
{
$left = substr($to, 0, strlen($removePath));
if ($left == $removePath)
{
$to = substr($to, strlen($removePath));
}
}
$to = '/' . trim($this->dir, '/') . '/' . trim($to, '/');
return $ret;
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* JPA archive extraction class
*/
class AKUnarchiverJPA extends AKAbstractUnarchiver
{
protected $archiveHeaderData = array();
return false;
}
if ($sig != 'JPA')
{
// Not a JPA file
debugMsg('Invalid archive signature');
$this->setError(AKText::_('ERR_NOT_A_JPA_FILE'));
return false;
}
// Read and parse the known portion of header data (14 bytes)
$bin_data = fread($this->fp, 14);
$header_data = unpack('Cmajor/Cminor/Vcount/Vuncsize/Vcsize',
$bin_data);
if ($rest_length > 0)
{
$junk = fread($this->fp, $rest_length);
}
else
{
$junk = '';
}
// Array-to-object conversion
foreach ($temp as $key => $value)
{
$this->archiveHeaderData->{$key} = $value;
}
debugMsg('Header data:');
debugMsg('Length : ' . $header_length);
debugMsg('Major : ' . $header_data['major']);
debugMsg('Minor : ' . $header_data['minor']);
debugMsg('File count : ' . $header_data['count']);
debugMsg('Uncompressed size : ' . $header_data['uncsize']);
debugMsg('Compressed size : ' . $header_data['csize']);
$this->currentPartOffset = @ftell($this->fp);
$this->dataReadLength = 0;
return true;
}
/**
* Concrete classes must use this method to read the file header
*
* @return bool True if reading the file was successful, false if an error
occurred or we reached end of archive
*/
protected function readFileHeader()
{
// If the current part is over, proceed to the next part please
if ($this->isEOF(true))
{
debugMsg('Archive part EOF; moving to next file');
$this->nextFile();
}
$this->currentPartOffset = ftell($this->fp);
// Check signature
if ($signature != 'JPF')
{
if ($this->isEOF(true))
{
// This file is finished; make sure it's the last one
$this->nextFile();
if (!$this->isEOF(false))
{
debugMsg('Invalid file signature before end of
archive encountered');
$this-
>setError(AKText::sprintf('INVALID_FILE_HEADER', $this->currentPartNumber, $this-
>currentPartOffset));
return false;
}
if (AKFactory::get('kickstart.setup.ignoreerrors', false))
{
debugMsg('Invalid file block signature; launching
heuristic file block signature scanner');
$screwed = !$this->heuristicFileHeaderLocator();
if (!$screwed)
{
$signature = 'JPF';
}
else
{
debugMsg('Heuristics failed. Brace yourself for
the imminent crash.');
}
}
if ($screwed)
{
debugMsg('Invalid file block signature');
// This is not a file block! The archive is corrupt.
$this-
>setError(AKText::sprintf('INVALID_FILE_HEADER', $this->currentPartNumber, $this-
>currentPartOffset));
return false;
}
}
}
// This a JPA Entity Block. Process the header.
$isBannedFile = false;
if ($restBytes > 0)
{
// Start reading the extra fields
while ($restBytes >= 4)
{
$extra_header_data = fread($this->fp, 4);
$extra_header = unpack('vsignature/vlength',
$extra_header_data);
$restBytes -= 4;
$extra_header['length'] -= 4;
switch ($extra_header['signature'])
{
case 256:
// File modified timestamp
if ($extra_header['length'] > 0)
{
$bindata = fread($this->fp,
$extra_header['length']);
$restBytes -= $extra_header['length'];
$timestamps =
unpack('Vmodified', substr($bindata, 0, 4));
$filectime =
$timestamps['modified'];
$this->fileHeader->timestamp =
$filectime;
}
break;
default:
// Unknown field
if ($extra_header['length'] > 0)
{
$junk = fread($this->fp,
$extra_header['length']);
$restBytes -= $extra_header['length'];
}
break;
}
}
if ($restBytes > 0)
{
$junk = fread($this->fp, $restBytes);
}
}
$compressionType = $header_data['compression'];
switch ($header_data['type'])
{
case 0:
$this->fileHeader->type = 'dir';
break;
case 1:
$this->fileHeader->type = 'file';
break;
case 2:
$this->fileHeader->type = 'link';
break;
}
switch ($compressionType)
{
case 0:
$this->fileHeader->compression = 'none';
break;
case 1:
$this->fileHeader->compression = 'gzip';
break;
case 2:
$this->fileHeader->compression = 'bzip2';
break;
}
$this->fileHeader->permissions = $header_data['perms'];
$this->currentPartOffset = @ftell($this->fp);
$this->runState = AK_STATE_DONE;
return true;
}
$this->postProcEngine->processFilename(null);
}
else
{
// Symlink; do not post-process
$this->postProcEngine->processFilename(null);
}
$this->createDirectory();
}
// Header is read
$this->runState = AK_STATE_HEADER;
$this->dataReadLength = 0;
return true;
}
if ($this->isEOF(true))
{
$this->nextFile();
}
if ($this->isEOF(false))
{
$fullEOF = true;
continue;
}
// Read 512Kb
$chunk = fread($this->fp, 524288);
$size_read = mb_strlen($chunk, '8bit');
//$pos = strpos($chunk, 'JPF');
$pos = mb_strpos($chunk, 'JPF', 0, '8bit');
return $ret;
}
/**
* Creates the directory this file points to
*/
protected function createDirectory()
{
if ($this->mustSkip())
{
return true;
}
if (($this->postProcEngine->createDirRecursive($dirName, $perms) ==
false) && (!$ignore))
{
$this->setError(AKText::sprintf('COULDNT_CREATE_DIR', $dirName));
return false;
}
else
{
return true;
}
}
/**
* Concrete classes must use this method to process file data. It must set
$runState to AK_STATE_DATAREAD when
* it's finished processing the file data.
*
* @return bool True if processing the file data was successful, false if an
error occurred
*/
protected function processFileData()
{
switch ($this->fileHeader->type)
{
case 'dir':
return $this->processTypeDir();
break;
case 'link':
return $this->processTypeLink();
break;
case 'file':
switch ($this->fileHeader->compression)
{
case 'none':
return $this->processTypeFileUncompressed();
break;
case 'gzip':
case 'bzip2':
return $this-
>processTypeFileCompressedSimple();
break;
}
break;
default:
debugMsg('Unknown file type ' . $this->fileHeader->type);
break;
}
}
/**
* Process the file data of a directory entry
*
* @return bool
*/
private function processTypeDir()
{
// Directory entries in the JPA do not have file data, therefore we're
done processing the entry
$this->runState = AK_STATE_DATAREAD;
return true;
}
/**
* Process the file data of a link entry
*
* @return bool
*/
private function processTypeLink()
{
$readBytes = 0;
$toReadBytes = 0;
$leftBytes = $this->fileHeader->compressed;
$data = '';
return false;
}
}
}
if (!$this->mustSkip())
{
// Try to remove an existing file or directory by the same name
if (file_exists($filename))
{
@unlink($filename);
@rmdir($filename);
}
$this->runState = AK_STATE_DATAREAD;
if ($this->dataReadLength == 0)
{
$outfp = @fopen($this->fileHeader->realFile, 'wb');
}
else
{
$outfp = @fopen($this->fileHeader->realFile, 'ab');
}
return false;
}
}
// Does the file have any data, at all?
if ($this->fileHeader->compressed == 0)
{
// No file data!
if (!$this->mustSkip() && is_resource($outfp))
{
@fclose($outfp);
}
$this->runState = AK_STATE_DATAREAD;
return true;
}
$toReadBytes = 0;
$leftBytes = $this->fileHeader->compressed - $this->dataReadLength;
return false;
}
}
if (!$this->mustSkip())
{
if (is_resource($outfp))
{
@fwrite($outfp, $data);
}
}
}
return true;
}
return false;
}
}
return true;
}
return false;
}
}
if ($this->fileHeader->compression == 'gzip')
{
$unzipData = gzinflate($zipData);
}
elseif ($this->fileHeader->compression == 'bzip2')
{
$unzipData = bzdecompress($zipData);
}
unset($zipData);
$this->runState = AK_STATE_DATAREAD;
return true;
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* ZIP archive extraction class
*
* Since the file data portion of ZIP and JPA are similarly structured (it's empty
for dirs,
* linked node name for symlinks, dumped binary data for no compressions and dumped
gzipped
* binary data for gzip compression) we just have to subclass AKUnarchiverJPA and
change the
* header reading bits. Reusable code ;)
*/
class AKUnarchiverZIP extends AKUnarchiverJPA
{
var $expectDataDescriptor = false;
return false;
}
$multiPartSigs = array(
0x08074b50, // Multi-part ZIP
0x30304b50, // Multi-part ZIP (alternate)
0x04034b50 // Single file
);
if (!in_array($headerData['sig'], $multiPartSigs))
{
debugMsg('Invalid header signature ' .
dechex($headerData['sig']));
$this->setError(AKText::_('ERR_CORRUPT_ARCHIVE'));
return false;
}
$this->currentPartOffset = @ftell($this->fp);
debugMsg('Current part offset after reading header: ' . $this-
>currentPartOffset);
$this->dataReadLength = 0;
return true;
}
/**
* Concrete classes must use this method to read the file header
*
* @return bool True if reading the file was successful, false if an error
occurred or we reached end of archive
*/
protected function readFileHeader()
{
// If the current part is over, proceed to the next part please
if ($this->isEOF(true))
{
debugMsg('Opening next archive part');
$this->nextFile();
}
$this->currentPartOffset = ftell($this->fp);
if ($this->expectDataDescriptor)
{
// The last file had bit 3 of the general purpose bit flag set.
This means that we have a
// 12 byte data descriptor we need to skip. To make things worse,
there might also be a 4
// byte optional data descriptor header (0x08074b50).
$junk = @fread($this->fp, 4);
$junk = unpack('Vsig', $junk);
if ($junk['sig'] == 0x08074b50)
{
// Yes, there was a signature
$junk = @fread($this->fp, 12);
debugMsg('Data descriptor (w/ header) skipped at ' .
(ftell($this->fp) - 12));
}
else
{
// No, there was no signature, just read another 8 bytes
$junk = @fread($this->fp, 8);
debugMsg('Data descriptor (w/out header) skipped at ' .
(ftell($this->fp) - 8));
}
$this->nextFile();
}
}
unpack('Vsig/C2ver/vbitflag/vcompmethod/vlastmodtime/vlastmoddate/Vcrc/Vcompsize/Vu
ncomp/vfnamelen/veflen', $headerBinary);
// Check signature
if (!($headerData['sig'] == 0x04034b50))
{
debugMsg('Not a file signature at ' . (ftell($this->fp) - 4));
// The signature is not the one used for files. Is this a central
directory record (i.e. we're done)?
if ($headerData['sig'] == 0x02014b50)
{
debugMsg('EOCD signature at ' . (ftell($this->fp) - 4));
// End of ZIP file detected. We'll just skip to the end of
file...
while ($this->nextFile())
{
};
@fseek($this->fp, 0, SEEK_END); // Go to EOF
return false;
}
else
{
debugMsg('Invalid signature ' .
dechex($headerData['sig']) . ' at ' . ftell($this->fp));
$this->setError(AKText::_('ERR_CORRUPT_ARCHIVE'));
return false;
}
}
$isBannedFile = false;
$this->fileHeader->compressed = $headerData['compsize'];
$this->fileHeader->uncompressed = $headerData['uncomp'];
$nameFieldLength = $headerData['fnamelen'];
$extraFieldLength = $headerData['eflen'];
switch ($headerData['compmethod'])
{
case 0:
$this->fileHeader->compression = 'none';
break;
case 8:
$this->fileHeader->compression = 'gzip';
break;
}
$this->currentPartOffset = @ftell($this->fp);
$this->runState = AK_STATE_DONE;
return true;
}
$dir = $this->fileHeader->file;
$this->postProcEngine->createDirRecursive($dir, 0755);
$this->postProcEngine->processFilename(null);
}
else
{
// Symlink; do not post-process
$this->fileHeader->timestamp = 0;
$this->postProcEngine->processFilename(null);
}
$this->createDirectory();
}
// Header is read
$this->runState = AK_STATE_HEADER;
return true;
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* JPS archive extraction class
*/
class AKUnarchiverJPS extends AKUnarchiverJPA
{
/**
* Header data for the archive
*
* @var array
*/
protected $archiveHeaderData = array();
/**
* Plaintext password from which the encryption key will be derived with
PBKDF2
*
* @var string
*/
protected $password = '';
/**
* Which hash algorithm should I use for key derivation with PBKDF2.
*
* @var string
*/
private $pbkdf2Algorithm = 'sha1';
/**
* How many iterations should I use for key derivation with PBKDF2
*
* @var int
*/
private $pbkdf2Iterations = 1000;
/**
* Should I use a static salt for key derivation with PBKDF2?
*
* @var bool
*/
private $pbkdf2UseStaticSalt = 0;
/**
* Static salt for key derivation with PBKDF2
*
* @var string
*/
private $pbkdf2StaticSalt = "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0";
/**
* How much compressed data I have read since the last file header read
*
* @var int
*/
private $compressedSizeReadSinceLastFileHeader = 0;
if ($sig != 'JPS')
{
// Not a JPA file
$this->setError(AKText::_('ERR_NOT_A_JPS_FILE'));
return false;
}
// Is this a v2 archive?
$versionHumanReadable = $header_data['major'] . '.' .
$header_data['minor'];
$isV2Archive = version_compare($versionHumanReadable, '2.0', 'ge');
$this->currentPartOffset = @ftell($this->fp);
$this->dataReadLength = 0;
return true;
}
/**
* Concrete classes must use this method to read the file header
*
* @return bool True if reading the file was successful, false if an error
occurred or we reached end of archive
*/
protected function readFileHeader()
{
// If the current part is over, proceed to the next part please
if ($this->isEOF(true))
{
$this->nextFile();
}
$this->currentPartOffset = ftell($this->fp);
return true;
}
$this->fileHeader = new stdClass();
$this->fileHeader->timestamp = 0;
// Check signature
if ($signature != 'JPF')
{
if ($this->isEOF(true))
{
// This file is finished; make sure it's the last one
$this->nextFile();
if (!$this->isEOF(false))
{
$this-
>setError(AKText::sprintf('INVALID_FILE_HEADER', $this->currentPartNumber, $this-
>currentPartOffset));
return false;
}
$this->setError(AKText::sprintf('INVALID_FILE_HEADER',
$this->currentPartNumber, $this->currentPartOffset));
return false;
}
}
$isBannedFile = false;
$this->fileHeader->timestamp = $header_data['filectime'];
$compressionType = $header_data['compression'];
case 1:
$this->fileHeader->type = 'file';
break;
case 2:
$this->fileHeader->type = 'link';
break;
}
switch ($compressionType)
{
case 0:
$this->fileHeader->compression = 'none';
break;
case 1:
$this->fileHeader->compression = 'gzip';
break;
case 2:
$this->fileHeader->compression = 'bzip2';
break;
}
$this->fileHeader->permissions = $header_data['perms'];
$this->currentPartOffset = @ftell($this->fp);
$this->runState = AK_STATE_DONE;
$this->fileHeader->compressed = $this-
>compressedSizeReadSinceLastFileHeader;
$this->compressedSizeReadSinceLastFileHeader = 0;
return true;
}
if (!$this->mustSkip())
{
if ($this->fileHeader->type == 'file')
{
// Regular file; ask the postproc engine to process its
filename
if ($restorePerms)
{
$this->fileHeader->realFile =
$this->postProcEngine->processFilename($this-
>fileHeader->file, $this->fileHeader->permissions);
}
else
{
$this->fileHeader->realFile = $this->postProcEngine-
>processFilename($this->fileHeader->file);
}
}
elseif ($this->fileHeader->type == 'dir')
{
$dir = $this->fileHeader->file;
$this->fileHeader->realFile = $dir;
$this->postProcEngine->processFilename(null);
}
else
{
// Symlink; do not post-process
$this->postProcEngine->processFilename(null);
}
$this->createDirectory();
}
$this->fileHeader->compressed = $this-
>compressedSizeReadSinceLastFileHeader;
$this->compressedSizeReadSinceLastFileHeader = 0;
// Header is read
$this->runState = AK_STATE_HEADER;
$this->dataReadLength = 0;
return true;
}
/**
* Creates the directory this file points to
*/
protected function createDirectory()
{
if ($this->mustSkip())
{
return true;
}
if (($this->postProcEngine->createDirRecursive($dirName, $perms) ==
false) && (!$ignore))
{
$this->setError(AKText::sprintf('COULDNT_CREATE_DIR', $dirName));
return false;
}
return true;
}
/**
* Concrete classes must use this method to process file data. It must set
$runState to AK_STATE_DATAREAD when
* it's finished processing the file data.
*
* @return bool True if processing the file data was successful, false if an
error occurred
*/
protected function processFileData()
{
switch ($this->fileHeader->type)
{
case 'dir':
return $this->processTypeDir();
break;
case 'link':
return $this->processTypeLink();
break;
case 'file':
switch ($this->fileHeader->compression)
{
case 'none':
return $this->processTypeFileUncompressed();
break;
case 'gzip':
case 'bzip2':
return $this-
>processTypeFileCompressedSimple();
break;
}
break;
}
}
/**
* Process the file data of a directory entry
*
* @return bool
*/
private function processTypeDir()
{
// Directory entries in the JPA do not have file data, therefore we're
done processing the entry
$this->runState = AK_STATE_DATAREAD;
return true;
}
/**
* Process the file data of a link entry
*
* @return bool
*/
private function processTypeLink()
{
return true;
}
if ($reallyReadBytes < 8)
{
// We read less than requested! Why? Did we hit local EOF?
if ($this->isEOF(true) && !$this->isEOF(false))
{
// Yeap. Let's go to the next file
$this->nextFile();
// Retry reading the header
$binMiniHeader = fread($this->fp, 8);
$reallyReadBytes = akstringlen($binMiniHeader);
// Still not enough data? If so, the archive is corrupt or
missing parts.
if ($reallyReadBytes < 8)
{
$this->setError(AKText::_('ERR_CORRUPT_ARCHIVE'));
return false;
}
}
else
{
// Nope. The archive is corrupt
$this->setError(AKText::_('ERR_CORRUPT_ARCHIVE'));
return false;
}
}
return false;
}
$data .= $restData;
}
else
{
// Nope. The archive is corrupt
$this->setError(AKText::_('ERR_CORRUPT_ARCHIVE'));
return false;
}
}
return false;
}
if (!$this->mustSkip())
{
// Try to remove an existing file or directory by the same name
if (file_exists($this->fileHeader->file))
{
@unlink($this->fileHeader->file);
@rmdir($this->fileHeader->file);
}
// Remove any trailing slash
if (substr($this->fileHeader->file, -1) == '/')
{
$this->fileHeader->file = substr($this->fileHeader->file,
0, -1);
}
// Create the symlink - only possible within PHP context. There's
no support built in the FTP protocol, so no postproc use is possible here :(
@symlink($data, $this->fileHeader->file);
}
$this->runState = AK_STATE_DATAREAD;
return false;
}
}
return true;
}
return false;
}
return false;
}
}
return true;
}
return false;
}
}
else
{
// Nope. The archive is corrupt
$this->setError(AKText::_('ERR_CORRUPT_ARCHIVE'));
return false;
}
}
$this->compressedSizeReadSinceLastFileHeader +=
$miniHeader['encsize'] + 8;
return false;
}
if (akstringlen($data) == 0)
{
$data = $restData;
}
else
{
$data .= $restData;
}
}
else
{
// Nope. The archive is corrupt
$this->setError(AKText::_('ERR_CORRUPT_ARCHIVE'));
return false;
}
}
return false;
}
// Decompress
$data = gzinflate($data);
$unc_len = akstringlen($data);
return true;
}
if ($signature != "JH\x00\x01")
{
// Not a valid JPS file
$this->setError(AKText::_('ERR_NOT_A_JPS_FILE'));
return false;
}
if ($header_data['length'] != 76)
{
// Not a valid JPS file
$this->setError(AKText::_('ERR_NOT_A_JPS_FILE'));
return false;
}
switch ($header_data['algo'])
{
case 0:
$algorithm = 'sha1';
break;
case 1:
$algorithm = 'sha256';
break;
case 2:
$algorithm = 'sha512';
break;
default:
// Not a valid JPS file
$this->setError(AKText::_('ERR_NOT_A_JPS_FILE'));
return false;
break;
}
$this->pbkdf2Algorithm = $algorithm;
$this->pbkdf2Iterations = $header_data['iterations'];
$this->pbkdf2UseStaticSalt = $header_data['useStaticSalt'];
$this->pbkdf2StaticSalt = fread($this->fp, 64);
return true;
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Timer class
*/
class AKCoreTimer extends AKAbstractObject
{
/** @var int Maximum execution time allowance per step */
private $max_exec_time = null;
/**
* Public constructor, creates the timer object and calculates the execution
time limits
*
* @return void
*/
public function __construct()
{
parent::__construct();
// Apply bias
$php_max_exec_time = $php_max_exec_time * $bias;
$config_max_exec_time = $config_max_exec_time * $bias;
/**
* Returns the current timestampt in decimal seconds
*/
private function microtime_float()
{
list($usec, $sec) = explode(" ", microtime());
/**
* Wake-up function to reset internal timer when we get unserialized
*/
public function __wakeup()
{
// Re-initialize start time on wake-up
$this->start_time = $this->microtime_float();
}
/**
* Gets the number of seconds left, before we hit the "must break" threshold
*
* @return float
*/
public function getTimeLeft()
{
return $this->max_exec_time - $this->getRunningTime();
}
/**
* Gets the time elapsed since object creation/unserialization, effectively
how
* long Akeeba Engine has been processing data
*
* @return float
*/
public function getRunningTime()
{
return $this->microtime_float() - $this->start_time;
}
/**
* Enforce the minimum execution time
*/
public function enforce_min_exec_time()
{
// Try to get a sane value for PHP's maximum_execution_time INI
parameter
if (@function_exists('ini_get'))
{
$php_max_exec = @ini_get("maximum_execution_time");
}
else
{
$php_max_exec = 10;
}
if (($php_max_exec == "") || ($php_max_exec == 0))
{
$php_max_exec = 10;
}
// Decrease $php_max_exec time by 500 msec we need (approx.) to tear
down
// the application, as well as another 500msec added for rounding
// error purposes. Also make sure this is never gonna be less than 0.
$php_max_exec = max($php_max_exec * 1000 - 1000, 0);
if (function_exists('usleep'))
{
usleep(1000 * $sleep_msec);
}
elseif (function_exists('time_nanosleep'))
{
$sleep_sec = floor($sleep_msec / 1000);
$sleep_nsec = 1000000 * ($sleep_msec - ($sleep_sec *
1000));
time_nanosleep($sleep_sec, $sleep_nsec);
}
elseif (function_exists('time_sleep_until'))
{
$until_timestamp = time() + $sleep_msec / 1000;
time_sleep_until($until_timestamp);
}
elseif (function_exists('sleep'))
{
$sleep_sec = ceil($sleep_msec / 1000);
sleep($sleep_sec);
}
}
}
/**
* Reset the timer. It should only be used in CLI mode!
*/
public function resetTime()
{
$this->start_time = $this->microtime_float();
}
/**
* @param int $max_exec_time
*/
public function setMaxExecTime($max_exec_time)
{
$this->max_exec_time = $max_exec_time;
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* A filesystem scanner which uses opendir()
*/
class AKUtilsLister extends AKAbstractObject
{
public function &getFiles($folder, $pattern = '*')
{
// Initialize variables
$arr = array();
$false = false;
if (!is_dir($folder))
{
return $false;
}
$handle = @opendir($folder);
// If directory is not accessible, just return FALSE
if ($handle === false)
{
$this->setWarning('Unreadable directory ' . $folder);
return $false;
}
return $arr;
}
if (!is_dir($folder))
{
return $false;
}
$handle = @opendir($folder);
// If directory is not accessible, just return FALSE
if ($handle === false)
{
$this->setWarning('Unreadable directory ' . $folder);
return $false;
}
return $arr;
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* A filesystem zapper - removes all files and folders under a root
*/
class AKUtilsZapper extends AKAbstractPart
{
/** @var array Directories left to be deleted */
private $directory_list;
/**
* Have we finished scanning all subdirectories of the current directory?
*
* @var boolean
*/
private $done_subdir_scanning = false;
/**
* Have we finished scanning all files of the current directory?
*
* @var boolean
*/
private $done_file_scanning = true;
/**
* Is the current directory completely excluded?
*
* @var boolean
*/
private $excluded_folder = false;
/** @var integer How many files have been processed in the current step */
private $processed_files_counter;
/**
* Implements the _prepare() abstract method
*
* Configuration parameters:
*
* root The root under which we are going to be deleting files
* excluded Absolute filesystem patterns to never delete (e.g.
/var/www/html/*.jpa)
*
* @return void
*/
protected function _prepare()
{
debugMsg(__CLASS__ . " :: Starting _prepare()");
$defaultExcluded = $this->getDefaultExclusions();
$parameters = array_merge(array(
'root' => rtrim(AKFactory::get('kickstart.setup.destdir'),
'/' . DIRECTORY_SEPARATOR),
'excluded' => $defaultExcluded,
'dryRun' => AKFactory::get('kickstart.setup.dryrun', false)
), $this->_parametersArray);
$this->root = $parameters['root'];
$this->excluded = $parameters['excluded'];
$this->directory_list[] = $this->root;
$this->done_subdir_scanning = true;
$this->done_file_scanning = true;
$this->total_files = 0;
$this->done_files = 0;
$this->total_folders = 0;
$this->done_folders = 0;
$this->dryRun = $parameters['dryRun'];
if (empty($this->root))
{
$error = "The folder to delete was not specified.";
debugMsg(__CLASS__ . " :: " . $error);
$this->setError($error);
return;
}
if (!is_dir($this->root))
{
$error = sprintf("Folder %s does not exist", $this->root);
return;
}
$this->setState('prepared');
return true;
}
// If I'm done scanning files and subdirectories and there are no more
files to pack get the next
// directory. This block is triggered in the first step in a new root.
if (empty($this->file_list) && $this->done_subdir_scanning && $this-
>done_file_scanning)
{
$this->progressMarkFolderDone();
if (!$this->getNextDirectory())
{
$this->setState('postrun');
return true;
}
}
// If I'm not done scanning for files and the file list is empty then
scan for more files
if (!$this->done_file_scanning && empty($this->file_list))
{
$this->scanFiles();
}
// If I have files left, delete them
elseif (!empty($this->file_list))
{
$this->delete_files();
}
// If I'm not done scanning subdirectories, go ahead and scan some more
of them
elseif (!$this->done_subdir_scanning)
{
$this->scanSubdirs();
}
// Do I have an error?
if ($this->getError())
{
return false;
}
return true;
}
/**
* Implements the _finalize() abstract method
*
*/
protected function _finalize()
{
// No finalization is required
$this->setState('finished');
}
//
===================================================================================
=========
// PRIVATE METHODS
//
===================================================================================
=========
/**
* Gets the next directory to scan from the stack. It also applies folder
* filters (directory exclusion, subdirectory exclusion, file exclusion),
* updating the operation toggle properties of the class.
*
* @return boolean True if we found a directory, false if the directory
* stack is empty. It also returns true if the folder is
* filtered (we are told to skip it)
*/
private function getNextDirectory()
{
// Reset the file / folder scanning positions
$this->done_file_scanning = false;
$this->done_subdir_scanning = false;
$this->excluded_folder = false;
if (count($this->directory_list) == 0)
{
// No directories left to scan
return false;
}
// Get and remove the last entry from the $directory_list array
$this->current_directory = array_pop($this->directory_list);
$this->setStep($this->current_directory);
$this->processed_files_counter = 0;
// Apply directory exclusion filters
if ($this->isFiltered($this->current_directory))
{
debugMsg("Skipping directory " . $this->current_directory);
$this->done_subdir_scanning = true;
$this->done_file_scanning = true;
$this->excluded_folder = true;
return true;
}
return true;
}
/**
* Try to delete some files from the $file_list
*
* @return boolean True if there were files deleted , false otherwise
* (empty filelist or fatal error)
*/
protected function delete_files()
{
// Get a reference to the archiver and the timer classes
$timer = AKFactory::getTimer();
return true;
}
debugMsg("Deleting files");
$numberOfFiles = 0;
$postProc = AKFactory::getPostProc();
$numberOfFiles++;
if (!$this->dryRun)
{
$postProc->unlink($file);
}
// Mark a done file
$this->progressMarkFileDone();
if ($this->getError())
{
return false;
}
$this->setSubstep($this->current_directory);
$this->notify((object) array(
'type' => 'deleteFolder',
'file' => $this->current_directory
));
if (!$this->dryRun)
{
/**
* The scanner goes from shallow to deep directory. However this means
that when it scans
* <root>/foo/bar/baz/bat
* it will only be able to remove the 'bat' directory, thus leaving
foo/bar/baz on the disk. The following
* method will check if the directory is a subdirectory of the site
root and work its way up the tree until
* it finds the site root. Therefore it will end up deleting the parent
folders as well.
*/
$this->deleteParentFolders($this->current_directory);
}
}
/**
* Returns the site root, the translated site root and the translated current
directory
*
* @return array
*/
protected function getCleanDirectoryComponents()
{
$root = $this->root;
$translated_root = $root;
$dir = TrimTrailingSlash($this->current_directory);
if (substr($dir, 0, strlen($new_translated_root)) ==
$new_translated_root)
{
$dir = substr($dir, strlen($new_translated_root));
}
}
if (substr($dir, 0, 1) == '/')
{
$dir = substr($dir, 1);
}
/**
* Steps the subdirectory scanning of the current directory
*
* @return boolean True on success, false on fatal error
*/
protected function scanSubdirs()
{
$lister = new AKUtilsLister();
// Get subdirectories
$subdirectories = $lister->getFolders($this->current_directory);
// Error propagation
$this->propagateFromObject($lister);
// Error control
if ($this->getError())
{
return false;
}
continue;
}
$this->file_list[] = $subdirectory;
$this->progressAddFile();
}
$this->directory_list[] = $subdirectory;
$this->progressAddFolder();
}
}
$this->done_subdir_scanning = true;
return true;
}
/**
* Steps the files scanning of the current directory
*
* @return boolean True on success, false on fatal error
*/
protected function scanFiles()
{
$lister = new AKUtilsLister();
// Error propagation
$this->propagateFromObject($lister);
// Error control
if ($this->getError())
{
return false;
}
$this->done_file_scanning = true;
return true;
}
if ($skipThisFile)
{
debugMsg("Skipping file $fileName");
continue;
}
$this->file_list[] = $fileName;
$this->processed_files_counter++;
$this->progressAddFile();
}
}
$this->done_file_scanning = true;
return true;
}
/**
* Is a file or folder filtered (protected from deletion)
*
* @param string $fileOrFolder
*
* @return bool
*/
private function isFiltered($fileOrFolder)
{
foreach ($this->excluded as $pattern)
{
if (fnmatch($pattern, $fileOrFolder))
{
return true;
}
}
return false;
}
/**
* Get the default exceptions from deletion
*
* @return array
*/
private function getDefaultExclusions()
{
$ret = array();
$destDir = AKFactory::get('kickstart.setup.destdir');
/**
* Exclude Kickstart / restore.php itself. Otherwise it'd crash!
*/
$myName = defined('KSSELFNAME') ? KSSELFNAME : basename(__FILE__);
$ret[] = KSROOTDIR . '/' . $myName;
/**
* Cheat: exclude the directory used in development (see
source/buildscripts/kickstart_test.php)
*
* This directory contains the non-concatenated source code for
Kickstart. We need to keep it protected.
*/
if (defined('MINIBUILD') && (MINIBUILD != $destDir))
{
$ret[] = TranslateWinPath(MINIBUILD);
}
/**
* Exclude the backup archive directory if it's not the site's root.
This prevents mindlessly deleting all your
* backups before you restore from a previous backup which might not be
the one you actually wanted. I will call
* this feature "clumsy-proofing".
*/
$backupArchive = AKFactory::get('kickstart.setup.sourcefile');
$backupDirectory = AKFactory::get('kickstart.setup.sourcepath');
$backupDirectory = empty($backupDirectory) ? dirname($backupArchive) :
$backupDirectory;
if ($backupDirectory != $destDir)
{
$ret[] = TranslateWinPath($backupDirectory);
}
/**
* Exclude the backup archive files
*
* This obviously only makes sense when the backup archives are stored
in the extraction target folder which is
* the most common use of Kickstart. In this case the backups folder is
not excluded above.
*/
$plainBackupName = basename($backupArchive, '.jpa');
$plainBackupName = basename($plainBackupName, '.jps');
$plainBackupName = basename($plainBackupName, '.zip');
$ret[] = TranslateWinPath($backupDirectory . '/' .
$plainBackupName) . '.*';
/**
* Exclude Kickstart language files. Only applies in Kickstart mode.
*/
if (defined('KICKSTART'))
{
$langDir = defined('KSLANGDIR') ? KSLANGDIR : KSROOTDIR;
$iniFilePattern = basename(KSSELFNAME, '.php') . '.*.ini';
if ($langDir != KSROOTDIR)
{
$ret[] = KSLANGDIR;
}
/**
* Exclude the configured Skipped Files ('kickstart.setup.skipfiles').
Also exclude the various restoration.php
* files if we are in restore.php mode and the files are present. These
are required for the integrated
* restoration to actually work :)
*/
$skippedFiles = AKFactory::get('kickstart.setup.skipfiles', array(
basename(__FILE__), 'kickstart.php', 'abiautomation.ini',
'htaccess.bak', 'php.ini.bak',
'cacert.pem',
));
if (!defined('KICKSTART'))
{
// In restore.php mode we have to exclude the various
restoration.php files
$skippedFiles = array_merge(array(
// Akeeba Backup for Joomla!
'administrator/components/com_akeeba/restoration.php',
// Joomla! Update
'administrator/components/com_joomlaupdate/restoration.php'
,
// Akeeba Backup for WordPress
'wp-content/plugins/akeebabackupwp/app/restoration.php',
'wp-
content/plugins/akeebabackupcorewp/app/restoration.php',
'wp-content/plugins/akeebabackup/app/restoration.php',
'wp-
content/plugins/akeebabackupwpcore/app/restoration.php',
// Akeeba Solo
'app/restoration.php',
), $skippedFiles);
}
/**
* Exclude .htaccess if the stealth feature is enabled. Otherwise we'd
unset the stealth mode.
*/
if (AKFactory::get('kickstart.stealth.enable'))
{
$ret[] = $destDir . '/.htaccess';
}
return $ret;
}
/**
* Recursively delete an empty folder and any of its empty parent folders.
*
* @param string $folder The folder to deletes
*/
private function deleteParentFolders($folder)
{
// Don't try to delete an empty folder or the filesystem root
if (empty($folder) || ($folder == '/'))
{
return;
}
$folder = TranslateWinPath($folder);
$root = TranslateWinPath($this->root);
// If the leaf folder is not under the site's root don't delete its parents
if (strpos($folder, $root) !== 0)
{
return;
}
/**
* Runs the Zapper and returns a status table. The Zapper only runs if the feature
is enabled (kickstart.setup.zapbefore
* is 1) and there are more Zapper steps to run (its state is not postrun). If any
of these conditions is not met we
* return boolean false.
*
* @param AKAbstractPartObserver $observer Optional observer to attack to the
Zapper instance
*
* @return bool|array Boolean false or a status array
*/
function runZapper(AKAbstractPartObserver $observer = null)
{
// This method should only run in restore.php mode or when we have Kickstart
Professional.
$isKickstart = defined('KICKSTART');
$isPro = defined('KICKSTARTPRO') ? KICKSTARTPRO : false;
$isDebug = defined('KSDEBUG') ? KSDEBUG : false;
if (!$enabled)
{
return false;
}
if ($zapper->getState() == 'finished')
{
return false;
}
if ($ret['Error'] != '')
{
break;
}
}
$retArray = array(
'status' => true,
'message' => null,
'done' => false,
);
if ($ret['Error'] != '')
{
$retArray['status'] = false;
$retArray['done'] = true;
$retArray['message'] = $ret['Error'];
}
else
{
$retArray['files'] = 0;
$retArray['bytesIn'] = 0;
$retArray['bytesOut'] = 0;
$retArray['factory'] = AKFactory::serialize();
$retArray['lastfile'] = 'Deleting: ' . $zapper->getSubstep();
}
$timer->enforce_min_exec_time();
return $retArray;
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* A simple INI-based i18n engine
*/
class AKText extends AKAbstractObject
{
/**
* The default (en_GB) translation used when no other translation is
available
*
* @var array
*/
private $default_translation = array(
'AUTOMODEON' => 'Auto-mode enabled',
'ERR_NOT_A_JPA_FILE' => 'The file is not a JPA archive',
'ERR_CORRUPT_ARCHIVE' => 'The archive file is corrupt,
truncated or archive parts are missing',
'ERR_INVALID_LOGIN' => 'Invalid login',
'COULDNT_CREATE_DIR' => 'Could not create %s folder',
'COULDNT_WRITE_FILE' => 'Could not open %s for writing.',
'WRONG_FTP_HOST' => 'Wrong FTP host or port',
'WRONG_FTP_USER' => 'Wrong FTP username or password',
'WRONG_FTP_PATH1' => 'Wrong FTP initial directory - the
directory doesn\'t exist',
'FTP_CANT_CREATE_DIR' => 'Could not create directory %s',
'FTP_TEMPDIR_NOT_WRITABLE' => 'Could not find or create a
writable temporary directory',
'SFTP_TEMPDIR_NOT_WRITABLE' => 'Could not find or create a
writable temporary directory',
'FTP_COULDNT_UPLOAD' => 'Could not upload %s',
'THINGS_HEADER' => 'Things you should know about
Akeeba Kickstart',
'THINGS_01' => 'Kickstart is not an installer. It
is an archive extraction tool. The actual installer was put inside the archive file
at backup time.',
'THINGS_02' => 'Kickstart is not the only way to
extract the backup archive. You can use Akeeba eXtract Wizard and upload the
extracted files using FTP instead.',
'THINGS_03' => 'Kickstart is bound by your
server\'s configuration. As such, it may not work at all.',
'THINGS_04' => 'You should download and upload
your archive files using FTP in Binary transfer mode. Any other method could lead
to a corrupt backup archive and restoration failure.',
'THINGS_05' => 'Post-restoration site load errors
are usually caused by .htaccess or php.ini directives. You should understand that
blank pages, 404 and 500 errors can usually be worked around by editing the
aforementioned files. It is not our job to mess with your configuration files,
because this could be dangerous for your site.',
'THINGS_06' => 'Kickstart overwrites files
without a warning. If you are not sure that you are OK with that do not continue.',
'THINGS_07' => 'Trying to restore to the
temporary URL of a cPanel host (e.g. http://1.2.3.4/~username) will lead to
restoration failure and your site will appear to be not working. This is normal and
it\'s just how your server and CMS software work.',
'THINGS_08' => 'You are supposed to read the
documentation before using this software. Most issues can be avoided, or easily
worked around, by understanding how this software works.',
'THINGS_09' => 'This text does not imply that
there is a problem detected. It is standard text displayed every time you launch
Kickstart.',
'CLOSE_LIGHTBOX' => 'Click here or press ESC to close
this message',
'SELECT_ARCHIVE' => 'Select a backup archive',
'ARCHIVE_FILE' => 'Archive file:',
'SELECT_EXTRACTION' => 'Select an extraction method',
'WRITE_TO_FILES' => 'Write to files:',
'WRITE_HYBRID' => 'Hybrid (use FTP only if needed)',
'WRITE_DIRECTLY' => 'Directly',
'WRITE_FTP' => 'Use FTP for all files',
'WRITE_SFTP' => 'Use SFTP for all files',
'FTP_HOST' => '(S)FTP host name:',
'FTP_PORT' => '(S)FTP port:',
'FTP_FTPS' => 'Use FTP over SSL (FTPS)',
'FTP_PASSIVE' => 'Use FTP Passive Mode',
'FTP_USER' => '(S)FTP user name:',
'FTP_PASS' => '(S)FTP password:',
'FTP_DIR' => '(S)FTP directory:',
'FTP_TEMPDIR' => 'Temporary directory:',
'FTP_CONNECTION_OK' => 'FTP Connection Established',
'SFTP_CONNECTION_OK' => 'SFTP Connection Established',
'FTP_CONNECTION_FAILURE' => 'The FTP Connection Failed',
'SFTP_CONNECTION_FAILURE' => 'The SFTP Connection Failed',
'FTP_TEMPDIR_WRITABLE' => 'The temporary directory is
writable.',
'FTP_TEMPDIR_UNWRITABLE' => 'The temporary directory is not
writable. Please check the permissions.',
'FTPBROWSER_ERROR_HOSTNAME' => "Invalid FTP host or port",
'FTPBROWSER_ERROR_USERPASS' => "Invalid FTP username or
password",
'FTPBROWSER_ERROR_NOACCESS' => "Directory doesn't exist or you
don't have enough permissions to access it",
'FTPBROWSER_ERROR_UNSUPPORTED' => "Sorry, your FTP server doesn't
support our FTP directory browser.",
'FTPBROWSER_LBL_GOPARENT' => "<up one level>",
'FTPBROWSER_LBL_INSTRUCTIONS' => 'Click on a directory to navigate
into it. Click on OK to select that directory, Cancel to abort the procedure.',
'FTPBROWSER_LBL_ERROR' => 'An error occurred',
'SFTP_NO_SSH2' => 'Your web server does not have the
SSH2 PHP module, therefore can not connect to SFTP servers.',
'SFTP_NO_FTP_SUPPORT' => 'Your SSH server does not allow
SFTP connections',
'SFTP_WRONG_USER' => 'Wrong SFTP username or password',
'SFTP_WRONG_STARTING_DIR' => 'You must supply a valid absolute
path',
'SFTPBROWSER_ERROR_NOACCESS' => "Directory doesn't exist or you
don't have enough permissions to access it",
'SFTP_COULDNT_UPLOAD' => 'Could not upload %s',
'SFTP_CANT_CREATE_DIR' => 'Could not create directory %s',
'UI-ROOT' => '<root>',
'CONFIG_UI_FTPBROWSER_TITLE' => 'FTP Directory Browser',
'FTP_BROWSE' => 'Browse',
'BTN_CHECK' => 'Check',
'BTN_RESET' => 'Reset',
'BTN_TESTFTPCON' => 'Test FTP connection',
'BTN_TESTSFTPCON' => 'Test SFTP connection',
'BTN_GOTOSTART' => 'Start over',
'FINE_TUNE' => 'Fine tune',
'MIN_EXEC_TIME' => 'Minimum execution time:',
'MAX_EXEC_TIME' => 'Maximum execution time:',
'TIME_SETTINGS_HELP' => "Increase the minimum to 3 if you
get AJAX errors. Increase the maximum to 10 for faster extraction, decrease back to
5 if you get AJAX errors. Try minimum 5, maximum 1 (not a typo!) if you keep
getting AJAX errors.",
'SECONDS_PER_STEP' => 'seconds per step',
'EXTRACT_FILES' => 'Extract files',
'BTN_START' => 'Start',
'EXTRACTING' => 'Extracting',
'DO_NOT_CLOSE_EXTRACT' => 'Do not close this window while
the extraction is in progress',
'RESTACLEANUP' => 'Restoration and Clean Up',
'BTN_RUNINSTALLER' => 'Run the Installer',
'BTN_CLEANUP' => 'Clean Up',
'BTN_SITEFE' => 'Visit your site\'s frontend',
'BTN_SITEBE' => 'Visit your site\'s backend',
'WARNINGS' => 'Extraction Warnings',
'ERROR_OCCURED' => 'An error occurred',
'STEALTH_MODE' => 'Stealth mode',
'STEALTH_MODE_HELP' => 'When enabled, only visitors from
your IP address will be able to see the site until the restoration is complete.
Everyone else will be redirected to and only see the URL above. Your server must
see the real IP of the visitor (this is controlled by your host, not you or us).',
'STEALTH_URL' => 'HTML file to show to web
visitors',
'ERR_NOT_A_JPS_FILE' => 'The file is not a JPA archive',
'ERR_INVALID_JPS_PASSWORD' => 'The password you gave is wrong or
the archive is corrupt',
'JPS_PASSWORD' => 'Archive Password (for JPS
files)',
'INVALID_FILE_HEADER' => 'Invalid header in archive file,
part %s, offset %s',
'NEEDSOMEHELPKS' => 'Want some help to use this tool?
Read this first:',
'QUICKSTART' => 'Quick Start Guide',
'CANTGETITTOWORK' => 'Can\'t get it to work? Click
me!',
'NOARCHIVESCLICKHERE' => 'No archives detected. Click here
for troubleshooting instructions.',
'POSTRESTORATIONTROUBLESHOOTING' => 'Something not working after the
restoration? Click here for troubleshooting instructions.',
'UPDATE_HEADER' => 'An updated version of Akeeba
Kickstart (<span id="update-version">unknown</span>) is available!',
'UPDATE_NOTICE' => 'You are advised to always use the
latest version of Akeeba Kickstart available. Older versions may be subject to bugs
and will not be supported.',
'UPDATE_DLNOW' => 'Download now',
'UPDATE_MOREINFO' => 'More information',
'IGNORE_MOST_ERRORS' => 'Ignore most errors',
'WRONG_FTP_PATH2' => 'Wrong FTP initial directory - the
directory doesn\'t correspond to your site\'s web root',
'ARCHIVE_DIRECTORY' => 'Archive directory:',
'RELOAD_ARCHIVES' => 'Reload',
'CONFIG_UI_SFTPBROWSER_TITLE' => 'SFTP Directory Browser',
'ERR_COULD_NOT_OPEN_ARCHIVE_PART' => 'Could not open archive part file
%s for reading. Check that the file exists, is readable by the web server and is
not in a directory made out of reach by chroot, open_basedir restrictions or any
other restriction put in place by your host.',
'RENAME_FILES' => 'Rename server configuration
files',
'RENAME_FILES_HELP' => 'Renames .htaccess, web.config,
php.ini and .user.ini contained in the archive while extracting. Files are renamed
with a .bak extension. The file names are restored when you click on Clean Up.',
'RESTORE_PERMISSIONS' => 'Restore file permissions',
'RESTORE_PERMISSIONS_HELP' => 'Applies the file permissions (but
NOT file ownership) which was stored at backup time. Only works with JPA and JPS
archives. Does not work on Windows (PHP does not offer such a feature).',
'EXTRACT_LIST' => 'Files to extract',
'EXTRACT_LIST_HELP' => 'Enter a file path such as
<code>images/cat.png</code> or shell pattern such as <code>images/*.png</code> on
each line. Only files matching this list will be written to disk. Leave empty to
extract everything (default).',
'ZAPBEFORE' => 'Delete everything before
extraction',
'ZAPBEFORE_HELP' => 'Tries to delete all existing
files and folders under the directory where Kickstart is stored before extracting
the backup archive. It DOES NOT take into account which files and folders exist in
the backup archive. Files and folders deleted by this feature CAN NOT be recovered.
<strong>WARNING! THIS MAY DELETE FILES AND FOLDERS WHICH DO NOT BELONG TO YOUR
SITE. USE WITH EXTREME CAUTION. BY ENABLING THIS FEATURE YOU ASSUME ALL
RESPONSIBILITY AND LIABILITY.</strong>',
);
/**
* The array holding the translation keys
*
* @var array
*/
private $strings;
/**
* The currently detected language (ISO code)
*
* @var string
*/
private $language;
/*
* Initializes the translation engine
* @return AKText
*/
public function __construct()
{
// Start with the default translation
$this->strings = $this->default_translation;
// Try loading the translation file in English, if it exists
$this->loadTranslation('en-GB');
// Try loading the translation file in the browser's preferred
language, if it exists
$this->getBrowserLanguage();
if (!is_null($this->language))
{
$this->loadTranslation();
}
}
if (empty($lang))
{
$lang = $this->language;
}
if (!is_array($this->strings))
{
$this->strings = array();
}
if (empty($temp))
{
$this->strings = array_merge($this->default_translation, $this-
>strings);
}
else
{
$this->strings = array_merge($this->strings, $temp);
}
}
/**
* A PHP based INI file parser.
*
* Thanks to asohn ~at~ aircanopy ~dot~ net for posting this handy function
on
* the parse_ini_file page on http://gr.php.net/parse_ini_file
*
* @param string $file Filename to process
* @param bool $process_sections True to also process INI sections
*
* @return array An associative array of sections, keys and values
* @access private
*/
public static function parse_ini_file($file, $process_sections = false,
$raw_data = false)
{
$process_sections = ($process_sections !== true) ? false : true;
if (!$raw_data)
{
$ini = @file($file);
}
else
{
$ini = $file;
}
if (count($ini) == 0)
{
return array();
}
$sections = array();
$values = array();
$result = array();
$globals = array();
$i = 0;
if (!empty($ini))
{
foreach ($ini as $line)
{
$line = trim($line);
$line = str_replace("\t", " ", $line);
// Comments
if (!preg_match('/^[a-zA-Z0-9[]/', $line))
{
continue;
}
// Sections
if ($line{0} == '[')
{
$tmp = explode(']', $line);
$sections[] = trim(substr($tmp[0], 1));
$i++;
continue;
}
// Key-value pair
list($key, $value) = explode('=', $line, 2);
$key = trim($key);
$value = trim($value);
if (strstr($value, ";"))
{
$tmp = explode(';', $value);
if (count($tmp) == 2)
{
if ((($value{0} != '"') && ($value{0} != "'"))
||
preg_match('/^".*"\s*;/', $value) ||
preg_match('/^".*;[^"]*$/', $value) ||
preg_match("/^'.*'\s*;/", $value) ||
preg_match("/^'.*;[^']*$/", $value)
)
{
$value = $tmp[0];
}
}
else
{
if ($value{0} == '"')
{
$value = preg_replace('/^"(.*)".*/',
'$1', $value);
}
elseif ($value{0} == "'")
{
$value = preg_replace("/^'(.*)'.*/",
'$1', $value);
}
else
{
$value = $tmp[0];
}
}
}
$value = trim($value);
$value = trim($value, "'\"");
if ($i == 0)
{
if (substr($line, -1, 2) == '[]')
{
$globals[$key][] = $value;
}
else
{
$globals[$key] = $value;
}
}
else
{
if (substr($line, -1, 2) == '[]')
{
$values[$i - 1][$key][] = $value;
}
else
{
$values[$i - 1][$key] = $value;
}
}
}
}
$this->language = null;
$basename = basename(__FILE__, '.php') . '.ini';
if (is_array($iniFiles))
{
foreach ($user_languages as $languageStruct)
{
if (is_null($this->language))
{
// Get files matching the main lang part
$iniFiles = $fs->getFiles(KSROOTDIR,
$languageStruct[1] . '-??.' . $basename);
if (count($iniFiles) > 0)
{
$filename = $iniFiles[0];
$filename = substr($filename,
strlen(KSROOTDIR) + 1);
$this->language = substr($filename, 0, 5);
}
else
{
$this->language = null;
}
}
}
}
if (is_null($this->language))
{
// Try to find a full language match
foreach ($user_languages as $languageStruct)
{
if (@file_exists($languageStruct[0] . '.' . $basename) &&
is_null($this->language))
{
$this->language = $languageStruct[0];
}
else
{
}
}
}
else
{
// Do we have an exact match?
foreach ($user_languages as $languageStruct)
{
if (substr($this->language, 0, strlen($languageStruct[1]))
== $languageStruct[1])
{
if (file_exists($languageStruct[0] . '.' .
$basename))
{
$this->language = $languageStruct[0];
}
}
}
}
return '';
}
/**
* Singleton pattern for Language
*
* @return AKText The global AKText instance
*/
public static function &getInstance()
{
static $instance;
if (!is_object($instance))
{
$instance = new AKText();
}
return $instance;
}
$key = strtoupper($string);
$key = substr($key, 0, 1) == '_' ? substr($key, 1) : $key;
if (isset ($text->strings[$key]))
{
$string = $text->strings[$key];
}
else
{
if (defined($string))
{
$string = constant($string);
}
}
return $string;
}
return $out;
}
return $out;
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* The Akeeba Kickstart Factory class
*
* This class is reponssible for instantiating all Akeeba Kickstart classes
*/
class AKFactory
{
/** @var array A list of instantiated objects */
private $objectlist = array();
/**
* Gets a serialized snapshot of the Factory for safekeeping (hibernate)
*
* @return string The serialized snapshot of the Factory
*/
public static function serialize()
{
$engine = self::getUnarchiver();
$engine->shutdown();
$serialized = serialize(self::getInstance());
if (function_exists('base64_encode') &&
function_exists('base64_decode'))
{
$serialized = base64_encode($serialized);
}
return $serialized;
}
/**
* Gets the unarchiver engine
*
* @return AKAbstractUnarchiver
*/
public static function &getUnarchiver($configOverride = null)
{
static $class_name;
if (empty($class_name))
{
$filetype = self::get('kickstart.setup.filetype', null);
if (empty($filetype))
{
$filename = self::get('kickstart.setup.sourcefile',
null);
$basename = basename($filename);
$baseextension = strtoupper(substr($basename, -3));
switch ($baseextension)
{
case 'JPA':
$filetype = 'JPA';
break;
case 'JPS':
$filetype = 'JPS';
break;
case 'ZIP':
$filetype = 'ZIP';
break;
default:
die('Invalid archive type or extension in file
' . $filename);
break;
}
}
if (empty($destdir))
{
$destdir = KSROOTDIR;
}
if ($object->getState() == 'init')
{
$sourcePath = self::get('kickstart.setup.sourcepath', '');
$sourceFile = self::get('kickstart.setup.sourcefile', '');
if (!empty($sourcePath))
{
$sourceFile = rtrim($sourcePath, '/\\') . '/' .
$sourceFile;
}
if (!defined('KICKSTART'))
{
// In restore.php mode we have to exclude the
restoration.php files
$moreSkippedFiles = array(
// Akeeba Backup for Joomla!
'administrator/components/com_akeeba/restoration.php'
,
// Joomla! Update
'administrator/components/com_joomlaupdate/restoratio
n.php',
// Akeeba Backup for WordPress
'wp-
content/plugins/akeebabackupwp/app/restoration.php',
'wp-
content/plugins/akeebabackupcorewp/app/restoration.php',
'wp-
content/plugins/akeebabackup/app/restoration.php',
'wp-
content/plugins/akeebabackupwpcore/app/restoration.php',
// Akeeba Solo
'app/restoration.php',
);
$config['skip_files'] = array_merge($config['skip_files'],
$moreSkippedFiles);
}
if (!empty($configOverride))
{
$config = array_merge($config, $configOverride);
}
$object->setup($config);
}
return $object;
}
// ========================================================================
// Public factory interface
// ========================================================================
public static function get($key, $default = null)
{
$self = self::getInstance();
if (array_key_exists($key, $self->varlist))
{
return $self->varlist[$key];
}
return $default;
}
/**
* Gets a single, internally used instance of the Factory
*
* @param string $serialized_data [optional] Serialized data to spawn the
instance from
*
* @return AKFactory A reference to the unique Factory object instance
*/
protected static function &getInstance($serialized_data = null)
{
if (!is_object(self::$instance) || !is_null($serialized_data))
{
if (!is_null($serialized_data))
{
self::$instance = unserialize($serialized_data);
return self::$instance;
}
return self::$instance;
}
/**
* Internal function which instantiates a class named $class_name.
* The autoloader
*
* @param string $class_name
*
* @return object
*/
protected static function &getClassInstance($class_name)
{
$self = self::getInstance();
if (!isset($self->objectlist[$class_name]))
{
$self->objectlist[$class_name] = new $class_name;
}
return $self->objectlist[$class_name];
}
// ========================================================================
// Public hash data storage interface
// ========================================================================
/**
* Regenerates the full Factory state from a serialized snapshot (resume)
*
* @param string $serialized_data The serialized snapshot to resume from
*/
public static function unserialize($serialized_data)
{
if (function_exists('base64_encode') &&
function_exists('base64_decode'))
{
$serialized_data = base64_decode($serialized_data);
}
self::getInstance($serialized_data);
}
/**
* Reset the internal factory state, freeing all previously created objects
*/
public static function nuke()
{
self::$instance = null;
}
// ========================================================================
// Akeeba Kickstart classes
// ========================================================================
/**
* Gets the post processing engine
*
* @param string $proc_engine
*
* @return AKAbstractPostproc
*/
public static function &getPostProc($proc_engine = null)
{
static $class_name;
if (empty($class_name))
{
if (empty($proc_engine))
{
$proc_engine = self::get('kickstart.procengine', 'direct');
}
return self::getClassInstance($class_name);
}
/**
* Get the a reference to the Akeeba Engine's timer
*
* @return AKCoreTimer
*/
public static function &getTimer()
{
return self::getClassInstance('AKCoreTimer');
}
/**
* Get an instance of the filesystem zapper
*
* @return AKUtilsZapper
*/
public static function &getZapper()
{
return self::getClassInstance('AKUtilsZapper');
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Interface for AES encryption adapters
*/
interface AKEncryptionAESAdapterInterface
{
/**
* Decrypts a string. Returns the raw binary ciphertext, zero-padded.
*
* @param string $plainText The plaintext to encrypt
* @param string $key The raw binary key (will be zero-padded
or chopped if its size is different than the block size)
*
* @return string The raw encrypted binary string.
*/
public function decrypt($plainText, $key);
/**
* Returns the encryption block size in bytes
*
* @return int
*/
public function getBlockSize();
/**
* Is this adapter supported?
*
* @return bool
*/
public function isSupported();
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Abstract AES encryption class
*/
abstract class AKEncryptionAESAdapterAbstract
{
/**
* Trims or zero-pads a key / IV
*
* @param string $key The key or IV to treat
* @param int $size The block size of the currently used algorithm
*
* @return null|string Null if $key is null, treated string of $size byte
length otherwise
*/
public function resizeKey($key, $size)
{
if (empty($key))
{
return null;
}
$keyLength = strlen($key);
if (function_exists('mb_strlen'))
{
$keyLength = mb_strlen($key, 'ASCII');
}
if ($keyLength == $size)
{
return $key;
}
if (function_exists('mb_strlen'))
{
$stringSize = mb_strlen($string, 'ASCII');
}
if ($stringSize == $blockSize)
{
return '';
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
if (!function_exists('mcrypt_get_iv_size'))
{
return false;
}
if (!function_exists('mcrypt_create_iv'))
{
return false;
}
if (!function_exists('mcrypt_encrypt'))
{
return false;
}
if (!function_exists('mcrypt_decrypt'))
{
return false;
}
if (!function_exists('mcrypt_list_algorithms'))
{
return false;
}
if (!function_exists('hash'))
{
return false;
}
if (!function_exists('hash_algos'))
{
return false;
}
$algorightms = mcrypt_list_algorithms();
if (!in_array('rijndael-128', $algorightms))
{
return false;
}
if (!in_array('rijndael-192', $algorightms))
{
return false;
}
if (!in_array('rijndael-256', $algorightms))
{
return false;
}
$algorightms = hash_algos();
if (!in_array('sha256', $algorightms))
{
return false;
}
return true;
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* The encryption method to use
*
* @var string
*/
protected $method = 'aes-128-cbc';
if (!function_exists('openssl_random_pseudo_bytes'))
{
return false;
}
if (!function_exists('openssl_cipher_iv_length'))
{
return false;
}
if (!function_exists('openssl_encrypt'))
{
return false;
}
if (!function_exists('openssl_decrypt'))
{
return false;
}
if (!function_exists('hash'))
{
return false;
}
if (!function_exists('hash_algos'))
{
return false;
}
$algorightms = openssl_get_cipher_methods();
if (!in_array('aes-128-cbc', $algorightms))
{
return false;
}
$algorightms = hash_algos();
if (!in_array('sha256', $algorightms))
{
return false;
}
return true;
}
/**
* @return int
*/
public function getBlockSize()
{
return openssl_cipher_iv_length($this->method);
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* AES implementation in PHP (c) Chris Veness 2005-2016.
* Right to use and adapt is granted for under a simple creative commons
attribution
* licence. No warranty of any form is offered.
*
* Heavily modified for Akeeba Backup by Nicholas K. Dionysopoulos
* Also added AES-128 CBC mode (with mcrypt and OpenSSL) on top of AES CTR
* Removed CTR encrypt / decrypt (no longer used)
*/
class AKEncryptionAES
{
// Sbox is pre-computed multiplicative inverse in GF(2^8) used in SubBytes
and KeyExpansion [�5.1.1]
protected static $Sbox =
array(0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, 0x30, 0x01, 0x67,
0x2b, 0xfe, 0xd7, 0xab, 0x76,
0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59, 0x47, 0xf0, 0xad, 0xd4, 0xa2,
0xaf, 0x9c, 0xa4, 0x72, 0xc0,
0xb7, 0xfd, 0x93, 0x26, 0x36, 0x3f, 0xf7, 0xcc, 0x34, 0xa5, 0xe5,
0xf1, 0x71, 0xd8, 0x31, 0x15,
0x04, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x05, 0x9a, 0x07, 0x12, 0x80,
0xe2, 0xeb, 0x27, 0xb2, 0x75,
0x09, 0x83, 0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0, 0x52, 0x3b, 0xd6,
0xb3, 0x29, 0xe3, 0x2f, 0x84,
0x53, 0xd1, 0x00, 0xed, 0x20, 0xfc, 0xb1, 0x5b, 0x6a, 0xcb, 0xbe,
0x39, 0x4a, 0x4c, 0x58, 0xcf,
0xd0, 0xef, 0xaa, 0xfb, 0x43, 0x4d, 0x33, 0x85, 0x45, 0xf9, 0x02,
0x7f, 0x50, 0x3c, 0x9f, 0xa8,
0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5, 0xbc, 0xb6, 0xda,
0x21, 0x10, 0xff, 0xf3, 0xd2,
0xcd, 0x0c, 0x13, 0xec, 0x5f, 0x97, 0x44, 0x17, 0xc4, 0xa7, 0x7e,
0x3d, 0x64, 0x5d, 0x19, 0x73,
0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, 0x46, 0xee, 0xb8,
0x14, 0xde, 0x5e, 0x0b, 0xdb,
0xe0, 0x32, 0x3a, 0x0a, 0x49, 0x06, 0x24, 0x5c, 0xc2, 0xd3, 0xac,
0x62, 0x91, 0x95, 0xe4, 0x79,
0xe7, 0xc8, 0x37, 0x6d, 0x8d, 0xd5, 0x4e, 0xa9, 0x6c, 0x56, 0xf4,
0xea, 0x65, 0x7a, 0xae, 0x08,
0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6, 0xb4, 0xc6, 0xe8, 0xdd, 0x74,
0x1f, 0x4b, 0xbd, 0x8b, 0x8a,
0x70, 0x3e, 0xb5, 0x66, 0x48, 0x03, 0xf6, 0x0e, 0x61, 0x35, 0x57,
0xb9, 0x86, 0xc1, 0x1d, 0x9e,
0xe1, 0xf8, 0x98, 0x11, 0x69, 0xd9, 0x8e, 0x94, 0x9b, 0x1e, 0x87,
0xe9, 0xce, 0x55, 0x28, 0xdf,
0x8c, 0xa1, 0x89, 0x0d, 0xbf, 0xe6, 0x42, 0x68, 0x41, 0x99, 0x2d,
0x0f, 0xb0, 0x54, 0xbb, 0x16);
// Rcon is Round Constant used for the Key Expansion [1st col is 2^(r-1) in
GF(2^8)] [�5.2]
protected static $Rcon = array(
array(0x00, 0x00, 0x00, 0x00),
array(0x01, 0x00, 0x00, 0x00),
array(0x02, 0x00, 0x00, 0x00),
array(0x04, 0x00, 0x00, 0x00),
array(0x08, 0x00, 0x00, 0x00),
array(0x10, 0x00, 0x00, 0x00),
array(0x20, 0x00, 0x00, 0x00),
array(0x40, 0x00, 0x00, 0x00),
array(0x80, 0x00, 0x00, 0x00),
array(0x1b, 0x00, 0x00, 0x00),
array(0x36, 0x00, 0x00, 0x00));
/**
* The algorithm to use for PBKDF2. Must be a supported hash_hmac algorithm.
Default: sha1
*
* @var string
*/
private static $pbkdf2Algorithm = 'sha1';
/**
* Number of iterations to use for PBKDF2
*
* @var int
*/
private static $pbkdf2Iterations = 1000;
/**
* Should we use a static salt for PBKDF2?
*
* @var int
*/
private static $pbkdf2UseStaticSalt = 0;
/**
* The static salt to use for PBKDF2
*
* @var string
*/
private static $pbkdf2StaticSalt = "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0";
/**
* AES Cipher function: encrypt 'input' with Rijndael algorithm
*
* @param array $input Message as byte-array (16 bytes)
* @param array $w key schedule as 2D byte-array (Nr+1 x Nb bytes) -
* generated from the cipher key by KeyExpansion()
*
* @return string Ciphertext as byte-array (16 bytes)
*/
protected static function Cipher($input, $w)
{
// main Cipher function [�5.1]
$Nb = 4; // block size (in words): no of columns in
state (fixed at 4 for AES)
$Nr = count($w) / $Nb - 1; // no of rounds: 10/12/14 for 128/192/256-
bit keys
return $output;
}
return $state;
}
return $s;
}
return $s;
}
/**
* Key expansion for Rijndael Cipher(): performs key expansion on cipher key
* to generate a key schedule
*
* @param array $key Cipher key byte-array (16 bytes)
*
* @return array Key schedule as 2D byte-array (Nr+1 x Nb bytes)
*/
protected static function KeyExpansion($key)
{
// generate Key Schedule from Cipher Key [�5.2]
$w = array();
$temp = array();
return $w;
}
/*
* Unsigned right shift function, since PHP has neither >>> operator nor
unsigned ints
*
* @param a number to be shifted (32-bit integer)
* @param b number of bits to shift a to the right (0..31)
* @return a right-shifted and zero-filled by b bits
*/
return $w;
}
return $a;
}
/**
* AES decryption in CBC mode. This is the standard mode (the CTR methods
* actually use Rijndael-128 in CTR mode, which - technically - isn't AES).
*
* It supports AES-128 only. It assumes that the last 4 bytes
* contain a little-endian unsigned long integer representing the unpadded
* data length.
*
* @since 3.0.1
* @author Nicholas K. Dionysopoulos
*
* @param string $ciphertext The data to encrypt
* @param string $password Encryption password
*
* @return string The plaintext
*/
public static function AESDecryptCBC($ciphertext, $password)
{
$adapter = self::getAdapter();
if (!$adapter->isSupported())
{
return false;
}
$params = self::getKeyDerivationParameters();
$keySizeBytes = $params['keySize'];
$algorithm = $params['algorithm'];
$iterations = $params['iterations'];
$useStaticSalt = $params['useStaticSalt'];
if (substr($salt, 0, 4) == 'JPST')
{
// We have a stored salt. Retrieve it and tell decrypt to process
the string minus the last 44 bytes
// (4 bytes for JPST, 16 bytes for the salt, 4 bytes for JPIV, 16
bytes for the IV, 4 bytes for the
// uncompressed string length - note that using PBKDF2 means
we're also using a randomized IV per the
// format specification).
$salt = substr($salt, 4);
$rightStringLimit -= 68;
if (substr($iv, 0, 4) == 'JPIV')
{
// We have a stored IV. Retrieve it and tell mdecrypt to process
the string minus the last 24 bytes
// (4 bytes for JPIV, 16 bytes for the IV, 4 bytes for the
uncompressed string length)
$iv = substr($iv, 4);
$rightStringLimit -= 20;
}
else
{
// No stored IV. Do it the dumb way.
$iv = self::createTheWrongIV($password);
}
// Decrypt
$plaintext = $adapter->decrypt($iv . substr($ciphertext, 0,
$rightStringLimit), $key);
return $plaintext;
}
/**
* That's the old way of creating an IV that's definitely not
cryptographically sound.
*
* DO NOT USE, EVER, UNLESS YOU WANT TO DECRYPT LEGACY DATA
*
* @param string $password The raw password from which we create an IV in a
super bozo way
*
* @return string A 16-byte IV string
*/
public static function createTheWrongIV($password)
{
static $ivs = array();
$key = md5($password);
if (!isset($ivs[$key]))
{
$nBytes = 16; // AES uses a 128 -bit (16 byte) block size,
hence the IV size is always 16 bytes
$pwBytes = array();
for ($i = 0; $i < $nBytes; $i++)
{
$pwBytes[$i] = ord(substr($password, $i, 1)) & 0xff;
}
$iv = self::Cipher($pwBytes, self::KeyExpansion($pwBytes));
$newIV = '';
foreach ($iv as $int)
{
$newIV .= chr($int);
}
$ivs[$key] = $newIV;
}
return $ivs[$key];
}
/**
* Expand the password to an appropriate 128-bit encryption key
*
* @param string $password
*
* @return string
*
* @since 5.2.0
* @author Nicholas K. Dionysopoulos
*/
public static function expandKey($password)
{
// Try to fetch cached key or create it if it doesn't exist
$nBits = 128;
$lookupKey = md5($password . '-' . $nBits);
if (array_key_exists($lookupKey, self::$passwords))
{
$key = self::$passwords[$lookupKey];
return $key;
}
// use AES itself to encrypt password to get cipher key (using plain
password as source for
// key expansion) - gives us well encrypted key.
$nBytes = $nBits / 8; // Number of bytes in key
$pwBytes = array();
$key = $newKey;
self::$passwords[$lookupKey] = $key;
return $key;
}
/**
* Returns the correct AES-128 CBC encryption adapter
*
* @return AKEncryptionAESAdapterInterface
*
* @since 5.2.0
* @author Nicholas K. Dionysopoulos
*/
public static function getAdapter()
{
static $adapter = null;
if (!$adapter->isSupported())
{
$adapter = new Mcrypt();
}
return $adapter;
}
/**
* @return string
*/
public static function getPbkdf2Algorithm()
{
return self::$pbkdf2Algorithm;
}
/**
* @param string $pbkdf2Algorithm
* @return void
*/
public static function setPbkdf2Algorithm($pbkdf2Algorithm)
{
self::$pbkdf2Algorithm = $pbkdf2Algorithm;
}
/**
* @return int
*/
public static function getPbkdf2Iterations()
{
return self::$pbkdf2Iterations;
}
/**
* @param int $pbkdf2Iterations
* @return void
*/
public static function setPbkdf2Iterations($pbkdf2Iterations)
{
self::$pbkdf2Iterations = $pbkdf2Iterations;
}
/**
* @return int
*/
public static function getPbkdf2UseStaticSalt()
{
return self::$pbkdf2UseStaticSalt;
}
/**
* @param int $pbkdf2UseStaticSalt
* @return void
*/
public static function setPbkdf2UseStaticSalt($pbkdf2UseStaticSalt)
{
self::$pbkdf2UseStaticSalt = $pbkdf2UseStaticSalt;
}
/**
* @return string
*/
public static function getPbkdf2StaticSalt()
{
return self::$pbkdf2StaticSalt;
}
/**
* @param string $pbkdf2StaticSalt
* @return void
*/
public static function setPbkdf2StaticSalt($pbkdf2StaticSalt)
{
self::$pbkdf2StaticSalt = $pbkdf2StaticSalt;
}
/**
* Get the parameters fed into PBKDF2 to expand the user password into an
encryption key. These are the static
* parameters (key size, hashing algorithm and number of iterations). A new
salt is used for each encryption block
* to minimize the risk of attacks against the password.
*
* @return array
*/
public static function getKeyDerivationParameters()
{
return array(
'keySize' => 16,
'algorithm' => self::$pbkdf2Algorithm,
'iterations' => self::$pbkdf2Iterations,
'useStaticSalt' => self::$pbkdf2UseStaticSalt,
'staticSalt' => self::$pbkdf2StaticSalt,
);
}
/**
* PBKDF2 key derivation function as defined by RSA's PKCS #5:
https://www.ietf.org/rfc/rfc2898.txt
*
* Test vectors can be found here: https://www.ietf.org/rfc/rfc6070.txt
*
* This implementation of PBKDF2 was originally created by https://defuse.ca
* With improvements by http://www.variations-of-shadow.com
* Modified for Akeeba Engine by Akeeba Ltd (removed unnecessary checks to
make it faster)
*
* @param string $password The password.
* @param string $salt A salt that is unique to the password.
* @param string $algorithm The hash algorithm to use. Default is sha1.
* @param int $count Iteration count. Higher is better, but
slower. Default: 1000.
* @param int $key_length The length of the derived key in bytes.
*
* @return string A string of $key_length bytes
*/
public static function pbkdf2($password, $salt, $algorithm = 'sha1', $count =
1000, $key_length = 16)
{
if (function_exists("hash_pbkdf2"))
{
return hash_pbkdf2($algorithm, $password, $salt, $count,
$key_length, true);
}
$output = "";
// First iteration
$xorResult = hash_hmac($algorithm, $last, $password, true);
$last = $xorResult;
$output .= $xorResult;
}
if (!array_key_exists($lookupKey, self::$passwords))
{
self::$passwords[$lookupKey] = self::pbkdf2($password,
$staticSalt, $algorithm, $iterations, $keySizeBytes);
}
return self::$passwords[$lookupKey];
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* A timing safe equals comparison
*
* @param string $safe The internal (safe) value to be checked
* @param string $user The user submitted (unsafe) value
*
* @return boolean True if the two strings are identical.
*
* @see http://blog.ircmaxell.com/2014/11/its-all-about-time.html
*/
function timingSafeEquals($safe, $user)
{
$safeLen = strlen($safe);
$userLen = strlen($user);
if ($userLen != $safeLen)
{
return false;
}
$result = 0;
/**
* The Master Setup will read the configuration parameters from restoration.php or
* the JSON-encoded "configuration" input variable and return the status.
*
* @return bool True if the master configuration was applied to the Factory object
*/
function masterSetup()
{
// ------------------------------------------------------------
// 1. Import basic setup parameters
// ------------------------------------------------------------
$ini_data = null;
if (!file_exists($setupFile))
{
AKFactory::set('kickstart.enabled', false);
return false;
}
/**
* If the setup file was created more than 1.5 hours ago we can assume
that it's stale and someone forgot to
* remove it from the server. This hinders brute force attacks against
the Kickstart password. Even a simple
* 8 character simple alphanum (a-z, 0-9) password yields over 2.8e12.
Assuming a very fast server which can
* serve 100 requests to restore.php per second and an easy to attack
password requiring going over just 1% of
* the search space it'd still take over 282 million seconds to brute
force it. Our limit is more than 4 orders
* of magnitude lower than this best practical case scenario, giving us
adequate protection against all but the
* luckiest attacker (spoiler alert: the mathematics of probabilities
say you're not gonna get lucky).
*
* It is still advisable to remove the restoration.php file once you
are done with the extraction. This check
* here is only meant as a failsafe in case of a server error during
the extraction and subsequent lack of user
* action to remove the restoration.php file from their server.
*/
$setupFieCreationTime = filectime($setupFile);
return false;
}
$ini_data = $restoration_setup;
if (empty($ini_data))
{
// No parameters fetched. Darn, how am I supposed to work like
that?!
AKFactory::set('kickstart.enabled', false);
return false;
}
AKFactory::set('kickstart.enabled', true);
}
else
{
// Maybe we have $restoration_setup defined in the head of
kickstart.php
global $restoration_setup;
// Reinitialize $ini_data
$ini_data = null;
/**
* August 2018. Some third party developer with a dubious skill level (or
complete lack thereof) wrote a piece of
* code which uses restore.php with an empty password (and never deleted the
restoration.php file he created).
* According to his code comments he did this because he couldn't figure out
how to make encrypted requests work,
* DESPITE THE FACT that com_joomlaupdate (part of Joomla! itself) has
working code which does EXACTLY THAT. >:-o
*
* As a result of his actions all sites running his software have a massive
vulnerability inflicted upon them. An
* attacker can absuse the (unlocked) restore.php to upload and install any
arbitrary code in a ZIP archive,
* possibly overwriting core code. Discovering this problem takes a few
seconds and there is code which is doing
* exactly that published years ago (during the active maintenance period of
Joomla! 3.4, that long ago).
*
* This bit of code here detects an empty password and disables restore.php.
His badly written software fails to
* execute and, most importantly, the unlucky users of his software will no
longer have a remote code upload /
* remote code execution vulnerability on their sites.
*
* Remember, people, if you can't be bothered to take web application
security seriously DO NOT SELL WEB SOFTWARE
* FOR A LIVING. There are other honest jobs you can do which don't involve
using a computer in a dangerous and
* irresponsible manner.
*/
$password = AKFactory::get('kickstart.security.password', null);
return false;
}
// ------------------------------------------------------------
// 2. Explode JSON parameters into $_REQUEST scope
// ------------------------------------------------------------
if (!empty($_POST))
{
foreach ($_POST as $key => $value)
{
unset($_POST[$key]);
}
}
if (!empty($_GET))
{
foreach ($_GET as $key => $value)
{
unset($_GET[$key]);
}
}
/**
* Akeeba Restore 5.3.1 and earlier use a custom implementation of AES-128 in
CTR mode to encrypt the JSON data
* between client and server. This is not used as a means to maintain secrecy
(it's symmetrical encryption and the
* key is, by necessity, transmitted with the HTML page to the client). It's
meant as a form of authentication, so
* that the server part can ensure that it only receives commands by an
authorized client.
*
* The downside is that encryption in CTR mode (like CBC) is an all-or-
nothing affair. This opens the possibility
* for a padding oracle attack
(https://en.wikipedia.org/wiki/Padding_oracle_attack). While Akeeba Restore was
* hardened in 2014 to prevent the bulk of suck attacks it is still possible
to attack the encryption using a very
* large number of requests (several dozens of thousands).
*
* Since Akeeba Restore 5.4.0 we have removed this authentication method and
replaced it with the transmission of a
* very large length password. On the server side we use a timing safe
password comparison. By its very nature, it
* will only leak the (well known, constant and large) length of the password
but no more information about the
* password itself. See http://blog.ircmaxell.com/2014/11/its-all-about-
time.html As a result this form of
* authentication is many orders of magnitude harder to crack than regular
encryption.
*
* Now you may wonder "how is sending a password in the clear hardier than
encryption?". If you ask that question
* you were not paying attention. The password needs to be known by BOTH the
server AND the client (browser). Since
* this password is generated programmatically by the server, it MUST be sent
to the client by the server. If an
* attacker is able to intercept this transmission (man in the middle attack)
using encryption is irrelevant: the
* attacker already knows your password. This situation also applies when the
user sends their own password to the
* server, e.g. when logging into their site. The ONLY way to avoid security
issues regarding information being
* stolen in transit is using HTTPS with a commercially signed SSL
certificate. Unlike 2008, when Kickstart was
* originally written, obtaining such a certificate nowadays is trivial and
costs absolutely nothing thanks to Let's
* Encrypt (https://letsencrypt.org/).
*
* TL;DR: Use HTTPS with a commercially signed SSL certificate, e.g. a free
certificate from Let's Encrypt. Client-
* side cryptography does NOT protect you against an attacker (see
* https://www.nccgroup.trust/us/about-us/newsroom-and-
events/blog/2011/august/javascript-cryptography-considered-harmful/).
* Moreover, sending a plaintext password is safer than relying on client-
side encryption for authentication as it
* removes the possibility of an attacker inferring the contents of the
authentication key (password) in a relatively
* easy and automated manner.
*/
if (!empty($password))
{
// Timing-safe password comparison. See
http://blog.ircmaxell.com/2014/11/its-all-about-time.html
if (!timingSafeEquals($password, $userPassword))
{
die('###{"status":false,"message":"Invalid login"}###');
}
}
// ------------------------------------------------------------
// 3. Try the "factory" variable
// ------------------------------------------------------------
// A "factory" variable will override all other settings.
$serialized = getQueryParam('factory', null);
if (!is_null($serialized))
{
// Get the serialized factory
AKFactory::unserialize($serialized);
AKFactory::set('kickstart.enabled', true);
return true;
}
// ------------------------------------------------------------
// 4. Try the configuration variable for Kickstart
// ------------------------------------------------------------
if (defined('KICKSTART'))
{
$configuration = getQueryParam('configuration');
if (!is_null($configuration))
{
// Let's decode the configuration from JSON to array
$ini_data = json_decode($configuration, true);
}
else
{
// Neither exists. Enable Kickstart's interface anyway.
$ini_data = array('kickstart.enabled' => true);
}
AKFactory::set('kickstart.enabled', true);
return true;
}
}
}
/**
* Akeeba Restore
* A JSON-powered JPA, JPS and ZIP archive extraction library
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
if (!array_key_exists('type', get_object_vars($message)))
{
return;
}
if ($message->type == 'startfile')
{
$this->filesProcessed++;
$this->compressedTotal += $message->content->compressed;
$this->uncompressedTotal += $message->content-
>uncompressed;
}
}
// Import configuration
masterSetup();
$retArray = array(
'status' => true,
'message' => null
);
if ($enabled)
{
$task = getQueryParam('task');
switch ($task)
{
case 'ping':
// ping task - really does nothing!
$timer = AKFactory::getTimer();
$timer->enforce_min_exec_time();
break;
/**
* There are two separate steps here since we were using an
inefficient restoration initialization method in
* the past. Now both startRestore and stepRestore are identical.
The difference in behavior depends
* exclusively on the calling Javascript. If no serialized
factory was passed in the request then we start a
* new restoration. If a serialized factory was passed in the
request then the restoration is resumed. For
* this reason we should NEVER call AKFactory::nuke() in
startRestore anymore: that would simply reset the
* extraction engine configuration which was done in
masterSetup() leading to an error about the file being
* invalid (since no file is found).
*/
case 'startRestore':
case 'stepRestore':
/**
* First try to run the filesystem zapper (remove all
existing files and folders). If the Zapper is
* disabled or has already finished running we will get a
FALSE result. Otherwise it's a status array
* which we can pass directly back to the caller.
*/
$ret = runZapper();
break;
}
if ($ret['Error'] != '')
{
$retArray['status'] = false;
$retArray['done'] = true;
$retArray['message'] = $ret['Error'];
}
elseif (!$ret['HasRun'])
{
$retArray['files'] = $observer->filesProcessed;
$retArray['bytesIn'] = $observer->compressedTotal;
$retArray['bytesOut'] = $observer->uncompressedTotal;
$retArray['status'] = true;
$retArray['done'] = true;
}
else
{
$retArray['files'] = $observer->filesProcessed;
$retArray['bytesIn'] = $observer->compressedTotal;
$retArray['bytesOut'] = $observer->uncompressedTotal;
$retArray['status'] = true;
$retArray['done'] = false;
$retArray['factory'] = AKFactory::serialize();
}
$timer = AKFactory::getTimer();
$timer->enforce_min_exec_time();
break;
case 'finalizeRestore':
$root = AKFactory::get('kickstart.setup.destdir');
// Remove the installation directory
recursive_remove_directory($root . '/installation');
$postproc = AKFactory::getPostProc();
/**
* Should I rename the htaccess.bak and web.config.bak
files back to their live filenames...?
*/
$renameFiles =
AKFactory::get('kickstart.setup.postrenamefiles', true);
if ($renameFiles)
{
// Rename htaccess.bak to .htaccess
if (file_exists($root . '/htaccess.bak'))
{
if (file_exists($root . '/.htaccess'))
{
$postproc->unlink($root . '/.htaccess');
}
$postproc->rename($root . '/htaccess.bak',
$root . '/.htaccess');
}
// Remove restoration.php
$basepath = KSROOTDIR;
$basepath = rtrim(str_replace('\\', '/', $basepath), '/');
if (!empty($basepath))
{
$basepath .= '/';
}
$postproc->unlink($basepath . 'restoration.php');
default:
// Invalid task!
$enabled = false;
break;
}
}
// return success
return true;
}
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
class AKKickstartUtils
{
/**
* Guess the best path containing backup archives. The default strategy is
check in the current directory first,
* then attempt to find an Akeeba Backup for Joomla!, Akeeba Solo or Akeeba
Backup for WordPress default backup
* output directory under the current root. The first one containing backup
archives wins.
*
* @return string The path to get archives from
*/
public static function getBestArchivePath()
{
$basePath = self::getPath();
$basePathSlash = (empty($basePath) ? '.' : rtrim($basePath, '/\\')) .
'/';
$paths = array(
// Root, same as the directory we're in
$basePath,
// Standard temporary directory
$basePath . '/kicktemp',
// Akeeba Backup for Joomla!, default output directory
$basePathSlash . 'administrator/components/com_akeeba/backup',
// Akeeba Solo, default output directory
$basePathSlash . 'backups',
// Akeeba Backup for WordPress, default output directory
$basePathSlash . 'wp-content/plugins/akeebabackupwp/app/backups',
);
if (!empty($archives))
{
return $path;
}
}
return $basePath;
}
/**
* Gets the directory the file is in
*
* @return string
*/
public static function getPath()
{
$path = KSROOTDIR;
$path = rtrim(str_replace('\\', '/', $path), '/');
if (!empty($path))
{
$path .= '/';
}
return $path;
}
/**
* Scans the current directory for archive files (JPA, JPS and ZIP format)
*
* @param string $path The path to look for archives. null for automatic path
*
* @return array
*/
public static function findArchives($path)
{
$ret = array();
if (empty($path))
{
$path = self::getPath();
}
if (empty($path))
{
$path = '.';
}
$dh = @opendir($path);
if ($dotpos == strlen($file))
{
continue;
}
closedir($dh);
if (!empty($ret))
{
return $ret;
}
// On some hosts using opendir doesn't work. Let's try Dir instead
$d = dir($path);
if ($dotpos == strlen($file))
{
continue;
}
return $ret;
}
/**
* Gets the most appropriate temporary path
*
* @return string
*/
public static function getTemporaryPath()
{
$path = self::getPath();
$candidateDirs = array(
$path,
$path . '/kicktemp',
);
if (function_exists('sys_get_temp_dir'))
{
$candidateDirs[] = sys_get_temp_dir();
}
// Failsafe
return $path;
}
/**
* Scans the current directory for archive files and returns them as <OPTION>
tags
*
* @param string $path The path to look for archives. null for automatic path
*
* @return string
*/
public static function getArchivesAsOptions($path = null)
{
$ret = '';
$archives = self::findArchives($path);
if (empty($archives))
{
return $ret;
}
return $ret;
}
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
class ExtractionObserver extends AKAbstractPartObserver
{
public $compressedTotal = 0;
public $uncompressedTotal = 0;
public $filesProcessed = 0;
public $totalSize = null;
public $fileList = null;
public $lastFile = '';
if (!array_key_exists('type', get_object_vars($message)))
{
return;
}
switch ($message->type)
{
// Sent when we read the list of archive parts and their total
size
case 'totalsize':
$this->totalSize = $message->content->totalsize;
$this->fileList = $message->content->filelist;
break;
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
if (!is_array($extraFeatureObjects))
{
$extraFeatureObjects = array();
$allClasses = get_declared_classes();
foreach ($allClasses as $class)
{
if (substr($class, 0, 9) == 'AKFeature')
{
$extraFeatureObjects[] = new $class;
}
}
}
if (is_null($method))
{
return;
}
if (empty($extraFeatureObjects))
{
return;
}
$result = null;
foreach ($extraFeatureObjects as $o)
{
if (!method_exists($o, $method))
{
continue;
}
$result = call_user_func(array($o, $method), $params);
}
return $result;
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Removes trailing slash or backslash from a pathname
*
* @param string $path The path to treat
*
* @return string The path without the trailing slash/backslash
*/
function TrimTrailingSlash($path)
{
$newpath = $path;
function TranslateWinPath($p_path)
{
$is_unc = false;
if (KSWINDOWS)
{
// Is this a UNC path?
$is_unc = (substr($p_path, 0, 2) == '\\\\') || (substr($p_path, 0, 2)
== '//');
// Change potential windows directory separator
if ((strpos($p_path, '\\') > 0) || (substr($p_path, 0, 1) == '\\'))
{
$p_path = strtr($p_path, '\\', '/');
}
}
return $p_path;
}
/**
* FTP Functions
*/
function getListing($directory, $host, $port, $username, $password, $passive, $ssl)
{
$directory = resolvePath($directory);
$dir = $directory;
// Login
$result = @ftp_login($con, $username, $password);
if ($result === false)
{
return array(
'error' => 'FTPBROWSER_ERROR_USERPASS'
);
}
return array(
'error' => '',
'list' => $folders,
'breadcrumbs' => $parts,
'directory' => $directory,
'parent' => $parent_directory
);
}
function parse_rawlist($list)
{
$folders = array();
foreach ($list as $v)
{
$info = array();
$vinfo = preg_split("/[\s]+/", $v, 9);
if ($vinfo[0] !== "total")
{
$perms = $vinfo[0];
if (substr($perms, 0, 1) == 'd')
{
$folders[] = $vinfo[8];
}
}
}
asort($folders);
return $folders;
}
// Initialise
$connection = null;
$sftphandle = null;
// Open a connection
if (!function_exists('ssh2_connect'))
{
return array(
'error' => AKText::_('SFTP_NO_SSH2')
);
}
$sftphandle = ssh2_sftp($connection);
if (empty($dir))
{
$dir = ssh2_sftp_realpath($sftphandle, ".");
$directory = $dir;
if (!empty($copy_of_parts))
{
$parent_directory = '/' . implode('/', $copy_of_parts);
}
else
{
$parent_directory = '/';
}
}
else
{
$parent_directory = '';
}
}
$handle = opendir("ssh2.sftp://$sftphandle/$dir");
if (!is_resource($handle))
{
return array(
'error' => AKText::_('SFTPBROWSER_ERROR_NOACCESS')
);
}
$list[] = $entry;
}
closedir($handle);
if (!empty($list))
{
asort($list);
}
return array(
'error' => '',
'list' => $list,
'breadcrumbs' => $parts,
'directory' => $directory,
'parent' => $parent_directory
);
}
/**
* Simple function to resolve relative paths.
* Note that it is unable to resolve pathnames any higher than the present working
directory.
* I.E. It doesn't know about any directory names that you don't tell it about;
hence: ../../foo becomes foo.
*
* @param $filename
*
* @return string
*/
function resolvePath($filename)
{
$filename = str_replace('//', '/', $filename);
$parts = explode('/', $filename);
$out = array();
foreach ($parts as $part)
{
if ($part == '.')
{
continue;
}
if ($part == '..')
{
array_pop($out);
continue;
}
$out[] = $part;
}
function createStealthURL()
{
$filename = AKFactory::get('kickstart.stealth.url', '');
// We need an HTML file!
if (empty($filename))
{
return;
}
// Make sure it ends in .html or .htm
$filename = basename($filename);
if ((strtolower(substr($filename, -5)) != '.html') &&
(strtolower(substr($filename, -4)) != '.htm'))
{
return;
}
// Get the IP
$userIP = $_SERVER['REMOTE_ADDR'];
$userIP = str_replace('.', '\.', $userIP);
ENDHTACCESS;
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
function echoCSS()
{
echo <<<CSS
html {
background: #f0f0f0;
font-size: 62.5%;
}
body {
font-size: 12pt;
font-family: Calibri, "Helvetica Neue", Helvetica, Arial, sans-serif;
text-rendering: optimizeLegibility;
background: transparent;
color:#555;
width:100%;
max-width:980px;
margin: 0 auto;
}
#page-container {
position:relative;
margin:5% 0;
background: #f9f9f9;
border: medium solid #ddd;
}
#header {
color: #555;
background: #eaeaea;
background-clip: padding-box;
margin-bottom: 0.7em;
border-bottom: 2px solid #ddd;
padding:.25em;
font-size: 24pt;
line-height: 1.2;
text-align: center;
}
#footer {
font-size: 9pt;
color: #999;
text-align: center;
border-top: 1px solid #ddd;
padding: 1em 1em;
background: #eaeaea;
clear: both;
}
#footer a {
color: #88a;
text-decoration: none;
}
#error, .error {
x-display: none;
border: solid #cc0000;
border-width: 4px 0;
background: rgb(255,255,136);
color: #990000;
padding:1em 2em;
margin-bottom: 1.15em;
text-align:center;
}
.clr {
clear: both;
}
.circle {
display: block;
float: left;
border-radius: 2em;
border: 2px solid #e5e5e5;
font-weight: bold;
font-size: 14pt;
line-height:1.5em;
color: #fff;
height: 1.5em;
width: 1.5em;
margin: 0.75em;
text-align: center;
background: rgb(35,83,138);
}
.area-container {
margin: 1em 2em;
}
#page2a .area-container {
margin: 1em 0;
}
#runInstaller,
#runCleanup,
#gotoSite,
#gotoAdministrator,
#gotoPostRestorationRroubleshooting {
margin: 0 2em 1.3em;
}
h2 {
font-size: 18pt;
font-weight: normal;
line-height: 1.3;
border: solid #ddd;
border-left:none;
border-right:none;
padding: 0.5em 0;
background: #eaeaea;
}
#preextraction h2 {
margin-top:0;
border-top:0;
text-align:center;
}
input,
select,
textarea {
font-size : 100%;
margin : 0;
vertical-align : baseline;
*vertical-align: middle;
}
button,
input {
line-height : normal;
font-weight:normal;
*overflow: visible;
}
input,
select,
textarea {
background:#fff;
color:#777;
font-size: 12pt;
border:1px solid #d5d5d5;
border-radius: .25em;
box-sizing: border-box;
width:50%;
padding:0 0 0 .5em;
}
input[type="checkbox"] {
width:auto;
}
.field {
height:1.5em;
}
label {
display:inline-block;
width:30%;
font-size: 95%;
font-weight: normal;
cursor : pointer;
color: #333;
margin:.5em 0;
}
.help {
width: 60%;
margin-left: 30%;
margin-bottom: 1.5em;
font-size: small;
color: #888;
}
input:focus, input:hover {
background-color: #f6f6ff;
}
.button {
display: inline-block;
margin: 1em .25em;
padding: 1em 2em;
background: #2cb12c;
color:#fff;
border: 1px solid #ccc;
cursor: pointer;
border-radius: .25em;
transition: 0.3s linear all;
}
#checkFTPTempDir.button,
#resetFTPTempDir.button,
#testFTP.button,
#browseFTP,
#reloadArchives,
#notWorking.button {
padding: .5em 1em;
}
.button:hover {
background: #259625;
}
.button:active {
background: #3c3;
border: 1px solid #eaeaea;
border-radius: .25em;
}
#notWorking.button, .bluebutton {
text-decoration: none;
background: #66a8ff;
}
#notWorking.button:hover, .bluebutton:hover {
background: #4096ee;
}
#notWorking.button:active, .bluebutton:active {
background: #7abcff;
}
.loprofile {
padding: 0.5em 1em;
font-size: 80%;
}
.black_overlay{
display: none;
position: absolute;
top: 0%;
left: 0%;
width: 100%;
height: 100%;
background-color: black;
z-index:1001;
-moz-opacity: 0.8;
opacity:.80;
filter: alpha(opacity=80);
}
.white_content {
display: none;
position: absolute;
padding: 0 0 1em;
background: #fff;
border: 1px solid #ddd;
border: 1px solid rgba(0,0,0,.3);
z-index:1002;
overflow: hidden;
}
.white_content a{
margin-left:4em;
}
ol {
margin:0 2em;
padding:0 2em 1em;
}
li {
margin : 0 0 .5em;
}
#genericerror {
background-color: #f0f000 !important;
border: 4px solid #fcc !important;
}
#genericerrorInner {
font-size: 110%;
color: #33000;
}
#warn-not-close, .warn-not-close {
padding: 0.2em 0.5em;
text-align: center;
background: #fcfc00;
font-size: smaller;
font-weight: bold;
}
#progressbar, .progressbar {
display: block;
width: 80%;
height: 32px;
border: 1px solid #ccc;
margin: 1em 10% 0.2em;
border-radius: .25em;
}
#progressbar-inner, .progressbar-inner {
display: block;
width: 100%;
height: 100%;
background: #4096ee;
}
#currentFile {
font-family: Consolas, "Courier New", Courier, monospace;
font-size: 9pt;
height: 10pt;
overflow: hidden;
text-overflow: ellipsis;
background: #ccc;
margin: 0 10% 1em;
padding:.125em;
}
#extractionComplete {
}
#warningsContainer {
border-bottom: 2px solid brown;
border-left: 2px solid brown;
border-right: 2px solid brown;
padding: 5px 0;
background: #ffffcc;
border-bottom-right-radius: 5px;
border-bottom-left-radius: 5px;
}
#warningsHeader h2 {
color: black;
border-top: 2px solid brown;
border-left: 2px solid brown;
border-right: 2px solid brown;
border-bottom: thin solid brown;
border-top-right-radius: 5px;
border-top-left-radius: 5px;
background: yellow;
font-size: large;
padding: 2px 5px;
margin: 0px;
}
#warnings {
height: 200px;
overflow-y: scroll;
}
#warnings div {
background: #eeeeee;
font-size: small;
padding: 2px 4px;
border-bottom: thin solid #333333;
}
#automode {
display: inline-block;
padding: 6pt 12pt;
background-color: #cc0000;
border: thick solid yellow;
color: white;
font-weight: bold;
font-size: 125%;
position: absolute;
float: right;
top: 1em;
right: 1em;
}
.helpme,
#warn-not-close {
background: rgb(255,255,136);
padding: 0.75em 0.5em;
border: solid #febf01;
border-width: 1px 0;
text-align: center;
}
#update-notification {
margin: 1em;
padding: 0.5em;
background-color: #FF9;
color: #F33;
text-align: center;
border-radius: 20px;
border: medium solid red;
}
.update-notify {
font-size: 20pt;
font-weight: bold;
}
.update-links {
color: #333;
font-size: 14pt;
}
#update-dlnow {
text-decoration: none;
color: #333;
border: thin solid #333;
padding: 0.5em;
border-radius: 5px;
background-color: #f0f0f0;
}
#update-dlnow:hover {
background-color: #fff;
}
#update-whatsnew {
font-size: 11pt;
color: blue;
text-decoration: underline;
}
.update-whyupdate {
color: #333;
font-size: 9pt;
}
/* FTP Browser */
.breadcrumb {background-color: #F5F5F5; border-radius: 4px; list-style: none
outside none; margin: 0 0 18px; padding: 8px 15px;}
.breadcrumb > li {display: inline-block; text-shadow: 0 1px 0 #FFFFFF;}
#ak_crumbs span {padding: 1px 3px;}
#ak_crumbs a {cursor: pointer;}
#ftpBrowserFolderList a{cursor:pointer}
/* Bootstrap porting */
.table {margin-bottom: 18px;width: 100%;}
.table th, .table td {border-top: 1px solid #DDDDDD; line-height: 18px; padding:
8px; text-align: left; vertical-align: top;}
.table-striped tbody > tr:nth-child(2n+1) > td, .table-striped tbody > tr:nth-
child(2n+1) > th { background-color: #F9F9F9;}
/* Layout helpers
----------------------------------*/
.ui-helper-hidden { display: none; }
.ui-helper-hidden-accessible { border: 0; clip: rect(0 0 0 0); height: 1px; margin:
-1px; overflow: hidden; padding: 0; position: absolute; width: 1px; }
.ui-helper-reset { margin: 0; padding: 0; border: 0; outline: 0; line-height: 1.3;
text-decoration: none; font-size: 100%; list-style: none; }
.ui-helper-clearfix:before, .ui-helper-clearfix:after { content: ""; display:
table; }
.ui-helper-clearfix:after { clear: both; }
.ui-helper-clearfix { zoom: 1; }
.ui-helper-zfix { width: 100%; height: 100%; top: 0; left: 0; position: absolute;
opacity: 0; filter:Alpha(Opacity=0); }
/* Interaction Cues
----------------------------------*/
.ui-state-disabled { cursor: default !important; }
/* Icons
----------------------------------*/
/* Misc visuals
----------------------------------*/
/* Overlays */
.ui-widget-overlay { position: absolute; top: 0; left: 0; width: 100%; height:
100%; }
.ui-resizable { position: relative;}
.ui-resizable-handle { position: absolute;font-size: 0.1px; display: block; }
.ui-resizable-disabled .ui-resizable-handle, .ui-resizable-autohide .ui-resizable-
handle { display: none; }
.ui-resizable-n { cursor: n-resize; height: 7px; width: 100%; top: -5px; left: 0; }
.ui-resizable-s { cursor: s-resize; height: 7px; width: 100%; bottom: -5px; left:
0; }
.ui-resizable-e { cursor: e-resize; width: 7px; right: -5px; top: 0; height:
100%; }
.ui-resizable-w { cursor: w-resize; width: 7px; left: -5px; top: 0; height: 100%; }
.ui-resizable-se { cursor: se-resize; width: 12px; height: 12px; right: 1px;
bottom: 1px; }
.ui-resizable-sw { cursor: sw-resize; width: 9px; height: 9px; left: -5px; bottom:
-5px; }
.ui-resizable-nw { cursor: nw-resize; width: 9px; height: 9px; left: -5px; top:
-5px; }
.ui-resizable-ne { cursor: ne-resize; width: 9px; height: 9px; right: -5px; top:
-5px;}
.ui-button { display: inline-block; position: relative; padding: 0; margin-right: .
1em; cursor: pointer; text-align: center; zoom: 1; overflow: visible; } /* the
overflow property removes extra width in IE */
.ui-button, .ui-button:link, .ui-button:visited, .ui-button:hover, .ui-
button:active { text-decoration: none; }
.ui-button-icon-only { width: 2.2em; } /* to make room for the icon, a width needs
to be set here */
button.ui-button-icon-only { width: 2.4em; } /* button elements seem to need a
little more width */
.ui-button-icons-only { width: 3.4em; }
button.ui-button-icons-only { width: 3.7em; }
/*button sets*/
.ui-buttonset { margin-right: 7px; }
.ui-buttonset .ui-button { margin-left: 0; margin-right: -.3em; }
/* workarounds */
button.ui-button::-moz-focus-inner { border: 0; padding: 0; } /* reset extra
padding in Firefox */
.ui-dialog { position: absolute; top: 0; left: 0; padding: .2em; width: 300px;
overflow: hidden; }
.ui-dialog .ui-dialog-titlebar { padding: .4em 1em; position: relative; }
.ui-dialog .ui-dialog-title { float: left; margin: .1em 16px .1em 0; }
.ui-dialog .ui-dialog-titlebar-close { position: absolute; right: .3em; top: 50%;
width: 19px; margin: -10px 0 0 0; padding: 1px; height: 18px; display:none}
.ui-dialog .ui-dialog-titlebar-close span { display: none; margin: 1px; }
.ui-dialog .ui-dialog-titlebar-close:hover, .ui-dialog .ui-dialog-titlebar-
close:focus { padding: 0; }
.ui-dialog .ui-dialog-content { position: relative; border: 0; padding: .5em 1em;
background: none; overflow: auto; zoom: 1; }
.ui-dialog .ui-dialog-buttonpane { text-align: left; border-width: 1px 0 0 0;
background-image: none; margin: .5em 0 0 0; padding: .3em 1em .5em .4em; }
.ui-dialog .ui-dialog-buttonpane .ui-dialog-buttonset { float: right; }
.ui-dialog .ui-dialog-buttonpane button { margin: .5em .4em .5em 0; cursor:
pointer; }
.ui-dialog .ui-resizable-se { width: 14px; height: 14px; right: 3px; bottom: 3px; }
.ui-draggable .ui-dialog-titlebar { cursor: move; }
/* Component containers
----------------------------------*/
.ui-widget-content { border: 1px solid #a6c9e2; background: #fcfdfd; color:
#222222; }
.ui-widget-content a { color: #222222; }
.ui-widget-header { border: 1px solid #4297d7; background: #5c9ccc ; color:
#ffffff; font-weight: bold; }
.ui-widget-header a { color: #ffffff; }
/* Interaction states
----------------------------------*/
.ui-state-default a, .ui-state-default a:link, .ui-state-default a:visited {text-
decoration: none; }
.ui-state-hover, .ui-widget-content .ui-state-hover, .ui-widget-header .ui-state-
hover{
background: #4096ee;
background: -moz-linear-gradient(top, #4096ee 0%, #60abf8 56%, #7abcff 100%);
background: -webkit-gradient(linear, left top, left bottom, color-
stop(0%,#4096ee), color-stop(56%,#60abf8), color-stop(100%,#7abcff));
background: -webkit-linear-gradient(top, #4096ee 0%,#60abf8 56%,#7abcff
100%);
background: -o-linear-gradient(top, #4096ee 0%,#60abf8 56%,#7abcff 100%);
background: -ms-linear-gradient(top, #4096ee 0%,#60abf8 56%,#7abcff 100%);
background: linear-gradient(top, #4096ee 0%,#60abf8 56%,#7abcff 100%);
}
.ui-state-hover a, .ui-state-hover a:hover, .ui-state-hover a:link, .ui-state-hover
a:visited { color: #1d5987; text-decoration: none; }
.ui-state-active a, .ui-state-active a:link, .ui-state-active a:visited { color:
#e17009; text-decoration: none; }
/* Interaction Cues
----------------------------------*/
.ui-state-highlight, .ui-widget-content .ui-state-highlight, .ui-widget-header .ui-
state-highlight {border: 1px solid #fad42e; background: #fbec88 ; color:
#363636; }
.ui-state-highlight a, .ui-widget-content .ui-state-highlight a,.ui-widget-
header .ui-state-highlight a { color: #363636; }
.ui-state-error, .ui-widget-content .ui-state-error, .ui-widget-header .ui-state-
error {border: 1px solid #cd0a0a; background: #fef1ec ; color: #cd0a0a; }
.ui-state-error a, .ui-widget-content .ui-state-error a, .ui-widget-header .ui-
state-error a { color: #cd0a0a; }
.ui-state-error-text, .ui-widget-content .ui-state-error-text, .ui-widget-header
.ui-state-error-text { color: #cd0a0a; }
.ui-priority-primary, .ui-widget-content .ui-priority-primary, .ui-widget-header
.ui-priority-primary { font-weight: bold; }
.ui-priority-secondary, .ui-widget-content .ui-priority-secondary, .ui-widget-
header .ui-priority-secondary { opacity: .7; filter:Alpha(Opacity=70); font-weight:
normal; }
.ui-state-disabled, .ui-widget-content .ui-state-disabled, .ui-widget-header .ui-
state-disabled { opacity: .35; filter:Alpha(Opacity=35); background-image: none; }
.ui-state-disabled .ui-icon { filter:Alpha(Opacity=35); } /* For IE8 - See #6059 */
/* Icons
----------------------------------*/
/* Misc visuals
----------------------------------*/
/* Corner radius */
.ui-corner-all, .ui-corner-top, .ui-corner-left, .ui-corner-tl { -moz-border-
radius-topleft: 5px; -webkit-border-top-left-radius: 5px; -khtml-border-top-left-
radius: 5px; border-top-left-radius: 5px; }
.ui-corner-all, .ui-corner-top, .ui-corner-right, .ui-corner-tr { -moz-border-
radius-topright: 5px; -webkit-border-top-right-radius: 5px; -khtml-border-top-
right-radius: 5px; border-top-right-radius: 5px; }
.ui-corner-all, .ui-corner-bottom, .ui-corner-left, .ui-corner-bl { -moz-border-
radius-bottomleft: 5px; -webkit-border-bottom-left-radius: 5px; -khtml-border-
bottom-left-radius: 5px; border-bottom-left-radius: 5px; }
.ui-corner-all, .ui-corner-bottom, .ui-corner-right, .ui-corner-br { -moz-border-
radius-bottomright: 5px; -webkit-border-bottom-right-radius: 5px; -khtml-border-
bottom-right-radius: 5px; border-bottom-right-radius: 5px; }
/* Overlays */
.ui-widget-overlay { background: #000000 ; opacity: .8;filter:Alpha(Opacity=80); }
.ui-widget-shadow { margin: -8px 0 0 -8px; padding: 8px; background: #000000 ;
opacity: .8;filter:Alpha(Opacity=80); -moz-border-radius: 8px; -khtml-border-
radius: 8px; -webkit-border-radius: 8px; border-radius: 8px; }
.ui-button {
font-family: Calibri,"Helvetica Neue",Helvetica,Arial,sans-serif;
font-size: 1.4rem;
display: inline-block;
padding: .5em 1em;
margin: 1em .25em;
color:#fff;
text-decoration: none;
background: #7abcff;
border: solid #ddd;
cursor: pointer;
border-radius: .25em;
transition: 0.3s linear all;
}
CSS;
callExtraFeature('onExtraHeadCSS');
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
function echoHeadJavascript()
{
?>
<script type="text/javascript" language="javascript">
var akeeba = {};
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/*
* https://raw.githubusercontent.com/douglascrockford/JSON-js/master/json2.js
* 2016-05-01
* Public Domain.
* NO WARRANTY EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK.
* See http://www.JSON.org/js.html
*/
// Create a JSON object only if one does not already exist. We create the
// methods in a closure to avoid creating global variables.
(function ()
{
"use strict";
function f(n)
{
// Format integers to have at least two digits.
return n < 10
? "0" + n
: n;
}
function this_value()
{
return this.valueOf();
}
Date.prototype.toJSON = function ()
{
return isFinite(this.valueOf())
? this.getUTCFullYear() + "-" +
f(this.getUTCMonth() + 1) + "-" +
f(this.getUTCDate()) + "T" +
f(this.getUTCHours()) + ":" +
f(this.getUTCMinutes()) + ":" +
f(this.getUTCSeconds()) + "Z"
: null;
};
Boolean.prototype.toJSON = this_value;
Number.prototype.toJSON = this_value;
String.prototype.toJSON = this_value;
}
var gap;
var indent;
var meta;
var rep;
function quote(string)
{
rx_escapable.lastIndex = 0;
return rx_escapable.test(string)
? "\"" + string.replace(rx_escapable, function (a)
{
var c = meta[a];
return typeof c === "string"
? c
: "\\u" + ("0000" + a.charCodeAt(0).toString(16)).slice(-
4);
}) + "\""
: "\"" + string + "\"";
}
case "number":
return isFinite(value)
? String(value)
: "null";
case "boolean":
case "null":
return String(value);
case "object":
if (!value)
{
return "null";
}
// Make an array to hold the partial results of stringifying this object value.
gap += indent;
partial = [];
// Join all of the elements together, separated with commas, and wrap them in
// brackets.
v = partial.length === 0
? "[]"
: gap
? "[\n" + gap + partial.join(",\n" + gap)
+ "\n" + mind + "]"
: "[" + partial.join(",") + "]";
gap = mind;
return v;
}
for (k in value)
{
if (Object.prototype.hasOwnProperty.call(value,
k))
{
v = str(k, value);
if (v)
{
partial.push(quote(k) + (
gap
? ": "
: ":"
) + v);
}
}
}
}
v = partial.length === 0
? "{}"
: gap
? "{\n" + gap + partial.join(",\n" + gap) +
"\n" + mind + "}"
: "{" + partial.join(",") + "}";
gap = mind;
return v;
}
}
// If the JSON object does not yet have a stringify method, give it one.
// The stringify method takes a value and an optional replacer, and an optional
// space parameter, and returns a JSON text. The replacer can be a function
// that can replace values, or an array of strings that will select the keys.
// A default replacer method can be provided. Use of the space parameter can
// produce text that is more easily readable.
var i;
gap = "";
indent = "";
}
else if (typeof space === "string")
{
indent = space;
}
rep = replacer;
if (replacer && typeof replacer !== "function" &&
(typeof replacer !== "object" ||
typeof replacer.length !== "number"))
{
throw new Error("JSON.stringify");
}
// Make a fake root object containing our value under the key of "".
// Return the result of stringifying the value.
// If the JSON object does not yet have a parse method, give it one.
// The parse method takes a text and an optional reviver function, and returns
// a JavaScript value if the text is a valid JSON text.
var j;
var k;
var v;
var value = holder[key];
if (value && typeof value === "object")
{
for (k in value)
{
if (Object.prototype.hasOwnProperty.call(value,
k))
{
v = walk(value, k);
if (v !== undefined)
{
value[k] = v;
}
else
{
delete value[k];
}
}
}
}
return reviver.call(holder, key, value);
}
text = String(text);
rx_dangerous.lastIndex = 0;
if (rx_dangerous.test(text))
{
text = text.replace(rx_dangerous, function (a)
{
return "\\u" +
("0000" + a.charCodeAt(0).toString(16)).slice(-
4);
});
}
// In the second stage, we run the text against regular expressions that look
// for non-JSON patterns. We are especially concerned with "()" and "new"
// because they can cause invocation, and "=" because it can cause mutation.
// But just to be safe, we want to reject all unexpected forms.
// We split the second stage into 4 regexp operations in order to work around
// crippling inefficiencies in IE's and Safari's regexp engines. First we
// replace the JSON backslash pairs with "@" (a non-JSON character). Second, we
// replace all simple value tokens with "]" characters. Third, we delete all
// open brackets that follow a colon or comma or that begin the text. Finally,
// we look to see that the remaining characters are only whitespace or "]" or
// "," or ":" or "{" or "}". If that is so, then the text is safe for eval.
if (
rx_one.test(
text
.replace(rx_two, "@")
.replace(rx_three, "]")
.replace(rx_four, "")
)
)
{
// In the third stage we use the eval function to compile the text into a
// JavaScript structure. The "{" operator is subject to a syntactic ambiguity
// in JavaScript: it can begin a block or an object literal. We wrap the text
// in parens to eliminate the ambiguity.
// In the optional fourth stage, we recursively walk the new structure, passing
// each name/value pair to a reviver function for possible transformation.
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Returns the version of Internet Explorer or a -1
* (indicating the use of another browser).
*
* @return integer MSIE version or -1
*/
function getInternetExplorerVersion()
{
var rv = -1; // Return value assumes failure.
if (navigator.appName == "Microsoft Internet Explorer")
{
var ua = navigator.userAgent;
var re = new RegExp("MSIE ([0-9]{1,}[\.0-9]{0,})");
if (re.exec(ua) != null)
{
rv = parseFloat(RegExp.$1);
}
}
return rv;
}
function resolvePath(filename)
{
filename = filename.replace("\/\/g", "\/");
var parts = filename.split("/");
var out = [];
out.push(part);
}
return out.join("/");
}
/*
* Courtesy of PHPjs -- http://phpjs.org
* @license GPL, version 2
*/
function is_array(mixed_var)
{
var key = "";
var getFuncName = function (fn)
{
var name = (/\W*function\s+([\w\$]+)\s*\(/).exec(fn);
if (!name)
{
return "(Anonymous)";
}
return name[1];
};
if (!mixed_var)
{
return false;
}
// BEGIN REDUNDANT
this.php_js = this.php_js || {};
this.php_js.ini = this.php_js.ini || {};
// END REDUNDANT
(this.php_js.ini["phpjs.objectsAsArrays"].local_value.toLowerCase &&
// built-in function
getFuncName(mixed_var.constructor) !== "String"; // exclude
String()
}
if (mixed_var.hasOwnProperty)
{
for (key in mixed_var)
{
// Checks whether the object has the specified property
// if not, we figure it's not an object in the sense of a
php-associative-array.
if (false === mixed_var.hasOwnProperty(key))
{
return false;
}
}
}
return false;
}
function size_format(filesize)
{
if (filesize >= 1073741824)
{
filesize = number_format(filesize / 1073741824, 2, ".", "") + " GB";
}
else
{
if (filesize >= 1048576)
{
filesize = number_format(filesize / 1048576, 2, ".", "") + " MB";
}
else
{
filesize = number_format(filesize / 1024, 2, ".", "") + " KB";
}
}
return filesize;
}
/**
* Checks if a variable is empty. From the php.js library.
*/
function empty(mixed_var)
{
var key;
return false;
}
function array_shift(inputArr)
{
// http://kevin.vanzonneveld.net
// + original by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
// + improved by: Martijn Wieringa
// % note 1: Currently does not handle objects
// * example 1: array_shift(['Kevin', 'van', 'Zonneveld']);
// * returns 1: 'Kevin'
if (inputArr.length === 0)
{
return null;
}
if (inputArr.length > 0)
{
return inputArr.shift();
}
}
if (!charlist)
{
// default list
whitespace =
"
\n\r\t\f\x0b\xa0\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\
u200b\u2028\u2029\u3000";
}
else
{
// preg_quote custom list
charlist += "";
whitespace = charlist.replace(/([\[\]\(\)\.\?\/\*\{\}\+\$\^\:])/g,
"$1");
}
l = str.length;
for (i = 0; i < l; i++)
{
if (whitespace.indexOf(str.charAt(i)) === -1)
{
str = str.substring(i);
break;
}
}
l = str.length;
for (i = l - 1; i >= 0; i--)
{
if (whitespace.indexOf(str.charAt(i)) === -1)
{
str = str.substring(0, i + 1);
break;
}
}
function array_merge()
{
// Merges elements from passed arrays into one array
//
// version: 1103.1210
// discuss at: http://phpjs.org/functions/array_merge
// + original by: Brett Zamir (http://brett-zamir.me)
// + bugfixed by: Nate
// + input by: josh
// + bugfixed by: Brett Zamir (http://brett-zamir.me)
// * example 1: arr1 = {"color": "red", 0: 2, 1: 4}
// * example 1: arr2 = {0: "a", 1: "b", "color": "green", "shape":
"trapezoid", 2: 4}
// * example 1: array_merge(arr1, arr2)
// * returns 1: {"color": "green", 0: 2, 1: 4, 2: "a", 3: "b", "shape":
"trapezoid", 4: 4}
// * example 2: arr1 = []
// * example 2: arr2 = {1: "data"}
// * example 2: array_merge(arr1, arr2)
// * returns 2: {0: "data"}
var args = Array.prototype.slice.call(arguments),
retObj = {},
k, j = 0,
i = 0,
retArr = true;
if (retArr)
{
retArr = [];
for (i = 0; i < args.length; i++)
{
retArr = retArr.concat(args[i]);
}
return retArr;
}
var ct = 0;
function array_diff(arr1)
{ // eslint-disable-line camelcase
// discuss at: http://locutus.io/php/array_diff/
// original by: Kevin van Zonneveld (http://kvz.io)
// improved by: Sanjoy Roy
// revised by: Brett Zamir (http://brett-zamir.me)
// example 1: array_diff(['Kevin', 'van', 'Zonneveld'], ['van',
'Zonneveld'])
// returns 1: {0:'Kevin'}
return retArr;
}
//=============================================================================
// Object.keys polyfill
//=============================================================================
// From https://developer.mozilla.org/en-
US/docs/Web/JavaScript/Reference/Global_Objects/Object/keys
if (!Object.keys)
{
Object.keys = (function () {
"use strict";
var hasOwnProperty = Object.prototype.hasOwnProperty,
hasDontEnumBug = !({toString:
null}).propertyIsEnumerable("toString"),
dontEnums = [
"toString",
"toLocaleString",
"valueOf",
"hasOwnProperty",
"isPrototypeOf",
"propertyIsEnumerable",
"constructor"
],
dontEnumsLength = dontEnums.length;
if (hasDontEnumBug)
{
for (i = 0; i < dontEnumsLength; i++)
{
if (hasOwnProperty.call(obj, dontEnums[i]))
{
result.push(dontEnums[i]);
}
}
}
return result;
};
}());
}
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
akeeba.System = {};
akeeba.System.notification = {
hasDesktopNotification: false,
iconURL: ''
};
akeeba.System.params = {
AjaxURL: '',
errorCallback: onGenericError,
password: '',
errorDialogId: 'errorDialog',
errorDialogMessageId: 'errorDialogPre'
};
/**
* An extremely simple error handler, dumping error messages to screen
*
* @param error The error message string
*/
akeeba.System.defaultErrorHandler = function (error)
{
alert("An error has occurred\n" + error);
};
akeeba.System.params.errorCallback = onGenericError;
/**
* Performs an AJAX request and returns the parsed JSON output.
* akeeba.System.params.AjaxURL is used as the AJAX proxy URL.
* If there is no errorCallback, the global akeeba.System.params.errorCallback is
used.
*
* @param data An object with the query data, e.g. a serialized form
* @param successCallback A function accepting a single object parameter, called
on success
* @param errorCallback A function accepting a single string parameter, called
on failure
* @param useCaching Should we use the cache?
* @param timeout Timeout before cancelling the request (default 60s)
*/
akeeba.System.doAjax = function (data, successCallback, errorCallback, useCaching,
timeout)
{
if (useCaching == null)
{
useCaching = true;
}
if (timeout == null)
{
timeout = 600000;
}
var structure =
{
type: "POST",
url: akeeba.System.params.AjaxURL,
cache: false,
data: data,
timeout: timeout,
success: function (msg)
{
// Initialize
var message = "";
if (errorCallback == null)
{
if (akeeba.System.params.errorCallback !=
null)
{
akeeba.System.params.errorCallback(msg);
}
}
else
{
errorCallback(msg);
}
return;
}
else if (valid_pos !== 0)
{
// Data is prefixed with junk
message = msg.substr(valid_pos);
}
else
{
message = msg;
}
try
{
var data = JSON.parse(message);
}
catch (err)
{
message =
akeeba.System.sanitizeErrorMessage(message);
msg = err.message + "\n<br/>\n<pre>\n" +
message + "\n</pre>";
if (errorCallback == null)
{
if (akeeba.System.params.errorCallback !=
null)
{
akeeba.System.params.errorCallback(msg);
}
}
else
{
errorCallback(msg);
}
return;
}
if (errorCallback == null)
{
if (akeeba.System.params.errorCallback != null)
{
akeeba.System.params.errorCallback(message);
}
}
else
{
errorCallback(message);
}
}
};
if (useCaching)
{
akeeba.Ajax.enqueue(structure);
}
else
{
akeeba.Ajax.ajax(structure);
}
};
/**
* Sanitize a message before displaying it in an error dialog. Some servers return
an HTML page with DOM modifying
* JavaScript when they block the backup script for any reason (usually with a 5xx
HTTP error code). Displaying the
* raw response in the error dialog has the side-effect of killing our backup
resumption JavaScript or even completely
* destroy the page, making backup restart impossible.
*
* @param {string} msg The message to sanitize
*
* @returns {string}
*/
akeeba.System.sanitizeErrorMessage = function (msg)
{
if (msg.indexOf("<script") > -1)
{
msg = "(HTML containing script tags)";
}
return msg;
};
/**
* Get and set data to elements. Use:
* akeeba.System.data.set(element, property, value)
* akeeba.System.data.get(element, property, defaultValue)
*
* On modern browsers (minimum IE 11, Chrome 8, FF 6, Opera 11, Safari 6) this will
use the data-* attributes of the
* elements where possible. On old browsers it will use an internal cache and
manually apply data-* attributes.
*/
akeeba.System.data = (function ()
{
var lastId = 0,
store = {};
return {
set: function (element, property, value)
{
// IE 11, modern browsers
if (element.dataset)
{
element.dataset[property] = value;
if (value == null)
{
delete element.dataset[property];
}
return;
}
if (element.setAttribute)
{
element.setAttribute(dataAttributeName, value);
}
if (value == null)
{
// IE 8 throws an exception on "delete"
try
{
delete store[id][property];
element.removeAttribute(dataAttributeName);
}
catch (e)
{
store[id][property] = null;
}
}
},
return element.dataset[property];
}
// IE 8 to 10, old browsers
/**
* Adds an event listener to an element
*
* @param element
* @param eventName
* @param listener
*/
akeeba.System.addEventListener = function (element, eventName, listener)
{
// Allow the passing of an element ID string instead of the DOM elem
if (typeof element === "string")
{
element = document.getElementById(element);
}
if (element == null)
{
return;
}
// Handles the listener in a way that returning boolean false will cancel the
event propagation
function listenHandler(e)
{
var ret = listener.apply(this, arguments);
if (e.preventDefault)
{
e.preventDefault();
}
else
{
e.returnValue = false;
}
}
return (ret);
}
return (ret);
}
if (element.addEventListener)
{
element.addEventListener(eventName, listenHandler, false);
return;
}
/**
* Remove an event listener from an element
*
* @param element
* @param eventName
* @param listener
*/
akeeba.System.removeEventListener = function (element, eventName, listener)
{
// Allow the passing of an element ID string instead of the DOM elem
if (typeof element === "string")
{
element = document.getElementById(element);
}
if (element == null)
{
return;
}
if (element.removeEventListener)
{
element.removeEventListener(eventName, listener);
return;
}
return;
}
return;
}
// Call this when the document is ready. This function protects itself
against being called more than once.
function ready()
{
if (!readyFired)
{
// This must be set to true before we start calling callbacks
readyFired = true;
/**
* Solely for the benefit of Internet Explorer
*/
function readyStateChange()
{
if (document.readyState === "complete")
{
ready();
}
}
/**
* This is the one public interface:
*
* akeeba.System.documentReady(fn, context);
*
* @param callback The callback function to execute when the document is
ready.
* @param context Optional. If present, it will be passed as an argument
to the callback.
*/
//
//
//
baseObj[funcName] = function (callback, context)
{
// If ready() has already fired, then just schedule the callback to
fire asynchronously
if (readyFired)
{
setTimeout(function ()
{
callback(context);
}, 1);
return;
}
/**
* If the document is already ready, schedule the ready() function to
run immediately.
*
* Note: IE is only safe when the readyState is "complete", other
browsers are safe when the readyState is
* "interactive"
*/
if (document.readyState === "complete" || (!document.attachEvent &&
document.readyState === "interactive"))
{
setTimeout(ready, 1);
return;
}
if (document.addEventListener)
{
// Prefer using the DOMContentLoaded event
document.addEventListener("DOMContentLoaded", ready, false);
// Our backup is the window's "load" event
window.addEventListener("load", ready, false);
return;
}
element.className = '';
if (element.className.trim)
{
element.className = element.className.trim();
}
};
element.className = '';
if (element.className.trim)
{
element.className = element.className.trim();
}
};
return false;
};
return;
}
akeeba.System.addClass(element, aClass);
};
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* An AJAX abstraction layer for use with Akeeba software
*/
akeeba.Ajax = {
// Maps nonsense HTTP status codes to what should actually be returned
xhrSuccessStatus: {
// File protocol always yields status code 0, assume 200
0: 200,
// Support: IE <=9 only. Sometimes IE returns 1223 when it should be
204
1223: 204
},
// Used for chained AJAX: each request will be launched once the previous one
is done (successfully or not)
requestArray: [],
processingQueue: false
};
/**
* Performs an asynchronous AJAX request. Mostly compatible with jQuery 1.5+
calling conventions, or at least the
* subset
* of the features we used in our software.
*
* The parameters can be
* method string HTTP method (GET, POST, PUT, ...). Default: POST.
* url string URL to access over AJAX. Required.
* timeout int Request timeout in msec. Default: 600,000 (ten minutes)
* data object Data to send to the AJAX URL. Default: empty
* success function function(string responseText, string responseStatus,
XMLHttpRequest xhr)
* error function function(XMLHttpRequest xhr, string errorType,
Exception e)
* beforeSend function function(XMLHttpRequest xhr, object parameters) You
can modify xhr, not parameters. Return
* false to abort the request.
*
* @param url {string} URL to send the AJAX request to
* @param parameters {object} Configuration parameters
*/
akeeba.Ajax.ajax = function (url, parameters)
{
// Handles jQuery 1.0 calling style of .ajax(parameters), passing the URL as
a property of the parameters object
if (typeof(parameters) === "undefined")
{
parameters = url;
url = parameters.url;
}
if (!cache)
{
var now = new Date().getTime() / 1000;
var s = parseInt(now, 10);
data._cacheBustingJunk = Math.round((now - s) * 1000) / 1000;
}
if (errorCallback)
{
akeeba.Ajax.triggerCallbacks(errorCallback, xhr, "error", null);
}
};
if (timeout > 0)
{
xhr.timeout = timeout;
}
// --- Call the beforeSend event handler. If it returns false the request is
canceled.
if (typeof(parameters.beforeSend) !== "undefined")
{
if (parameters.beforeSend(xhr, parameters) === false)
{
return;
}
}
xhr.send(sendData);
};
/**
* Adds an AJAX request to the request queue and begins processing the queue if
it's not already started. The request
* queue is a FIFO buffer. Each request will be executed as soon as the one
preceeding it has completed processing
* (successfully or otherwise).
*
* It's the same syntax as .ajax() with the difference that the request is queued
instead of executed right away.
*
* @param url {string} The URL to send the request to
* @param parameters {object} Configuration parameters
*/
akeeba.Ajax.enqueue = function (url, parameters)
{
// Handles jQuery 1.0 calling style of .ajax(parameters), passing the URL as
a property of the parameters object
if (typeof(parameters) === "undefined")
{
parameters = url;
url = parameters.url;
}
parameters.url = url;
akeeba.Ajax.requestArray.push(parameters);
akeeba.Ajax.processQueue();
};
/**
* Converts a simple object containing query string parameters to a single, escaped
query string
*
* @param object {object} A plain object containing the query parameters to
pass
* @param prefix {string} Prefix for array-type parameters
*
* @returns {string}
*
* @access private
*/
akeeba.Ajax.interpolateParameters = function (object, prefix)
{
prefix = prefix || "";
var encodedString = "";
continue;
}
/**
* Goes through a list of callbacks and calls them in succession. Accepts a
variable number of arguments.
*/
akeeba.Ajax.triggerCallbacks = function ()
{
// converts arguments to real array
var args = Array.prototype.slice.call(arguments);
var callbackList = args.shift();
return null;
};
/**
* This helper function triggers the request queue processing using a short (50
msec) timer. This prevents a long
* function nesting which could cause some browser to abort processing.
*
* @access private
*/
akeeba.Ajax.processQueueHelper = function ()
{
akeeba.Ajax.processingQueue = false;
setTimeout("akeeba.Ajax.processQueue();", 50);
};
/**
* Processes the request queue
*
* @access private
*/
akeeba.Ajax.processQueue = function ()
{
// If I don't have any more requests reset and return
if (!akeeba.Ajax.requestArray.length)
{
akeeba.Ajax.processingQueue = false;
return;
}
/**
* Add our queue processing helper to the top of the success and error
callback function stacks, ensuring that we
* will process the next request in the queue as soon as the previous one
completes (successfully or not)
*/
var successCallback = (typeof(parameters.success) === "undefined") ? [] :
parameters.success;
var errorCallback = (typeof(parameters.error) === "undefined") ? [] :
parameters.error;
successCallback.unshift(akeeba.Ajax.processQueueHelper);
errorCallback.unshift(akeeba.Ajax.processQueueHelper);
parameters.success = successCallback;
parameters.error = errorCallback;
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
function translateGUI()
{
var allElements = document.querySelectorAll('*');
transKey = e.innerHTML;
if (!array_key_exists(transKey, translation))
{
continue;
}
e.innerHTML = translation[transKey];
}
}
function trans(key)
{
if (array_key_exists(key, translation))
{
return translation[key];
}
return key;
}
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
akeeba.System.documentReady(function () {
// Hide 2nd Page
document.getElementById("page2").style.display = "none";
akeeba.System.addEventListener(document.getElementById("kickstart.procengine"),
"change", onChangeProcengine);
akeeba.System.addEventListener(document.getElementById("kickstart.setup.sourcepath"
), "change", onArchiveListReload);
akeeba.System.addEventListener(document.getElementById("reloadArchives"),
"click", onArchiveListReload);
akeeba.System.addEventListener(document.getElementById("checkFTPTempDir"),
"click", oncheckFTPTempDirClick);
akeeba.System.addEventListener(document.getElementById("resetFTPTempDir"),
"click", onresetFTPTempDir);
akeeba.System.addEventListener(document.getElementById("testFTP"), "click",
onTestFTPClick);
akeeba.System.addEventListener(document.getElementById("gobutton_top"),
"click", onStartExtraction);
akeeba.System.addEventListener(document.getElementById("gobutton"), "click",
onStartExtraction);
akeeba.System.addEventListener(document.getElementById("gobutton"), "click",
onStartExtraction);
akeeba.System.addEventListener(document.getElementById("runCleanup"),
"click", onRunCleanupClick);
akeeba.System.addEventListener(document.getElementById("runInstaller"),
"click", onRunInstallerClick);
akeeba.System.addEventListener(document.getElementById("gotoStart"), "click",
onGotoStartClick);
akeeba.System.addEventListener(document.getElementById("gotoSite"), "click",
function (event)
{
window.open("index.php", "finalstepsite");
window.close();
});
akeeba.System.addEventListener(document.getElementById("gotoAdministrator"),
"click", function (event)
{
window.open("administrator/index.php", "finalstepadmin");
window.close();
});
// Show IE warning
var msieVersion = getInternetExplorerVersion();
if ((msieVersion !== -1) && (msieVersion < 10))
{
document.getElementById("ie7Warning").style.display = "block";
}
if (!akeeba_debug)
{
document.getElementById("preextraction").style.display = "block";
document.getElementById("fade").style.display = "block";
}
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Generic error handler
*
* @param {string} msg Error message to display
*/
function onGenericError(msg)
{
document.getElementById("genericerrorInner").innerHTML = msg;
document.getElementById("genericerror").style.display = "block";
document.getElementById("fade").style.display = "block";
/**
* Set the progress bar to a specific percentage
*
* @param {int} percent Percentage (or float 0.0 to 1.0) to display in the
progress bar.
*/
function setProgressBar(percent)
{
var newValue = percent;
if (percent <= 1)
{
newValue = 100 * percent;
}
/**
* Close the lightbox
*
* @param {KeyboardEvent|MouseEvent} event
*/
function closeLightbox(event)
{
var closeMe = false;
if (!closeMe)
{
return;
}
document.getElementById("preextraction").style.display = "none";
document.getElementById("genericerror").style.display = "none";
document.getElementById("fade").style.display = "none";
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Event handler for changing the Archive directory.
*/
function onArchiveListReload()
{
post = {
'task': 'listArchives',
'json': JSON.stringify({
path:
document.getElementById('kickstart\.setup\.sourcepath').value
})
};
/**
* Event handler for switching the Write To File method
*
* @param {Event} event
*/
function onChangeProcengine(event)
{
var elProcEngine = document.getElementById("kickstart.procengine");
var procEngine = elProcEngine.value;
var elFtpOptions = document.getElementById("ftp-options");
var elPassive = document.getElementById("ftp-ssl-passive");
var elTestBtn = document.getElementById("testFTP");
// Only hide the (S)FTP options when using direct file writes
elFtpOptions.style.display = (procEngine === "direct") ? "none" : "block";
// Hide the passive mode (it's an FTP-only thing) and change the button
label.
elPassive.style.display = "none";
elTestBtn.innerHTML = trans("BTN_TESTSFTPCON");
}
}
/**
* Event handler for the Check button next to the Temporary Directory
*
* @param {MouseEvent} event
*/
function oncheckFTPTempDirClick(event)
{
var data = {
'task': 'checkTempdir',
'json': JSON.stringify({
'kickstart.ftp.tempdir':
document.getElementById('kickstart\.ftp.tempdir').value
})
};
alert(trans(key));
});
}
/**
* Event handler for the Reset button next to the Temporary Directory
*
* @param {MouseEvent} event
*/
function onresetFTPTempDir(event)
{
document.getElementById('kickstart\.ftp\.tempdir').value = default_temp_dir;
}
/**
* Event handler for the Test FTP Connection button
*
* @param {MouseEvent} event
*/
function onTestFTPClick(event)
{
var type = 'ftp';
var data = {
'task': 'checkFTP',
'json': JSON.stringify({
'type': type,
'kickstart.ftp.host':
document.getElementById('kickstart.ftp.host').value,
'kickstart.ftp.port':
document.getElementById('kickstart.ftp.port').value,
'kickstart.ftp.ssl':
document.getElementById('kickstart.ftp.ssl').checked,
'kickstart.ftp.passive':
document.getElementById('kickstart.ftp.passive').checked,
'kickstart.ftp.user':
document.getElementById('kickstart.ftp.user').value,
'kickstart.ftp.pass':
document.getElementById('kickstart.ftp.pass').value,
'kickstart.ftp.dir':
document.getElementById('kickstart.ftp.dir').value,
'kickstart.ftp.tempdir':
document.getElementById('kickstart.ftp.tempdir').value
})
};
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Generic error handler
*
* @param {string} msg The error message to display
*/
function errorHandler(msg)
{
document.getElementById("errorMessage").innerHTML = msg;
document.getElementById("error").style.display = "block";
}
/**
* Initialize the archive extraction
*/
function onStartExtraction()
{
document.getElementById("page1").style.display = "none";
document.getElementById("page2").style.display = "block";
document.getElementById("currentFile").innerText = "";
akeeba_error_callback = errorHandler;
var zapBefore = 0;
var elZap = document.getElementById("kickstart\.setup\.zapbefore");
var elRestorePermissions =
document.getElementById("kickstart\.setup\.restoreperms");
var restorePermissions = false;
akeeba_next_step_post = {
"task": "startExtracting",
"json": JSON.stringify({
"kickstart.setup.sourcepath":
document.getElementById("kickstart\.setup\.sourcepath").value,
"kickstart.setup.sourcefile":
document.getElementById("kickstart\.setup\.sourcefile").value,
"kickstart.jps.password":
document.getElementById("kickstart\.jps\.password").value,
"kickstart.tuning.min_exec_time":
document.getElementById("kickstart\.tuning\.min_exec_time").value,
"kickstart.tuning.max_exec_time":
document.getElementById("kickstart\.tuning\.max_exec_time").value,
"kickstart.stealth.enable":
document.getElementById("kickstart\.stealth\.enable").checked,
"kickstart.stealth.url":
document.getElementById("kickstart\.stealth\.url").value,
"kickstart.setup.zapbefore": zapBefore,
"kickstart.tuning.run_time_bias": 75,
"kickstart.setup.restoreperms": restorePermissions,
"kickstart.setup.dryrun": 0,
"kickstart.setup.ignoreerrors":
document.getElementById("kickstart\.setup\.ignoreerrors").checked,
"kickstart.enabled": 1,
"kickstart.security.password": "",
"kickstart.setup.renameback":
document.getElementById("kickstart\.setup\.renameback").checked,
"kickstart.procengine":
document.getElementById("kickstart\.procengine").value,
"kickstart.ftp.host":
document.getElementById("kickstart\.ftp\.host").value,
"kickstart.ftp.port":
document.getElementById("kickstart\.ftp\.port").value,
"kickstart.ftp.ssl":
document.getElementById("kickstart\.ftp\.ssl").checked,
"kickstart.ftp.passive":
document.getElementById("kickstart\.ftp\.passive").checked,
"kickstart.ftp.user":
document.getElementById("kickstart\.ftp\.user").value,
"kickstart.ftp.pass":
document.getElementById("kickstart\.ftp\.pass").value,
"kickstart.ftp.dir":
document.getElementById("kickstart\.ftp\.dir").value,
"kickstart.ftp.tempdir":
document.getElementById("kickstart\.ftp\.tempdir").value,
"kickstart.setup.extract_list":
document.getElementById("kickstart\.setup\.extract_list").value
})
};
setTimeout(runNextExtractionStep, 10);
}
/**
* Runs an extraction step.
*
* We call it through setTimeout to avoid crashing the JS due to stack exhaustion
after a long list of chained function
* calls.
*/
function runNextExtractionStep()
{
akeeba.System.doAjax(akeeba_next_step_post, function (ret)
{
processRestorationStep(ret);
});
}
/**
* AJAX callback whenever a restoration step runs
*
* @param {object} data
*/
function processRestorationStep(data)
{
// Look for errors
if (!data.status)
{
errorHandler(data.message);
return;
}
elWarningRow.innerHTML = item;
elWarnings.appendChild(elWarningRow);
elWarningsBox.style.display = "block";
}
}
akeeba_restoration_stat_outbytes = 0;
akeeba_restoration_stat_inbytes = 0;
akeeba_restoration_stat_files = 0;
}
// Update GUI
akeeba_restoration_stat_inbytes += data.bytesIn;
akeeba_restoration_stat_outbytes += data.bytesOut;
akeeba_restoration_stat_files += data.files;
var percentage = 0;
if (akeeba_restoration_stat_total > 0)
{
percentage = 100 * akeeba_restoration_stat_inbytes /
akeeba_restoration_stat_total;
percentage = Math.max(0, percentage);
percentage = Math.min(percentage, 100);
}
if (data.done)
{
percentage = 100;
}
setProgressBar(percentage);
document.getElementById("currentFile").innerText = data.lastfile;
if (!empty(data.factory))
{
akeeba_factory = data.factory;
}
if (!data.done)
{
akeeba_next_step_post = {
"task": "continueExtracting",
"json": JSON.stringify({factory: akeeba_factory})
};
setTimeout(runNextExtractionStep, 10);
return;
}
document.getElementById("page2a").style.display = "none";
document.getElementById("extractionComplete").style.display = "block";
document.getElementById("runInstaller").style.display = "inline-block";
}
/*
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
function onGotoStartClick(event)
{
document.getElementById("page2").style.display = "none";
document.getElementById("error").style.display = "none";
document.getElementById("page1").style.display = "block";
}
function onRunInstallerClick(event)
{
var windowReference = window.open("installation/index.php", "installer");
if (!windowReference.opener)
{
windowReference.opener = this.window;
}
document.getElementById("runCleanup").style.display = "inline-block";
document.getElementById("runInstaller").style.display = "none";
}
function onRunCleanupClick(event)
{
post = {
"task": "isJoomla",
// Passing the factory preserves the renamed files array
"json": JSON.stringify({factory: akeeba_factory})
};
function onRealRunCleanupClick()
{
post = {
"task": "cleanUp",
// Passing the factory preserves the renamed files array
"json": JSON.stringify({factory: akeeba_factory})
};
document.getElementById("gotoPostRestorationRroubleshooting").style.display =
"block";
if (isJoomla)
{
document.getElementById("gotoAdministrator").style.display =
"inline-block";
}
});
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
function echoTranslationStrings()
{
callExtraFeature('onLoadTranslations');
$translation = AKText::getInstance();
echo $translation->asJavascript();
}
function echoPage()
{
$edition = KICKSTARTPRO ? 'Professional' : 'Core';
$bestArchivePath = AKKickstartUtils::getBestArchivePath();
$filelist = AKKickstartUtils::getArchivesAsOptions($bestArchivePath);
?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Akeeba Kickstart <?php echo $edition ?> <?php echo VERSION ?
></title>
<style type="text/css" media="all" rel="stylesheet">
<?php echoCSS();?>
</style>
<?php echoHeadJavascript(); ?>
</head>
<body>
<div id="page-container">
<h2>THINGS_HEADER</h2>
<ol>
<li>THINGS_01</li>
<li>THINGS_02</li>
<li>THINGS_03</li>
<li>THINGS_04</li>
<li>THINGS_05</li>
<li>THINGS_06</li>
<li>THINGS_07</li>
<li>THINGS_08</li>
<li>THINGS_09</li>
</ol>
<a href="javascript:void(0)"
onclick="closeLightbox();">CLOSE_LIGHTBOX</a>
</div>
<div id="header">
<div class="title">
<img
src="
AAHFAAABxQG6eNsrAAAA/0lEQVRIx+WVUQ2DMBCGv01B5wAHYw4qARxMwiQgYQ6YAyRMAjgAB+CgezmSSwO
jZfRplzSXNE2/3t1/V/hXuwMjkKcC5ICTNQImBaRVEAfUKdLkFlaWMorDo8lUHXIvsvEoyEMubFai21TaOQ
ByVeqaFWVUPewRkEz5FqjEa+Aus3KpAfqVojsvjXYtdacVSA8MAgqRaSd+AMrYzt6zgmriv3zaeNS08MhNi
D5UArcvoA64AE+1FySEYiF0I939Fj/3iFVFd6F9g1KU33yNGiX+JDYCbmIkXHm1sd6YX/pT7pKF3VYrSB87
fc8RZyfgJX5I8WEVsfn+5fs1/LV9AHnjYQzAbyUrAAAAAElFTkSuQmCC" alt="Akeeba Kickstart
logo">
Akeeba Kickstart <?php echo $edition ?> 6.0.1
</div>
</div>
<div id="page1">
<?php callExtraFeature('onPage1'); ?>
<div id="page1-content">
<div class="helpme">
<span>NEEDSOMEHELPKS</span> <a
href="https://www.akeebabackup.com/documentation/quick-start-guide/using-
kickstart.html"
target="_blank">QUICKSTART</a>
</div>
<div class="step1">
<div class="circle">1</div>
<h2>SELECT_ARCHIVE</h2>
<div class="area-container">
<?php callExtraFeature('onPage1Step1'); ?>
<div class="clr"></div>
<label
for="kickstart.setup.sourcepath">ARCHIVE_DIRECTORY</label>
<span class="field">
<input type="text" id="kickstart.setup.sourcepath"
value="<?php echo htmlentities($bestArchivePath); ?
>"/>
<span class="button" id="reloadArchives" style="margin-
top:0;margin-bottom:0">RELOAD_ARCHIVES</span>
</span>
<br/>
<label
for="kickstart.setup.sourcefile">ARCHIVE_FILE</label>
<span class="field" id="sourcefileContainer">
<?php if (!empty($filelist)): ?>
<select id="kickstart.setup.sourcefile">
<?php echo $filelist; ?>
</select>
<?php else: ?>
<a
href="https://www.akeebabackup.com/documentation/troubleshooter/ksnoarchives.html"
target="_blank">NOARCHIVESCLICKHERE</a>
<?php endif; ?>
</span>
<br/>
<label
for="kickstart.jps.password">JPS_PASSWORD</label>
<span class="field"><input type="password"
id="kickstart.jps.password" value=""/></span>
</div>
<div class="area-container">
<label for="gobutton_top"></label>
<span id="gobutton_top" class="button" style="padding:
0.5em 2em; margin: 0;">BTN_START</span>
</div>
</div>
<div class="clr"></div>
<div class="step2">
<div class="circle">2</div>
<h2>SELECT_EXTRACTION</h2>
<div class="area-container">
<label
for="kickstart.procengine">WRITE_TO_FILES</label>
<span class="field">
<select id="kickstart.procengine">
<option value="hybrid">WRITE_HYBRID</option>
<option value="direct">WRITE_DIRECTLY</option>
<option value="ftp">WRITE_FTP</option>
<option value="sftp">WRITE_SFTP</option>
</select>
</span><br/>
<label
for="kickstart.setup.ignoreerrors">IGNORE_MOST_ERRORS</label>
<span class="field"><input type="checkbox"
id="kickstart.setup.ignoreerrors"/></span>
<div id="ftp-options">
<label
for="kickstart.ftp.host">FTP_HOST</label>
<span class="field"><input type="text"
id="kickstart.ftp.host"
value="localhost"/></span><br/>
<label
for="kickstart.ftp.port">FTP_PORT</label>
<span class="field"><input type="text"
id="kickstart.ftp.port" value="21"/></span><br/>
<div id="ftp-ssl-passive">
<label
for="kickstart.ftp.ssl">FTP_FTPS</label>
<span class="field"><input
type="checkbox" id="kickstart.ftp.ssl"/></span><br/>
<label
for="kickstart.ftp.passive">FTP_PASSIVE</label>
<span class="field"><input
type="checkbox" id="kickstart.ftp.passive"
checked="checked"/></span><br/>
</div>
<label
for="kickstart.ftp.user">FTP_USER</label>
<span class="field"><input type="text"
id="kickstart.ftp.user" value=""/></span><br/>
<label
for="kickstart.ftp.pass">FTP_PASS</label>
<span class="field"><input
type="password" id="kickstart.ftp.pass" value=""/></span><br/>
<label
for="kickstart.ftp.dir">FTP_DIR</label>
<span class="field">
<input type="text" id="kickstart.ftp.dir" value=""/>
<?php //<span class="button" id="browseFTP" style="margin-
top:0;margin-bottom:0">FTP_BROWSE</span> ?>
</span><br/>
<label
for="kickstart.ftp.tempdir">FTP_TEMPDIR</label>
<span class="field">
<input type="text" id="kickstart.ftp.tempdir"
value="<?php echo
htmlentities(AKKickstartUtils::getTemporaryPath()) ?>"/>
<span class="button"
id="checkFTPTempDir">BTN_CHECK</span>
<span class="button"
id="resetFTPTempDir">BTN_RESET</span>
</span><br/>
<label></label>
<span class="button"
id="testFTP">BTN_TESTFTPCON</span>
<a id="notWorking" class="button"
href="https://www.akeebabackup.com/documentation/troubleshooter/kscantextract.html"
target="_blank">CANTGETITTOWORK</a>
<br/>
</div>
</div>
</div>
<div class="clr"></div>
<div class="step3">
<div class="circle">3</div>
<h2>FINE_TUNE</h2>
<div id="fine-tune-holder" class="area-container">
<label
for="kickstart.tuning.min_exec_time">MIN_EXEC_TIME</label>
<span class="field"><input type="text"
id="kickstart.tuning.min_exec_time" value="1"/></span>
<span>SECONDS_PER_STEP</span><br/>
<label
for="kickstart.tuning.max_exec_time">MAX_EXEC_TIME</label>
<span class="field"><input type="text"
id="kickstart.tuning.max_exec_time" value="5"/></span>
<span>SECONDS_PER_STEP</span><br/>
<div class="help">TIME_SETTINGS_HELP</div>
<label
for="kickstart.stealth.enable">STEALTH_MODE</label>
<span class="field"><input type="checkbox"
id="kickstart.stealth.enable"/></span><br/>
<label
for="kickstart.stealth.url">STEALTH_URL</label>
<span class="field"><input type="text"
id="kickstart.stealth.url" value=""/></span><br/>
<div class="help">STEALTH_MODE_HELP</div>
<label
for="kickstart.setup.renameback">RENAME_FILES</label>
<span class="field"><input type="checkbox"
id="kickstart.setup.renameback"
checked="checked"/></span><br/>
<div class="help">RENAME_FILES_HELP</div>
<label
for="kickstart.setup.restoreperms">RESTORE_PERMISSIONS</label>
<span class="field"><input type="checkbox"
id="kickstart.setup.restoreperms"/></span><br/>
<div class="help">RESTORE_PERMISSIONS_HELP</div>
<label
for="kickstart.setup.extract_list">EXTRACT_LIST</label>
<span class="field"><textarea
id="kickstart.setup.extract_list" rows="5" cols="50"></textarea></span><br/>
<div class="help">EXTRACT_LIST_HELP</div>
</div>
</div>
<div class="clr"></div>
<div class="step4">
<div class="circle">4</div>
<h2>EXTRACT_FILES</h2>
<div class="area-container">
<label for="gobutton"></label>
<span id="gobutton"
class="button">BTN_START</span>
</div>
</div>
<div class="clr"></div>
</div>
</div>
<div id="page2">
<div id="page2a">
<div class="circle">5</div>
<h2>EXTRACTING</h2>
<div class="area-container">
<div id="warn-not-close">DO_NOT_CLOSE_EXTRACT</div>
<div id="progressbar">
<div id="progressbar-inner"> </div>
</div>
<div id="currentFile"></div>
</div>
</div>
<div id="footer">
<div class="copyright">Copyright © 2008–<?php echo
date('Y'); ?> <a
href="http://www.akeebabackup.com">Nicholas K.
Dionysopoulos / Akeeba Backup</a>. All legal rights
reserved.<br/>
</div>
</body>
</html>
<?php
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Clear the code caches for the extracted files. Used when finalizing the
restoration.
*
* @return void
*/
function clearCodeCaches()
{
// Zend OPcache
if (function_exists('opcache_reset'))
{
opcache_reset();
}
/**
* Removes all files pertaining to Kickstart.
*
* Using when finalizing the archive extraction from the web
*
* @param AKAbstractPostproc $postProc The post-processing engine of Akeeba
Restore in use
*/
function removeKickstartFiles(AKAbstractPostproc $postProc)
{
// Remove self
$postProc->unlink(basename(__FILE__));
// Delete translations
removeKickstartTranslationFiles($postProc);
// Delete cacert.pem
$postProc->unlink('cacert.pem');
}
/**
* Remove feature files, e.g. kickstart.transfer.php
*
* @param AKAbstractPostproc $postProc
*
* @return void
*/
function deleteKickstartFeatureFiles(AKAbstractPostproc $postProc)
{
$dh = opendir(AKKickstartUtils::getPath());
closedir($dh);
}
/**
* Delete the temporary directory IF AND ONLY IF it's called "kicktemp"
*
* @param AKAbstractPostproc $postProc
*
* @return void
*/
function deleteKickstartTempDirectory(AKAbstractPostproc $postProc)
{
$tempDir = $postProc->getTempDir();
$tempDir = trim($tempDir);
if (empty($tempDir))
{
return;
}
$basename = basename($tempDir);
if (strtolower($basename) != 'kicktemp')
{
return;
}
recursive_remove_directory($tempDir);
}
/**
* Delete language files, e.g. el-GR.kickstart.ini
*
* @param AKAbstractPostproc $postProc
*
* @return void
*/
function removeKickstartTranslationFiles(AKAbstractPostproc $postProc)
{
$dh = opendir(AKKickstartUtils::getPath());
closedir($dh);
}
/**
* Finalization after the restoration. Removes the installation directory, the
backup archive and rolls back automatic
* file renames.
*
* @param AKAbstractUnarchiver $unarchiver The unarchiver engine used by Akeeba
Restore
* @param AKAbstractPostproc $postProc The post-processing engine used by
Akeeba Restore
*/
function finalizeAfterRestoration(AKAbstractUnarchiver $unarchiver,
AKAbstractPostproc $postProc)
{
// Remove installation
recursive_remove_directory('installation');
/**
* Rolls back automatic file renames.
*
* @param AKAbstractUnarchiver $unarchiver The unarchiver engine used by Akeeba
Restore
* @param AKAbstractPostproc $postProc The post-processing engine used by
Akeeba Restore
*/
function rollbackAutomaticRenames(AKAbstractUnarchiver $unarchiver,
AKAbstractPostproc $postProc)
{
$renameBack = AKFactory::get('kickstart.setup.renameback', true);
if ($renameBack)
{
$renames = $unarchiver->renameFiles;
if (!empty($renames))
{
foreach ($renames as $original => $renamed)
{
$postProc->rename($renamed, $original);
}
}
}
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Utility class to parse CLI parameters in a POSIX way
*/
class AKCliParams
{
/**
* POSIX-style CLI options. Access them with through the getOption method.
*
* @var array
*/
protected static $cliOptions = array();
/**
* Parses POSIX command line options and sets the self::$cliOptions
associative array. Each array item contains
* a single dimensional array of values. Arguments without a dash are
silently ignored.
*
* @return void
*/
public static function parseOptions()
{
global $argc, $argv;
if (!empty($_GET))
{
foreach ($_GET as $k => $v)
{
$query .= " $k";
if ($v != "")
{
$query .= "=$v";
}
}
}
$query = ltrim($query);
$argv = explode(' ', $query);
$argc = count($argv);
}
$currentName = "";
$options = array();
$value = $argument;
$name = $argument;
$value = null;
if (strstr($argument, '='))
{
list($name, $value) = explode('=', $argument, 2);
}
$currentName = $name;
if (!isset($options[$currentName]) ||
($options[$currentName] == null))
{
$options[$currentName] = array();
}
}
if (strstr($value, '='))
{
$parts = explode('=', $value, 2);
$key = $parts[0];
$value = $parts[1];
}
$values = $options[$currentName];
if (is_null($values))
{
$values = array();
}
if (is_null($key))
{
$values[] = $value;
}
else
{
$values[$key] = $value;
}
$options[$currentName] = $values;
}
}
self::$cliOptions = $options;
}
/**
* Returns the value of a command line option
*
* @param string $key The full name of the option, e.g. "foobar"
* @param mixed $default The default value to return
* @param bool $array Should I return an array parameter?
*
* @return mixed The value of the option
*/
public static function getOption($key, $default = null, $array = false)
{
// If the key doesn't exist set it to the default value
if (!array_key_exists($key, self::$cliOptions))
{
self::$cliOptions[$key] = is_array($default) ? $default :
array($default);
}
if ($array)
{
return self::$cliOptions[$key];
}
return self::$cliOptions[$key][0];
}
/**
* Is the specified param key used in the command line?
*
* @param string $key The full name of the option, e.g. "foobar"
*
* @return mixed The value of the option
*/
public static function hasOption($key)
{
return array_key_exists($key, self::$cliOptions);
}
}
if (self::$silent)
{
return;
}
if (!is_object($message))
{
return;
}
if (!array_key_exists('type', get_object_vars($message)))
{
return;
}
if ($message->type == 'startfile')
{
echo $message->content->file . "\n";
}
}
if (!is_object($message))
{
return;
}
if (!array_key_exists('type', get_object_vars($message)))
{
return;
}
switch ($message->type)
{
case 'setup':
echo "I will delete existing files and folders\n";
break;
case 'deleteFile':
echo "DELETE FILE : $message->file\n";
break;
case 'deleteFolder':
echo "DELETE FOLDER: $message->file\n";
break;
}
}
/**
* Routes the Kickstart CLI application
*/
function kickstart_application_cli()
{
AKCliParams::parseOptions();
$silent = AKCliParams::hasOption('silent');
$year = gmdate('Y');
if (!$silent)
{
echo <<< BANNER
Akeeba Kickstart CLI 6.0.1
Copyright (c) 2008-$year Akeeba Ltd / Nicholas K. Dionysopoulos
-------------------------------------------------------------------------------
Akeeba Kickstart is Free Software, distributed under the terms of the GNU General
Public License version 3 or, at your option, any later version.
This program comes with ABSOLUTELY NO WARRANTY as per sections 15 & 16 of the
license. See http://www.gnu.org/licenses/gpl-3.0.html for details.
-------------------------------------------------------------------------------
BANNER;
}
if (empty($paths))
{
global $argv;
HOWTOUSE;
die;
}
AKFactory::nuke();
$archiveForDisplay = $archive;
$cwd = getcwd();
if ($archivePath == realpath($cwd))
{
$archiveForDisplay = $archiveName;
}
if (!$silent)
{
echo <<< BANNER
Extracting $archiveForDisplay
to folder $targetPath
BANNER;
}
// What am I extracting?
AKFactory::set('kickstart.setup.sourcepath', $archivePath);
AKFactory::set('kickstart.setup.sourcefile', $archiveName);
// JPS password
AKFactory::set('kickstart.jps.password', AKCliParams::getOption('password'));
// Restore permissions?
AKFactory::set('kickstart.setup.restoreperms',
AKCliParams::hasOption('permissions'));
// Dry run?
AKFactory::set('kickstart.setup.dryrun', AKCliParams::hasOption('dry-run'));
// Ignore errors?
AKFactory::set('kickstart.setup.ignoreerrors',
AKCliParams::hasOption('ignore-errors'));
// Delete all files and folders before extraction?
AKFactory::set('kickstart.setup.zapbefore', AKCliParams::hasOption('delete-
before'));
// Which files should I extract?
AKFactory::set('kickstart.setup.extract_list',
AKCliParams::getOption('extract', '', true));
// Do not rename any files (this is the CLI...)
AKFactory::set('kickstart.setup.renamefiles', array());
// Optimize time limits
AKFactory::set('kickstart.tuning.max_exec_time', 20);
AKFactory::set('kickstart.tuning.run_time_bias', 75);
AKFactory::set('kickstart.tuning.min_exec_time', 0);
AKFactory::set('kickstart.procengine', 'direct');
// Make sure that the destination directory is always set (req'd by both FTP
and Direct Writes modes)
if (empty($targetPath))
{
$targetPath = AKKickstartUtils::getPath();
}
AKFactory::set('kickstart.setup.destdir', $targetPath);
$unarchiver = AKFactory::getUnarchiver();
$observer = new CLIExtractionObserver();
$unarchiver->attach($observer);
if ($silent)
{
CLIExtractionObserver::$silent = true;
}
if (!$silent)
{
echo "\n\n";
}
$retArray = array(
'done' => false,
);
while (!$retArray['done'])
{
$timer = AKFactory::getTimer();
$timer->resetTime();
/**
* First try to run the filesystem zapper (remove all existing files and
folders). If the Zapper is
* disabled or has already finished running we will get a FALSE result.
Otherwise it's a status array
* which we can pass directly back to the caller.
*/
$ret = runZapper(new CLIDeletionObserver());
// If the Zapper had a step to run we stop here and return its status array
to the caller.
if ($ret !== false)
{
continue;
}
$unarchiver->tick();
$ret = $unarchiver->getStatusArray();
if ($ret['Error'] != '')
{
$retArray['status'] = false;
$retArray['done'] = true;
$retArray['message'] = $ret['Error'];
}
elseif (!$ret['HasRun'])
{
$retArray['files'] = $observer->filesProcessed;
$retArray['bytesIn'] = $observer->compressedTotal;
$retArray['bytesOut'] = $observer->uncompressedTotal;
$retArray['status'] = true;
$retArray['done'] = true;
}
else
{
$retArray['files'] = $observer->filesProcessed;
$retArray['bytesIn'] = $observer->compressedTotal;
$retArray['bytesOut'] = $observer->uncompressedTotal;
$retArray['status'] = true;
$retArray['done'] = false;
}
if (!is_null($observer->totalSize))
{
$retArray['totalsize'] = $observer->totalSize;
$retArray['filelist'] = $observer->fileList;
}
$retArray['Warnings'] = $ret['Warnings'];
$retArray['lastfile'] = $observer->lastFile;
if (!empty($retArray['Warnings']) && !$silent)
{
echo "\n\n";
echo "\n";
}
}
if (!$silent)
{
echo "\n\n";
}
if (!$retArray['status'])
{
if (!$silent)
{
echo "An error has occurred:\n{$retArray['message']}\n\n";
}
exit(255);
}
// Finalize
$postProc = AKFactory::getPostProc();
rollbackAutomaticRenames($unarchiver, $postProc);
clearCodeCaches();
}
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/
/**
* Routes the Kickstart web application
*/
function kickstart_application_web()
{
$retArray = array(
'status' => true,
'message' => null
);
if (!empty($json))
{
$data = json_decode($json, true);
$dir = @$data['kickstart.ftp.tempdir'];
if (!empty($dir))
{
$retArray['status'] = is_writable($dir);
}
}
break;
case 'checkFTP':
$retArray['status'] = false;
if (!empty($json))
{
$data = json_decode($json, true);
if ($data['type'] == 'ftp')
{
$ftp = new AKPostprocFTP();
}
else
{
$ftp = new AKPostprocSFTP();
}
$retArray['message'] = $ftp->getError();
$retArray['status'] = empty($retArray['message']);
}
break;
case 'ftpbrowse':
if (!empty($json))
{
$data = json_decode($json, true);
$retArray =
getListing($data['directory'], $data['host'],
$data['port'], $data['username'], $data['password'], $data['passive'],
$data['ssl']);
}
break;
case 'sftpbrowse':
if (!empty($json))
{
$data = json_decode($json, true);
$retArray =
getSftpListing($data['directory'], $data['host'],
$data['port'], $data['username'], $data['password']);
}
break;
case 'startExtracting':
case 'continueExtracting':
// Look for configuration values
$retArray['status'] = false;
if (!empty($json))
{
if ($task == 'startExtracting')
{
AKFactory::nuke();
}
$oldJSON = $json;
$json = json_decode($json, true);
if (is_null($json))
{
$json = stripslashes($oldJSON);
$json = json_decode($json, true);
}
if (!empty($json))
{
foreach ($json as $key => $value)
{
if (substr($key, 0, 9) == 'kickstart')
{
AKFactory::set($key, $value);
}
}
}
if (empty($removePath))
{
AKFactory::set('kickstart.setup.destdir',
AKKickstartUtils::getPath());
}
if ($task == 'startExtracting')
{
// If the Stealth Mode is enabled, create the
.htaccess file
if (AKFactory::get('kickstart.stealth.enable',
false))
{
createStealthURL();
}
}
/**
* First try to run the filesystem zapper (remove all existing
files and folders). If the Zapper is
* disabled or has already finished running we will get a FALSE
result. Otherwise it's a status array
* which we can pass directly back to the caller.
*/
$ret = runZapper();
// If the Zapper had a step to run we stop here and return its
status array to the caller.
if ($ret !== false)
{
$retArray = array_merge($retArray, $ret);
break;
}
if ($ret['Error'] != '')
{
$retArray['status'] = false;
$retArray['done'] = true;
$retArray['message'] = $ret['Error'];
}
elseif (!$ret['HasRun'])
{
$retArray['files'] = $observer->filesProcessed;
$retArray['bytesIn'] = $observer->compressedTotal;
$retArray['bytesOut'] = $observer->uncompressedTotal;
$retArray['status'] = true;
$retArray['done'] = true;
}
else
{
$retArray['files'] = $observer->filesProcessed;
$retArray['bytesIn'] = $observer->compressedTotal;
$retArray['bytesOut'] = $observer->uncompressedTotal;
$retArray['status'] = true;
$retArray['done'] = false;
$retArray['factory'] = AKFactory::serialize();
}
if (!is_null($observer->totalSize))
{
$retArray['totalsize'] = $observer->totalSize;
$retArray['filelist'] = $observer->fileList;
}
$retArray['Warnings'] = $ret['Warnings'];
$retArray['lastfile'] = empty($observer->lastFile) ?
'Extracting, please wait...' : $observer->lastFile;
$timer = AKFactory::getTimer();
$timer->enforce_min_exec_time();
}
break;
case 'cleanUp':
if (!empty($json))
{
$json = json_decode($json, true);
if (array_key_exists('factory', $json))
{
// Get the serialized factory
$serialized = $json['factory'];
AKFactory::unserialize($serialized);
AKFactory::set('kickstart.enabled', true);
}
}
finalizeAfterRestoration($unarchiver, $postProc);
removeKickstartFiles($postProc);
clearCodeCaches();
break;
case 'display':
$ajax = false;
echoPage();
break;
case 'isJoomla':
$ajax = true;
if (!empty($json))
{
$json = json_decode($json, true);
if (array_key_exists('factory', $json))
{
// Get the serialized factory
$serialized = $json['factory'];
AKFactory::unserialize($serialized);
AKFactory::set('kickstart.enabled', true);
}
}
$path = AKFactory::get('kickstart.setup.destdir', '');
$path = rtrim($path, '/\\');
$isJoomla = @is_dir($path . '/administrator');
if ($isJoomla)
{
$isJoomla = @is_dir($path . '/libraries/joomla');
}
$retArray = $isJoomla;
break;
case 'listArchives':
$ajax = true;
$path = null;
if (!empty($json))
{
$json = json_decode($json, true);
if (array_key_exists('path', $json))
{
$path = $json['path'];
}
}
if (empty($path) || !@is_dir($path))
{
$filelist = null;
}
else
{
$filelist = AKKickstartUtils::getArchivesAsOptions($path);
}
if (empty($filelist))
{
$retArray =
'<a
href="https://www.akeebabackup.com/documentation/troubleshooter/ksnoarchives.html"
target="_blank">' .
AKText::_('NOARCHIVESCLICKHERE')
. '</a>';
}
else
{
$retArray = '<select id="kickstart.setup.sourcefile">' .
$filelist . '</select>';
}
break;
default:
$ajax = true;
if (!empty($json))
{
$params = json_decode($json, true);
}
else
{
$params = array();
}
break;
}
if ($ajax)
{
// JSON encode the message
$json = json_encode($retArray);
/**
* Akeeba Kickstart
* A JSON-powered archive extraction tool
*
* @copyright Copyright (c)2008-2019 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU GPL v2 or - at your option - any later version
* @package kickstart
*/