HEX
Server: LiteSpeed
System: Linux atali.colombiahosting.com.co 5.14.0-570.12.1.el9_6.x86_64 #1 SMP PREEMPT_DYNAMIC Tue May 13 06:11:55 EDT 2025 x86_64
User: coopserp (1713)
PHP: 8.2.29
Disabled: dl,exec,passthru,proc_open,proc_close,shell_exec,memory_limit,system,popen,curl_multi_exec,show_source,symlink,link,leak,listen,diskfreespace,tmpfile,ignore_user_abord,highlight_file,source,show_source,fpaththru,virtual,posix_ctermid,posix_getcwd,posix_getegid,posix_geteuid,posix_getgid,posix_getgrgid,posix_getgrnam,posix_getgroups,posix_getlogin,posix_getpgid,posix_getpgrp,posix_getpid,posix,posix_getppid,posix_getpwnam,posix_getpwuid,posix_getrlimit,posix_getsid,posix_getuid,posix_isatty,posix_kill,posix_mkfifo,posix_setegid,posix_seteuid,posix_setgid,posix_setpgid,posix_setsid,posix_setid,posix_times,posix_ttyname,posix_uname,proc_get_status,proc_nice,proc_terminate
Upload Files
File: //proc/self/cwd/wp-content/plugins/siteground-migrator/core/Transfer_Service/Transfer_Service.php
<?php

namespace SiteGround_Migrator\Transfer_Service;

use SiteGround_Migrator\Helper\Log_Service_Trait;
use SiteGround_Migrator\Helper\Helper;
use SiteGround_Migrator\Api_Service\Api_Service;
use SiteGround_Migrator\Files_Service\Files_Service;
use SiteGround_Migrator\Database_Service\Database_Service;
use SiteGround_Migrator\Background_Process\Background_Process;
use SiteGround_Migrator\Directory_Service\Directory_Service;
use SiteGround_Migrator\Email_Service\Email_Service;

/**
 * The transfer service class.
 */
class Transfer_Service {
	use Log_Service_Trait;

	/**
	 * A Siteground_Migrator_Api_Service instance.
	 *
	 * @var Siteground_Migrator_Api_Service object
	 *
	 * @since 1.0.0
	 */
	public $api_service;

	/**
	 * A Siteground_Migrator_Files_Service instance.
	 *
	 * @var Siteground_Migrator_Files_Service object
	 *
	 * @since 1.0.0
	 */
	private $file_service;

	/**
	 * A Siteground_Migrator_Database_Service instance.
	 *
	 * @var Siteground_Migrator_Database_Service object
	 *
	 * @since 1.0.0
	 */
	private $database_service;

	/**
	 * A Siteground_Migrator_Background_Process instance.
	 *
	 * @var Siteground_Migrator_Background_Process object
	 *
	 * @since 1.0.0
	 */
	public $background_process;

	/**
	 * A Siteground_Migrator_Directory_Service instance.
	 *
	 * @var Siteground_Migrator_Directory_Service object
	 *
	 * @since 1.0.0
	 */
	public $directory_service;

	/**
	 * A Helper instance.
	 *
	 * @var Helper object
	 *
	 * @since 2.0.9
	 */
	public $helper;

	/**
	 * A Siteground_Migrator_Email_Service instance.
	 *
	 * @var Siteground_Migrator_Email_Service object
	 *
	 * @since 1.0.0
	 */
	private $email_service;

	/**
	 * {@link Siteground_Migrator_Transfer_Service} singleton instance.
	 *
	 * @since  1.0.0
	 * @access private
	 * @var \Siteground_Migrator_Transfer_Service $instance {@link Siteground_Migrator_Transfer_Service} singleton instance.
	 */
	private static $instance;


	/**
	 * The constructor.
	 *
	 * @since 1.0.0
	 */
	public function __construct() {
		// Set the api service.
		$this->api_service        = new Api_Service();
		$this->file_service       = new Files_Service();
		$this->database_service   = new Database_Service();
		$this->background_process = new Background_Process();
		$this->directory_service  = new Directory_Service();
		$this->email_service      = new Email_Service();

		$this->helper = new Helper();

		self::$instance = $this;
	}

