HEX
Server: LiteSpeed
System: Linux s3604.bom1.stableserver.net 4.18.0-513.11.1.lve.el8.x86_64 #1 SMP Thu Jan 18 16:21:02 UTC 2024 x86_64
User: dmstechonline (1480)
PHP: 7.4.33
Disabled: NONE
Upload Files
File: /home/dmstechonline/giaconieditore.com/wp-content/plugins/backup/src/JetBackup/Cron/Task/Export.php
<?php

namespace JetBackup\Cron\Task;

use JetBackup\BackupJob\BackupJob;
use JetBackup\Data\Engine;
use JetBackup\Destination\Destination;
use JetBackup\Exception\ArchiveException;
use JetBackup\Exception\DBException;
use JetBackup\Exception\DownloaderException;
use JetBackup\Exception\ExportException;
use JetBackup\Exception\GzipException;
use JetBackup\Exception\IOException;
use JetBackup\Exception\SGBExtractorException;
use JetBackup\Exception\TaskException;
use JetBackup\JetBackup;
use JetBackup\License\License;
use JetBackup\Queue\Queue;
use JetBackup\Queue\QueueItem;
use JetBackup\Queue\QueueItemExport;
use JetBackup\SGB\Extractor;
use JetBackup\Snapshot\Snapshot;
use JetBackup\Snapshot\SnapshotDownload;
use JetBackup\Snapshot\SnapshotItem;
use JetBackup\Download\Download;
use SleekDB\Exceptions\InvalidArgumentException;

if (!defined( '__JETBACKUP__')) die('Direct access is not allowed');

class Export extends Task {

	const LOG_FILENAME = 'export';

	private Snapshot $_snapshot;
	
	private QueueItemExport $_queue_item_export;

	public function __construct() {
		parent::__construct(self::LOG_FILENAME);
	}

	/**
	 * @return void
	 * @throws InvalidArgumentException
	 * @throws DBException
	 * @throws TaskException
	 * @throws \SleekDB\Exceptions\IOException
	 */
	public function execute():void {
		parent::execute();

		$this->_queue_item_export = $this->getQueueItem()->getItemData();
		$this->_snapshot = new Snapshot($this->_queue_item_export->getSnapshotId());

		if($this->_snapshot->getEngine() == Engine::ENGINE_SGB) {
			$this->getLogController()->logError("You can't export legacy backups");
			$this->getQueueItem()->updateStatus(Queue::STATUS_ABORTED);
			$this->getQueueItem()->updateProgress('Export Aborted!', QueueItem::PROGRESS_LAST_STEP);
			return;
		}

		$destination = new Destination($this->_snapshot->getDestinationId());

		if(!License::isValid() && !in_array($destination->getType(), Destination::LICENSE_EXCLUDED)) {
			$this->getLogController()->logError("You can't export backups from {$destination->getType()} destination without a license");
			$this->getQueueItem()->updateStatus(Queue::STATUS_ABORTED);
			$this->getQueueItem()->updateProgress('Export Aborted!', QueueItem::PROGRESS_LAST_STEP);
			return;
		}

		if($this->getQueueItem()->getStatus() == Queue::STATUS_PENDING) {
			$this->getLogController()->logMessage('Starting Export Task');

			$this->getQueueItem()->getProgress()->setTotalItems(count(Queue::STATUS_EXPORT_NAMES)+3);
			$this->getQueueItem()->save();

			$this->getQueueItem()->updateProgress('Starting Export Task');
		} elseif($this->getQueueItem()->getStatus() > Queue::STATUS_PENDING) {
			$this->getLogController()->logMessage('Resumed Export Task');
		}

		try {
			$this->func([$this, '_download']);
			$this->func([$this, '_extract']);
			$this->func([$this, '_build']);
			if($this->getQueueItem()->getStatus() < Queue::STATUS_DONE  && !$this->getQueueItem()->getErrors()) $this->getQueueItem()->updateStatus(Queue::STATUS_DONE);
			else $this->getQueueItem()->updateStatus(Queue::STATUS_PARTIALLY);
			$this->getLogController()->logMessage('Completed!');
		} catch(ExportException $e) {
			$this->getQueueItem()->updateStatus(Queue::STATUS_FAILED);
			$this->getLogController()->logError($e->getMessage());
			$this->getLogController()->logMessage('Failed!');
		}

		$this->getQueueItem()->updateProgress(
			$this->getQueueItem()->getStatus() == Queue::STATUS_DONE
				? 'Export Completed!'
				: ($this->getQueueItem()->getStatus() == Queue::STATUS_PARTIALLY
				? 'Completed with errors (see logs)'
				: 'Export Failed!'),
			QueueItem::PROGRESS_LAST_STEP
		);

		$this->getLogController()->logMessage('Total time: ' . $this->getExecutionTimeElapsed());
	}

