<?php
/**
 * Exposes the public API to handle site duplication.
 *
 * @package WP_Ultimo
 * @subpackage Helper
 * @since 2.0.0
 */

namespace WP_Ultimo\Helpers;

use WP_Ultimo\Dependencies\Psr\Log\LogLevel;

// Exit if accessed directly
defined('ABSPATH') || exit;

require_once WP_ULTIMO_PLUGIN_DIR . '/inc/duplication/duplicate.php';

if (!defined('MUCD_PRIMARY_SITE_ID')) {

	define('MUCD_PRIMARY_SITE_ID', get_current_network_id()); // phpcs:ignore

} // end if;

/**
 * Exposes the public API to handle site duplication.
 *
 * The decision to create a buffer interface (this file), as the API layer
 * for the duplication functions is simple: it allows us to swith the duplication
 * component used without breaking backwards-compatibility in the future.
 *
 * @since 2.0.0
 */
class Site_Duplicator {

	/**
	 * Static-only class.
	 */
	private function __construct() {} // end __construct;

	/**
	 * Duplicate an existing network site.
	 *
	 * @since 2.0.0
	 *
	 * @param int    $from_site_id ID of the site you wish to copy.
	 * @param string $title Title of the new site.
	 * @param array  $args List of duplication parameters, check Site_Duplicator::process_duplication for reference.
	 * @return int|\WP_Error ID of the newly created site or error.
	 */
	public static function duplicate_site($from_site_id, $title, $args = array()) {

		$args['from_site_id'] = $from_site_id;
		$args['title']        = $title;

		$duplicate_site = Site_Duplicator::process_duplication($args);

		if (is_wp_error($duplicate_site)) {

			// translators: %s id the template site id and %s is the error message returned.
			$message = sprintf(__('Attempt to duplicate site %1$d failed: %2$s', 'wp-ultimo'), $from_site_id, $duplicate_site->get_error_message());

			wu_log_add('site-duplication', $message, LogLevel::ERROR);

			return $duplicate_site;

		} // end if;

		// translators: %1$d is the ID of the site template used, and %2$d is the id of the new site.
		$message = sprintf(__('Attempt to duplicate site %1$d successful - New site id: %2$d', 'wp-ultimo'), $from_site_id, $duplicate_site);

		wu_log_add('site-duplication', $message);

		return $duplicate_site;

	} // end duplicate_site;

	/**
	 * Replace the contents of a site with the contents of another.
	 *
	 * @since 2.0.0
	 *
	 * @param int   $from_site_id Site to get the data from.
	 * @param int   $to_site_id Site to override.
	 * @param array $args List of duplication parameters, check Site_Duplicator::process_duplication for reference.
	 * @return int|false ID of the created site.
	 */
	public static function override_site($from_site_id, $to_site_id, $args = array()) {

		$to_site = wu_get_site($to_site_id);

		$to_site_membership_id = $to_site->get_membership_id();

		$to_site_membership = $to_site->get_membership();

		$to_site_customer = $to_site_membership->get_customer();

		$args = wp_parse_args($args, array(
			'email'        => $to_site_customer->get_email_address(),
			'title'        => $to_site->get_title(),
			'path'         => $to_site->get_path(),
			'from_site_id' => $from_site_id,
			'to_site_id'   => $to_site_id,
			'meta'         => $to_site->meta
		));

		$duplicate_site_id = self::process_duplication($args);

		if (is_wp_error($duplicate_site_id)) {

			// translators: %s id the template site id and %s is the error message returned.
			$message = sprintf(__('Attempt to override site %1$d with data from site %2$d failed: %3$s', 'wp-ultimo'), $from_site_id, $to_site_id, $duplicate_site->get_error_message());

			wu_log_add('site-duplication', $message, LogLevel::ERROR);

			return false;

		} // end if;

		$new_to_site = wu_get_site($duplicate_site_id);

		$new_to_site->set_membership_id($to_site_membership_id);

		$new_to_site->set_customer_id($to_site->get_customer_id());

		$new_to_site->set_template_id($from_site_id);

		$new_to_site->set_type('customer_owned');

		$new_to_site->set_title($to_site->get_title());

		$saved = $new_to_site->save();

		if ($saved) {

			// translators: %1$d is the ID of the site template used, and %2$d is the ID of the overriden site.
			$message = sprintf(__('Attempt to override site %1$d with data from site %2$d successful.', 'wp-ultimo'), $from_site_id, $duplicate_site_id);

			wu_log_add('site-duplication', $message);

			return $saved;

		} // end if;

	} // end override_site;