	/**
	 * Get {@link Siteground_Migrator_Transfer_Service} singleton instance.
	 *
	 * @since 1.0.0
	 *
	 * @return Siteground_Migrator_Transfer_Service {@link Siteground_Migrator_Transfer_Service} singleton instance.
	 */
	public static function get_instance() {
		return self::$instance;
	}

	/**
	 * Hide all errors and notices on our custom dashboard.
	 *
	 * @since  1.0.6
	 */
	public function hide_errors_and_notices() {
		// Hide all error on our dashboard.
		if (
			isset( $_GET['page'] ) &&
			'siteground-migrator' === $_GET['page']
		) {
			remove_all_actions( 'network_admin_notices' );
			remove_all_actions( 'user_admin_notices' );
			remove_all_actions( 'admin_notices' );
			remove_all_actions( 'all_admin_notices' );
		}
	}

	/**
	 * Check the current hosting enviroment before starting the transfer.
	 *
	 * @since  1.0.26
	 *
	 * @return bool True if the enviromenment has issues, false otherwise.
	 */
	public function check_environment_before_transfer() {
		// Requre the file so is_plugin_active is available.
		require_once( ABSPATH . '/wp-admin/includes/plugin.php' );

		// Check if we have a woocommerce active on the website.
		if ( \is_plugin_active( 'woocommerce/woocommerce.php' ) ) {
			$plugins = get_plugins();

			// Check if the woocommerce version is one of the known that cause trouble.
			if (
				version_compare( $plugins['woocommerce/woocommerce.php']['Version'], '4.0.0', '>=' ) &&
				version_compare( $plugins['woocommerce/woocommerce.php']['Version'], '4.3.0', '<' )
			) {
				self::update_status(
					esc_html__( 'Your WooCommerce Needs Update!', 'siteground-migrator' ),
					0,
					__( 'You are using a version of WooCommerce that has a known issue preventing our plugin from migrating your content successfully. Please, update WooCommerce to version 4.2.1 or newer and start the migration anew! <br><br> For more information on how to solve this problem, please read <a href="https://www.siteground.com/kb/wordpress-migrator-woocommerce-needs-update" target="_blank">this article</a>', 'siteground-migrator' )
				);

				return true;
			}
		}

		// Bail if temp dirs didn't exist or we have permissions issues.
		if ( false === $this->directory_service->check_if_temp_dirs_extist() ) {
			// Update the status.
			self::update_status(
				esc_html__( 'Transfer cannot be initiated due to permissions error.', 'siteground-migrator' ),
				0,
				__( 'For the purposes of this transfer we need to create temporary files on your current hosting account. Please fix your files permissions at your current host and make sure your wp-content folder is writable. Files should be set to 644 and folders to 755. <br><br> For more information on how to solve this problem, please read <a href="https://www.siteground.com/kb/wordpress-migrator-permissions-error" target="_blank">this article</a>', 'siteground-migrator' )
			);

			return true;
		}

		return false;
	}

	/**
	 * Retrieve information from api to check if the domain is the same
	 * and if there is enough free space for migration.
	 *
	 * @since  1.0.0
	 *
	 * @return bool True if everything is ok
	 */
	private function before_start_transfer() {

		// Bail if transfer token in not parsed or invalid.
		if ( false === $this->api_service->parse_transfer_token() ) {
			return false;
		}

		$data = $this->api_service->get_installation_info();

		$response = $this->api_service->do_request( '/transfer/init/', $data );

		// Bail if the transfer init has failed.
		if (
			0 !== $response['status_code'] &&
			200 !== $response['status_code']
		) {

			self::update_status(
				$response['message'],
				0,
				$this->helper->before_transfer_messages( $response['message'] )
			);

			return false;
		}

		// Bail if there is not enough space on new server.
		if ( false === $this->check_size( $response['transfer_info'], $data['wp_size'] ) ) {
			return false;
		}

		// Bail if the new domain is different from current one.
		if ( false === $this->validate_domain( $response['transfer_info'] ) ) {
			return false;
		}

		return true;
	}