	/**
	 * @return void
	 * @throws ExportException
	 * @throws InvalidArgumentException
	 * @throws \SleekDB\Exceptions\IOException
	 */
	public function _download() {

		$queue_item = $this->getQueueItem();

		$this->getLogController()->logMessage('Execution time: ' . $this->getExecutionTimeElapsed());
		$this->getLogController()->logMessage('TTL time: ' . $this->getExecutionTimeLimit());

		$queue_item->updateStatus(Queue::STATUS_EXPORT_DOWNLOAD);
		$queue_item->updateProgress('Getting files for extraction');
		$this->getLogController()->logMessage('Downloading backup files');

		try {
			$download = new SnapshotDownload($this->_snapshot, $this->getQueueItem()->getWorkspace());
			$download->setLogController($this->getLogController());
			$download->setQueueItem($this->getQueueItem());
			$download->setTask($this);
			$download->downloadAll();
		} catch (\Exception $e) {
			throw new ExportException($e->getMessage());
		}

		// done downloading, reset sub process bar
		$queue_item->getProgress()->resetSub();
		$queue_item->save();
	}

	/**
	 * @return void
	 * @throws ExportException
	 */
	public function _extract():void {

		$queue_item = $this->getQueueItem();

		$this->getLogController()->logMessage('Execution time: ' . $this->getExecutionTimeElapsed());
		$this->getLogController()->logMessage('TTL time: ' . $this->getExecutionTimeLimit());

		$queue_item->updateStatus(Queue::STATUS_EXPORT_EXTRACT);
		$queue_item->updateProgress('Extracting data');
		$this->getLogController()->logMessage('Extracting backup data to ' . $this->getQueueItem()->getWorkspace());

		$this->foreachCallable([$this, '_getItems'], [], function($i, $item_id) {
			$item = new SnapshotItem($item_id);
			try {
				$item->extract($this->getQueueItem()->getWorkspace(), $this->getLogController(), function(string $type, string $action, int $total, int  $read) {

					$progress = $this->getQueueItem()->getProgress();
					$progress->setMessage($type); // gzip / archive
					$progress->setSubMessage($action); // decompress / extract
					$progress->setTotalSubItems($total);
					$progress->setCurrentSubItem($read);
					$this->getQueueItem()->save();

					// Call checkExecutionTime, passing the desired variables
					$this->checkExecutionTime(function () use ($type, $action, $total, $read) {
						$progress = $this->getQueueItem()->getProgress();
						$progress->setMessage($type);
						$progress->setSubMessage('Waiting for next cron');
						$progress->setTotalSubItems($total);
						$progress->setCurrentSubItem($read);
						$this->getQueueItem()->save();
					});
				});
			} catch(ArchiveException|GzipException $e) {
				throw new ExportException($e->getMessage());
			}
		}, 'extract_items');
	}

	/**
	 * @return void
	 * @throws ExportException
	 */
	public function _build() {

		$queue_item = $this->getQueueItem();

		$this->getLogController()->logMessage('Execution time: ' . $this->getExecutionTimeElapsed());
		$this->getLogController()->logMessage('TTL time: ' . $this->getExecutionTimeLimit());

		$queue_item->updateStatus(Queue::STATUS_EXPORT_BUILD);
		$queue_item->updateProgress('Extracting data');
		$this->getLogController()->logMessage('Extracting backup data to ' . $this->getQueueItem()->getWorkspace());

		$type = $this->_queue_item_export->getType();
		$workspace = $this->getQueueItem()->getWorkspace();
		
		$homedir = $workspace . JetBackup::SEP . Snapshot::SKELETON_FILES_DIRNAME;
		$destination = $workspace . JetBackup::SEP . 'skeleton';
		$database_tables = [];

		try {

			$list = SnapshotItem::query()
                ->where([SnapshotItem::PARENT_ID, '=', $this->_snapshot->getId()])
                ->where([SnapshotItem::BACKUP_CONTAINS, '=', BackupJob::BACKUP_ACCOUNT_CONTAINS_DATABASE])
                ->getQuery()
                ->fetch();
		} catch(\Exception $e) {
			throw new ExportException($e->getMessage());
		}

		foreach($list as $item_details) {
			$database_tables[] = $workspace . JetBackup::SEP . Snapshot::SKELETON_DATABASE_DIRNAME . JetBackup::SEP . $item_details[SnapshotItem::NAME] . '.sql';
		}

		if(!is_dir($destination)) mkdir($destination, 0700);
		
		$export = new \JetBackup\Export\Export($this);
		
		try {
			$file = $export->build($type, $homedir, $database_tables, $destination);
		} catch(IOException $e) {
			throw new ExportException($e->getMessage());
		}
		
		$this->func([$this, '_moveDownload'], [$file]);
	}

	/**
	 * @param $file
	 *
	 * @return void
	 * @throws ExportException
	 * @throws \SleekDB\Exceptions\IOException
	 * @throws InvalidArgumentException
	 */
	public function _moveDownload($file) {
		
		try {
			$this->getLogController()->logDebug("[_moveDownload]] File: $file " );
			$download = Download::create($file);
		} catch(\Exception $e) {
			throw new ExportException($e->getMessage());
		}
		
		$this->_queue_item_export->setDownloadId($download->getId());
		$this->getQueueItem()->save();
		
		$this->getLogController()->logMessage("Download Id: " . $download->getId());
	}

	/**
	 * @return array
	 * @throws ExportException
	 */
	public function _getItems():array {

		$items = [];
		
		try {
			$list = SnapshotItem::query()
				->where([SnapshotItem::PARENT_ID, '=', $this->_snapshot->getId()])
				->getQuery()
				->fetch();
		} catch(\Exception $e) {
			throw new ExportException($e->getMessage());
		}

		foreach($list as $item_details) $items[] = $item_details[JetBackup::ID_FIELD];

		return $items;
	}
}