	/**
	 * Processes a site duplication.
	 *
	 * @since 2.0.0
	 *
	 * @param array $args List of parameters of the duplication.
	 *                    - email Email of the admin user to be created.
	 *                    - title Title of the (new or not) site.
	 *                    - path  Path of the new site.
	 *                    - from_site_id ID of the template site being used.
	 *                    - to_site_id   ID of the target site. Can be false to create new site.
	 *                    - keep_users   If we should keep users or not. Defaults to true.
	 *                    - copy_files   If we should copy the uploaded files or not. Defaults to true.
	 *                    - public       If the (new or not) site should be public. Defaults to true.
	 *                    - domain       The domain of the new site.
	 *                    - network_id   The network ID to allow for multi-network support.
	 * @return int|WP_Error The Site ID.
	 */
	protected static function process_duplication($args) {

		global $current_site, $wpdb;

		$args = wp_parse_args($args, array(
			'email'        => '',    // Required arguments.
			'title'        => '',    // Required arguments.
			'path'         => '/',   // Required arguments.
			'from_site_id' => false, // Required arguments.
			'to_site_id'   => false,
			'keep_users'   => true,
			'public'       => true,
			'domain'       => $current_site->domain,
			'copy_files'   => wu_get_setting('copy_media', true),
			'network_id'   => get_current_network_id(),
			'meta'         => array(),
			'user_id'      => 0,
		));

		// Checks
		$args = (object) $args;

		$site_domain = $args->domain . $args->path;

		$wpdb->hide_errors();

		if (!$args->from_site_id) {

			return new \WP_Error('from_site_id_required', __('You need to provide a valid site to duplicate.', 'wp-ultimo'));

		} // end if;

		$user_id = !empty($args->user_id) ? $args->user_id : self::create_admin($args->email, $site_domain);

		if (is_wp_error($user_id)) {

			return $user_id;

		} // end if;

		if (!$args->to_site_id) {

			$meta = array_merge($args->meta, array('public' => $args->public));

			$args->to_site_id = wpmu_create_blog($args->domain, $args->path, $args->title, $user_id, $meta, $args->network_id);

			$wpdb->show_errors();

		} // end if;

		if (is_wp_error($args->to_site_id)) {

			return $args->to_site_id;

		} // end if;

		if (!is_numeric($args->to_site_id)) {

			return new \WP_Error('site_creation_failed', __('An attempt to create a new site failed.', 'wp-ultimo'));

		} // end if;

		if (!is_super_admin($user_id) && !get_user_option('primary_blog', $user_id)) {

			update_user_option($user_id, 'primary_blog', $args->to_site_id, true);

		} // end if;

		\MUCD_Duplicate::bypass_server_limit();

		if ($args->copy_files) {

			$result = \MUCD_Files::copy_files($args->from_site_id, $args->to_site_id);

		} // end if;

		/**
		 * Supress email change notification on site duplication processes.
		 */
		add_filter('send_site_admin_email_change_email', '__return_false');

		$result = \MUCD_Data::copy_data($args->from_site_id, $args->to_site_id);

		if ($args->keep_users) {

			$result = \MUCD_Duplicate::copy_users($args->from_site_id, $args->to_site_id);

		} // end if;

		wp_cache_flush();

		/**
		 * Allow developers to hook after a site duplication happens.
		 *
		 * @since 1.9.4
		 * @return void
		 */
		do_action('wu_duplicate_site', array(
			'site_id' => $args->to_site_id,
		));

		return $args->to_site_id;

	} // end process_duplication;

	/**
	 * Creates an admin user if no user exists with this email.
	 *
	 * @since 2.0.0
	 * @param  string $email The email.
	 * @param  string $domain The domain.
	 * @return int Id of the user created.
	 */
	public static function create_admin($email, $domain) {

		// Create New site Admin if not exists
		$password = 'N/A';

		$user_id = email_exists($email);

		if (!$user_id) { // Create a new user with a random password

			$password = wp_generate_password(12, false);

			$user_id = wpmu_create_user($domain, $password, $email);

			if (false === $user_id) {

				return new \WP_Error('user_creation_error', __('We were not able to create a new admin user for the site being duplicated.', 'wp-ultimo'));

			} else {

				wp_new_user_notification($user_id);

			} // end if;

		} // end if;

		return $user_id;

	} // end create_admin;

} // end class Site_Duplicator;