	/**
	 * Check if the current domain match the domain where the site will be migrated.
	 *
	 * @since  1.0.0
	 *
	 * @param  object $transfer_info Transfer info provided by the remote server.
	 *
	 * @return Bool                  True if the domain matches, false otherwise.
	 */
	private function validate_domain( $transfer_info ) {
		$src_url = $transfer_info->src_domain . $transfer_info->src_path;
		$dst_url = $transfer_info->dst_domain . $transfer_info->dst_path;

		if ( untrailingslashit( $src_url ) !== untrailingslashit( $dst_url ) ) {
			self::update_status(
				sprintf(
					esc_html__(
						'Site domain to be changed to %s',
						'siteground-migrator'
					),
					$dst_url
				),
				5,
				esc_html__( 'While generating the transfer token, you have chosen to use a different domain than the one currently used with your WordPress. To accommodate this change we will transfer a copy of your current database settings and replace the domain information in the migrated database. Your live website will not be affected.', 'siteground-migrator' )
			);

			return false;
		}

		return true;
	}

	/**
	 * Check if the remote server has enough space to host the current installation.
	 *
	 * @since  1.0.0
	 *
	 * @param  object $transfer_info Transfer info provided by the remote server.
	 * @param  int    $wp_size       Size of current WordPress installation.
	 *
	 * @return bool                  True if the space is ok, false otherwise.
	 */
	private function check_size( $transfer_info, $wp_size ) {
		if ( empty( $transfer_info->free_space ) ) {
			return false;
		}

		if ( $wp_size > $transfer_info->free_space ) {
			self::update_status(
				esc_html__( 'There is no enough free space on your new server.', 'siteground-migrator' ),
				0,
				esc_html__( 'Please either free some space at your receiving SiteGround hosting account, or upgrade it to a higher plan that will provide you enough space for the website you want to transfer.', 'siteground-migrator' )
			);

			return false;
		}

		return true;
	}

	/**
	 * Send request to SG api to start the transfer.
	 *
	 * @since  1.0.0
	 */
	public function run_background_processes() {
		// Prepare the background process actions.
		$processes = array(
			array(
				'class'    => $this->file_service,
				'method'   => 'prepare_archives_for_download',
				'attempts' => 3,
			),
			array(
				'class'    => $this->database_service,
				'method'   => 'export_database',
				'attempts' => 3,
			),
			array(
				'class'    => $this->file_service,
				'method'   => 'create_transfer_manifest',
				'attempts' => 3,
			),
			array(
				'class'    => $this,
				'method'   => 'transfer_prepared',
				'attempts' => 3,
			),
		);

		// Loop through all processes and add them to the queue.
		foreach ( $processes as $process ) {
			$this->background_process->push_to_queue( $process );
		}

		// Dispatch.
		$this->background_process->save()->dispatch();
	}

	/**
	 * Start the transfer.
	 *
	 * @since  1.0.0
	 */
	public function transfer_start() {
		// Update the status, that transfer has started.
		self::update_status( esc_html__( 'Transfer started. Creating archives of files...', 'siteground-migrator' ) );

		// Reset the current step.
		update_option( 'siteground_migrator_current_step', 0 );

		// Bail if transfer cannot be initated.
		if ( false === $this->before_start_transfer() ) {
			return;
		}

		// Start the transfer.
		$this->run_background_processes();

	}

	/**
	 * Notify the remote server that transfer is prepared.
	 *
	 * @since  1.0.0
	 */
	public function transfer_prepared() {
		// Make the request.
		$server_response = $this->api_service->do_request( '/transfer/prepared/' );

		switch ( $server_response['status_code'] ) {
			case 0:
			case 200:
				$response = array(
					'status' => 2,
					'title' => esc_html__( 'Transfer request has been sent to SiteGround server.', 'siteground-migrator' ),
				);
				break;
			case 500:
				$response = array(
					'status'      => 0,
					'title'       => esc_html__( 'Network connection problem', 'siteground-migrator' ),
					'description' => esc_html__( 'The transfer was interrupted due to connectivity issues. Please restart transfer.', 'siteground-migrator' ),
				);
				break;
			default:
				$response = array(
					'status'        => 0,
					'title'         => $server_response['message'],
					'skip_retrying' => true,
				);
		}

		return $response;
	}

	/**
	 * Cancel the transfer.
	 *
	 * @param mixed $hard_reset Whetherto delete the transfer status ot not.
	 *
	 * @since  1.0.0
	 */
	public function transfer_cancelled( $hard_reset = true ) {
		// Invalidate the token.
		$this->cancel_and_reset( $hard_reset );

		wp_send_json_success();
	}

	/**
	 * Cancel the transfer and remove the data if necesary.
	 *
	 * @since  2.0.0
	 *
	 * @param  boolean $hard_reset Do we have to reset the database options.
	 *
	 * @return bool
	 */
	public function cancel_and_reset( $hard_reset = true ) {
		// Invalidate the token.
		$this->api_service->do_request( '/transfer/cancel/' );

		if ( false !== $hard_reset ) {
			delete_option( 'siteground_migrator_transfer_status' );
			delete_option( 'siteground_migrator_transfer_token' );
		}

		// Remove temp directory.
		$this->directory_service->remove_temp_dir_content();

		// Cancel the process.
		$this->background_process->cancel_all();

		return true;
	}

	/**
	 * Update the status of transfer in database.
	 *
	 * @since  1.0.0
	 *
	 * @param  string $message     The response message.
	 * @param  string $status      Current status of migration.
	 *                             There are several types:
	 *                              - 0 - transfer has failed.
	 *                              - 1 - transfer is in progress.
	 *                              - 2 - waiting for remote server to complete the migration.
	 *                              - 3 - transfer completed.
	 *                              - 4 - completed with errors.
	 * @param  string $description The description of the status.
	 */
	public static function update_status( $message, $status = 1, $description = '' ) {
		// Build the data array.
		$data = array(
			'message'     => $message,
			'status'      => $status,
			'description' => $description,
		);

		// Write the result to the log.
		self::get_instance()->log_info( $message );

		// Update the current status of the transfer.
		update_option( 'siteground_migrator_transfer_status', $data );
	}

	/**
	 * Update the progress bar of currently running transfer.
	 *
	 * @since  1.0.0
	 *
	 * @param  int $step The step to update the progress.
	 */
	public static function update_transfer_progress( $step ) {
		// Get the current % of progress bar.
		$progress = get_option( 'siteground_migrator_progress', 0 );

		// Round the result to integer.
		$new_progress = ceil( $progress ) + ceil( $step );

		// If progress is more than 100, do not update it.
		if ( 100 < $new_progress ) {
			return;
		}
		// Update the progress.
		update_option( 'siteground_migrator_progress', $new_progress );
	}

	/**
	 * Handle transfer status updates from remote api.
	 *
	 * @since  1.0.0
	 */
	public function update_transfer_status_endpoint() {
		// Bail if the data parameter is not set.
		if ( empty( $_POST['data'] ) ) {
			$this->log_die( '`data` parameter is required.' );
		}

		// Authenitcate the request.
		$this->api_service->authenticate( stripcslashes( $_POST['data'] ) );
		// Convert the data to array.
		$data = json_decode( sanitize_text_field( wp_unslash( $_POST['data'] ) ), true );

		$step = 5;

		$current_step = get_option( 'siteground_migrator_current_step', 0 );

		// Translate the message from out api.
		// See http://keithdevon.com/using-variables-wordpress-translation-functions/.
		$data['message'] = __( $data['message'], 'siteground-migrator' );

		if ( ! empty( $data['description'] ) ) {
			$data['description'] = __( $data['description'], 'siteground-migrator' );
		}

		// Very ugly way to prevent unwanted messages to be displayed.
		if (
			empty( $data['tot_files'] ) ||
			(
				isset( $data['tot_files'] ) &&
				$data['n_file'] > $current_step
			)
		) {
			// Update the current step only if the `n_file` param exists.
			if ( isset( $data['n_file'] ) ) {
				update_option( 'siteground_migrator_current_step', $data['n_file'] );
			}

			// Update the progress bar.
			if ( isset( $data['tot_files'] ) ) {
				// Calculate the step to update the progress bar.
				$step            = 30 / ( $data['tot_files'] / 20 );
				$data['message'] = sprintf( __( 'Downloaded %1$d out of %2$d files...', 'siteground-migrator' ), $data['n_file'], $data['tot_files'] );
			}

			// Update the status of transfer.
			update_option( 'siteground_migrator_transfer_status', $data );
			// Update the progress bar as well.
			$this->update_transfer_progress( $step );
		}

		// Send notification to site admin when the transfer is completed or failed.
		$this->email_service->prepare_and_send_notification( $data );

		$data['success'] = 1;

		wp_send_json( $data, 1 );
	}
}