1562 lines
		
	
	
		
			61 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			1562 lines
		
	
	
		
			61 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
|     /**
 | |
|      * @package     Freemius
 | |
|      * @copyright   Copyright (c) 2015, Freemius, Inc.
 | |
|      * @license     https://www.gnu.org/licenses/gpl-3.0.html GNU General Public License Version 3
 | |
|      * @since       1.0.4
 | |
|      *
 | |
|      * @link        https://github.com/easydigitaldownloads/EDD-License-handler/blob/master/EDD_SL_Plugin_Updater.php
 | |
|      */
 | |
| 
 | |
|     if ( ! defined( 'ABSPATH' ) ) {
 | |
|         exit;
 | |
|     }
 | |
| 
 | |
|     class FS_Plugin_Updater {
 | |
| 
 | |
|         /**
 | |
|          * @var Freemius
 | |
|          * @since 1.0.4
 | |
|          */
 | |
|         private $_fs;
 | |
|         /**
 | |
|          * @var FS_Logger
 | |
|          * @since 1.0.4
 | |
|          */
 | |
|         private $_logger;
 | |
|         /**
 | |
|          * @var object
 | |
|          * @since 1.1.8.1
 | |
|          */
 | |
|         private $_update_details;
 | |
|         /**
 | |
|          * @var array
 | |
|          * @since 2.1.2
 | |
|          */
 | |
|         private $_translation_updates;
 | |
| 
 | |
|         private static $_upgrade_basename = null;
 | |
| 
 | |
|         #--------------------------------------------------------------------------------
 | |
|         #region Singleton
 | |
|         #--------------------------------------------------------------------------------
 | |
| 
 | |
|         /**
 | |
|          * @var FS_Plugin_Updater[]
 | |
|          * @since 2.0.0
 | |
|          */
 | |
|         private static $_INSTANCES = array();
 | |
| 
 | |
|         /**
 | |
|          * @param Freemius $freemius
 | |
|          *
 | |
|          * @return FS_Plugin_Updater
 | |
|          */
 | |
|         static function instance( Freemius $freemius ) {
 | |
|             $key = $freemius->get_id();
 | |
| 
 | |
|             if ( ! isset( self::$_INSTANCES[ $key ] ) ) {
 | |
|                 self::$_INSTANCES[ $key ] = new self( $freemius );
 | |
|             }
 | |
| 
 | |
|             return self::$_INSTANCES[ $key ];
 | |
|         }
 | |
| 
 | |
|         #endregion
 | |
| 
 | |
|         private function __construct( Freemius $freemius ) {
 | |
|             $this->_fs = $freemius;
 | |
| 
 | |
|             $this->_logger = FS_Logger::get_logger( WP_FS__SLUG . '_' . $freemius->get_slug() . '_updater', WP_FS__DEBUG_SDK, WP_FS__ECHO_DEBUG_SDK );
 | |
| 
 | |
|             $this->filters();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Initiate required filters.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.0.4
 | |
|          */
 | |
|         private function filters() {
 | |
|             // Override request for plugin information
 | |
|             add_filter( 'plugins_api', array( &$this, 'plugins_api_filter' ), 10, 3 );
 | |
| 
 | |
|             $this->add_transient_filters();
 | |
| 
 | |
|             /**
 | |
|              * If user has the premium plugin's code but do NOT have an active license,
 | |
|              * encourage him to upgrade by showing that there's a new release, but instead
 | |
|              * of showing an update link, show upgrade link to the pricing page.
 | |
|              *
 | |
|              * @since 1.1.6
 | |
|              *
 | |
|              */
 | |
|             // WP 2.9+
 | |
|             add_action( "after_plugin_row_{$this->_fs->get_plugin_basename()}", array(
 | |
|                 &$this,
 | |
|                 'catch_plugin_update_row'
 | |
|             ), 9 );
 | |
|             add_action( "after_plugin_row_{$this->_fs->get_plugin_basename()}", array(
 | |
|                 &$this,
 | |
|                 'edit_and_echo_plugin_update_row'
 | |
|             ), 11, 2 );
 | |
| 
 | |
|             if ( ! $this->_fs->has_any_active_valid_license() ) {
 | |
|                 add_action( 'admin_head', array( &$this, 'catch_plugin_information_dialog_contents' ) );
 | |
|             }
 | |
| 
 | |
|             if ( ! WP_FS__IS_PRODUCTION_MODE ) {
 | |
|                 add_filter( 'http_request_host_is_external', array(
 | |
|                     $this,
 | |
|                     'http_request_host_is_external_filter'
 | |
|                 ), 10, 3 );
 | |
|             }
 | |
| 
 | |
|             if ( $this->_fs->is_premium() ) {
 | |
|                 if ( ! $this->is_correct_folder_name() ) {
 | |
|                     add_filter( 'upgrader_post_install', array( &$this, '_maybe_update_folder_name' ), 10, 3 );
 | |
|                 }
 | |
| 
 | |
|                 add_filter( 'upgrader_pre_install', array( 'FS_Plugin_Updater', '_store_basename_for_source_adjustment' ), 1, 2 );
 | |
|                 add_filter( 'upgrader_source_selection', array( 'FS_Plugin_Updater', '_maybe_adjust_source_dir' ), 1, 3 );
 | |
| 
 | |
|                 if ( ! $this->_fs->has_any_active_valid_license() ) {
 | |
|                     add_filter( 'wp_prepare_themes_for_js', array( &$this, 'change_theme_update_info_html' ), 10, 1 );
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since 2.1.4
 | |
|          */
 | |
|         function catch_plugin_information_dialog_contents() {
 | |
|             if (
 | |
|                 'plugin-information' !== fs_request_get( 'tab', false ) ||
 | |
|                 $this->_fs->get_slug() !== fs_request_get( 'plugin', false )
 | |
|             ) {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             add_action( 'admin_footer', array( &$this, 'edit_and_echo_plugin_information_dialog_contents' ), 0, 1 );
 | |
| 
 | |
|             ob_start();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since 2.1.4
 | |
|          *
 | |
|          * @param string $hook_suffix
 | |
|          */
 | |
|         function edit_and_echo_plugin_information_dialog_contents( $hook_suffix ) {
 | |
|             if (
 | |
|                 'plugin-information' !== fs_request_get( 'tab', false ) ||
 | |
|                 $this->_fs->get_slug() !== fs_request_get( 'plugin', false )
 | |
|             ) {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             $license = $this->_fs->_get_license();
 | |
| 
 | |
|             $subscription = ( is_object( $license ) && ! $license->is_lifetime() ) ?
 | |
|                 $this->_fs->_get_subscription( $license->id ) :
 | |
|                 null;
 | |
| 
 | |
|             $contents = ob_get_clean();
 | |
| 
 | |
|             $update_button_id_attribute_pos = strpos( $contents, 'id="plugin_update_from_iframe"' );
 | |
| 
 | |
|             if ( false !== $update_button_id_attribute_pos ) {
 | |
|                 $update_button_start_pos = strrpos(
 | |
|                     substr( $contents, 0, $update_button_id_attribute_pos ),
 | |
|                     '<a'
 | |
|                 );
 | |
| 
 | |
|                 $update_button_end_pos = ( strpos( $contents, '</a>', $update_button_id_attribute_pos ) + strlen( '</a>' ) );
 | |
| 
 | |
|                 /**
 | |
|                  * The part of the contents without the update button.
 | |
|                  *
 | |
|                  * @author Leo Fajardo (@leorw)
 | |
|                  * @since 2.2.5
 | |
|                  */
 | |
|                 $modified_contents = substr( $contents, 0, $update_button_start_pos );
 | |
| 
 | |
|                 $update_button = substr( $contents, $update_button_start_pos, ( $update_button_end_pos - $update_button_start_pos ) );
 | |
| 
 | |
|                 /**
 | |
|                  * Replace the plugin information dialog's "Install Update Now" button's text and URL. If there's a license,
 | |
|                  * the text will be "Renew license" and will link to the checkout page with the license's billing cycle
 | |
|                  * and quota. If there's no license, the text will be "Buy license" and will link to the pricing page.
 | |
|                  */
 | |
|                 $update_button = preg_replace(
 | |
|                     '/(\<a.+)(id="plugin_update_from_iframe")(.+href=")([^\s]+)(".*\>)(.+)(\<\/a>)/is',
 | |
|                     is_object( $license ) ?
 | |
|                         sprintf(
 | |
|                             '$1$3%s$5%s$7',
 | |
|                             $this->_fs->checkout_url(
 | |
|                                 is_object( $subscription ) ?
 | |
|                                     ( 1 == $subscription->billing_cycle ? WP_FS__PERIOD_MONTHLY : WP_FS__PERIOD_ANNUALLY ) :
 | |
|                                     WP_FS__PERIOD_LIFETIME,
 | |
|                                 false,
 | |
|                                 array( 'licenses' => $license->quota )
 | |
|                             ),
 | |
|                             fs_text_inline( 'Renew license', 'renew-license', $this->_fs->get_slug() )
 | |
|                         ) :
 | |
|                         sprintf(
 | |
|                             '$1$3%s$5%s$7',
 | |
|                             $this->_fs->pricing_url(),
 | |
|                             fs_text_inline( 'Buy license', 'buy-license', $this->_fs->get_slug() )
 | |
|                         ),
 | |
|                     $update_button
 | |
|                 );
 | |
| 
 | |
|                 /**
 | |
|                  * Append the modified button.
 | |
|                  *
 | |
|                  * @author Leo Fajardo (@leorw)
 | |
|                  * @since 2.2.5
 | |
|                  */
 | |
|                 $modified_contents .= $update_button;
 | |
| 
 | |
|                 /**
 | |
|                  * Append the remaining part of the contents after the update button.
 | |
|                  *
 | |
|                  * @author Leo Fajardo (@leorw)
 | |
|                  * @since 2.2.5
 | |
|                  */
 | |
|                 $modified_contents .= substr( $contents, $update_button_end_pos );
 | |
| 
 | |
|                 $contents = $modified_contents;
 | |
|             }
 | |
| 
 | |
|             echo $contents;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  2.0.0
 | |
|          */
 | |
|         private function add_transient_filters() {
 | |
|             if ( $this->_fs->is_premium() && ! $this->_fs->is_tracking_allowed() ) {
 | |
|                 $this->_logger->log( 'Opted out sites cannot receive automatic software updates.' );
 | |
| 
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             add_filter( 'pre_set_site_transient_update_plugins', array(
 | |
|                 &$this,
 | |
|                 'pre_set_site_transient_update_plugins_filter'
 | |
|             ) );
 | |
| 
 | |
|             add_filter( 'pre_set_site_transient_update_themes', array(
 | |
|                 &$this,
 | |
|                 'pre_set_site_transient_update_plugins_filter'
 | |
|             ) );
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  2.0.0
 | |
|          */
 | |
|         private function remove_transient_filters() {
 | |
|             remove_filter( 'pre_set_site_transient_update_plugins', array(
 | |
|                 &$this,
 | |
|                 'pre_set_site_transient_update_plugins_filter'
 | |
|             ) );
 | |
| 
 | |
|             remove_filter( 'pre_set_site_transient_update_themes', array(
 | |
|                 &$this,
 | |
|                 'pre_set_site_transient_update_plugins_filter'
 | |
|             ) );
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Capture plugin update row by turning output buffering.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.1.6
 | |
|          */
 | |
|         function catch_plugin_update_row() {
 | |
|             ob_start();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Overrides default update message format with "renew your license" message.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.1.6
 | |
|          *
 | |
|          * @param string $file
 | |
|          * @param array  $plugin_data
 | |
|          */
 | |
|         function edit_and_echo_plugin_update_row( $file, $plugin_data ) {
 | |
|             $plugin_update_row = ob_get_clean();
 | |
| 
 | |
|             $current = get_site_transient( 'update_plugins' );
 | |
|             if ( ! isset( $current->response[ $file ] ) ) {
 | |
|                 echo $plugin_update_row;
 | |
| 
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             $r = $current->response[ $file ];
 | |
| 
 | |
|             $has_beta_update = $this->_fs->has_beta_update();
 | |
| 
 | |
|             if ( $this->_fs->has_any_active_valid_license() ) {
 | |
|                 if ( $has_beta_update ) {
 | |
|                     /**
 | |
|                      * Turn the "new version" text into "new Beta version".
 | |
|                      *
 | |
|                      * Sample input:
 | |
|                      *      There is a new version of Awesome Plugin available. <a href="...>View version x.y.z details</a> or <a href="...>update now</a>.
 | |
|                      * Output:
 | |
|                      *      There is a new Beta version of Awesome Plugin available. <a href="...>View version x.y.z details</a> or <a href="...>update now</a>.
 | |
|                      *
 | |
|                      * @author Leo Fajardo (@leorw)
 | |
|                      * @since 2.3.0
 | |
|                      */
 | |
|                     $plugin_update_row = preg_replace(
 | |
|                         '/(\<div.+>)(.+)(\<a.+href="([^\s]+)"([^\<]+)\>.+\<a.+)(\<\/div\>)/is',
 | |
|                         (
 | |
|                             '$1' .
 | |
|                             sprintf(
 | |
|                                 fs_text_inline( 'There is a %s of %s available.', 'new-version-available', $this->_fs->get_slug() ),
 | |
|                                 $has_beta_update ?
 | |
|                                     fs_text_inline( 'new Beta version', 'new-beta-version', $this->_fs->get_slug() ) :
 | |
|                                     fs_text_inline( 'new version', 'new-version', $this->_fs->get_slug() ),
 | |
|                                 $this->_fs->get_plugin_title()
 | |
|                             ) .
 | |
|                             ' ' .
 | |
|                             '$3' .
 | |
|                             '$6'
 | |
|                         ),
 | |
|                         $plugin_update_row
 | |
|                     );
 | |
|                 }
 | |
|             } else {
 | |
|                 /**
 | |
|                  * Turn the "new version" text into a link that opens the plugin information dialog when clicked and
 | |
|                  * make the "View version x details" text link to the checkout page instead of opening the plugin
 | |
|                  * information dialog when clicked.
 | |
|                  *
 | |
|                  * Sample input:
 | |
|                  *      There is a new version of Awesome Plugin available. <a href="...>View version x.y.z details</a> or <a href="...>update now</a>.
 | |
|                  * Output:
 | |
|                  *      There is a <a href="...>new version</a> of Awesome Plugin available. <a href="...>Buy a license now</a> to access version x.y.z security & feature updates, and support.
 | |
|                  *      OR
 | |
|                  *      There is a <a href="...>new Beta version</a> of Awesome Plugin available. <a href="...>Buy a license now</a> to access version x.y.z security & feature updates, and support.
 | |
|                  *
 | |
|                  * @author Leo Fajardo (@leorw)
 | |
|                  */
 | |
|                 $plugin_update_row = preg_replace(
 | |
|                     '/(\<div.+>)(.+)(\<a.+href="([^\s]+)"([^\<]+)\>.+\<a.+)(\<\/div\>)/is',
 | |
|                     (
 | |
|                         '$1' .
 | |
|                         sprintf(
 | |
|                             fs_text_inline( 'There is a %s of %s available.', 'new-version-available', $this->_fs->get_slug() ),
 | |
|                             sprintf(
 | |
|                                 '<a href="$4"%s>%s</a>',
 | |
|                                 '$5',
 | |
|                                 $has_beta_update ?
 | |
|                                     fs_text_inline( 'new Beta version', 'new-beta-version', $this->_fs->get_slug() ) :
 | |
|                                     fs_text_inline( 'new version', 'new-version', $this->_fs->get_slug() )
 | |
|                             ),
 | |
|                             $this->_fs->get_plugin_title()
 | |
|                         ) .
 | |
|                         ' ' .
 | |
|                         $this->_fs->version_upgrade_checkout_link( $r->new_version ) .
 | |
|                         '$6'
 | |
|                     ),
 | |
|                     $plugin_update_row
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             if (
 | |
|                 $this->_fs->is_plugin() &&
 | |
|                 isset( $r->upgrade_notice ) &&
 | |
|                 strlen( trim( $r->upgrade_notice ) ) > 0
 | |
|             ) {
 | |
|                 $slug = $this->_fs->get_slug();
 | |
| 
 | |
|                 $upgrade_notice_html = sprintf(
 | |
|                     '<p class="notice fs-upgrade-notice fs-slug-%1$s fs-type-%2$s" data-slug="%1$s" data-type="%2$s"><strong>%3$s</strong> %4$s</p>',
 | |
|                     $slug,
 | |
|                     $this->_fs->get_module_type(),
 | |
|                     fs_text_inline( 'Important Upgrade Notice:', 'upgrade_notice', $slug ),
 | |
|                     esc_html( $r->upgrade_notice )
 | |
|                 );
 | |
| 
 | |
|                 $plugin_update_row = str_replace( '</div>', '</div>' . $upgrade_notice_html, $plugin_update_row );
 | |
|             }
 | |
| 
 | |
|             echo $plugin_update_row;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since  2.0.2
 | |
|          *
 | |
|          * @param array $prepared_themes
 | |
|          *
 | |
|          * @return array
 | |
|          */
 | |
|         function change_theme_update_info_html( $prepared_themes ) {
 | |
|             $theme_basename = $this->_fs->get_plugin_basename();
 | |
| 
 | |
|             if ( ! isset( $prepared_themes[ $theme_basename ] ) ) {
 | |
|                 return $prepared_themes;
 | |
|             }
 | |
| 
 | |
|             $themes_update = get_site_transient( 'update_themes' );
 | |
|             if ( ! isset( $themes_update->response[ $theme_basename ] ) ||
 | |
|                 empty( $themes_update->response[ $theme_basename ]['package'] )
 | |
|             ) {
 | |
|                 return $prepared_themes;
 | |
|             }
 | |
| 
 | |
|             $prepared_themes[ $theme_basename ]['update'] = preg_replace(
 | |
|                 '/(\<p.+>)(.+)(\<a.+\<a.+)\.(.+\<\/p\>)/is',
 | |
|                 '$1 $2 ' . $this->_fs->version_upgrade_checkout_link( $themes_update->response[ $theme_basename ]['new_version'] ) .
 | |
|                 '$4',
 | |
|                 $prepared_themes[ $theme_basename ]['update']
 | |
|             );
 | |
| 
 | |
|             // Set to false to prevent the "Update now" link for the context theme from being shown on the "Themes" page.
 | |
|             $prepared_themes[ $theme_basename ]['hasPackage'] = false;
 | |
| 
 | |
|             return $prepared_themes;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Since WP version 3.6, a new security feature was added that denies access to repository with a local ip.
 | |
|          * During development mode we want to be able updating plugin versions via our localhost repository. This
 | |
|          * filter white-list all domains including "api.freemius".
 | |
|          *
 | |
|          * @link   http://www.emanueletessore.com/wordpress-download-failed-valid-url-provided/
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.0.4
 | |
|          *
 | |
|          * @param bool   $allow
 | |
|          * @param string $host
 | |
|          * @param string $url
 | |
|          *
 | |
|          * @return bool
 | |
|          */
 | |
|         function http_request_host_is_external_filter( $allow, $host, $url ) {
 | |
|             return ( false !== strpos( $host, 'freemius' ) ) ? true : $allow;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Check for Updates at the defined API endpoint and modify the update array.
 | |
|          *
 | |
|          * This function dives into the update api just when WordPress creates its update array,
 | |
|          * then adds a custom API call and injects the custom plugin data retrieved from the API.
 | |
|          * It is reassembled from parts of the native WordPress plugin update code.
 | |
|          * See wp-includes/update.php line 121 for the original wp_update_plugins() function.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.0.4
 | |
|          *
 | |
|          * @uses   FS_Api
 | |
|          *
 | |
|          * @param object $transient_data Update array build by WordPress.
 | |
|          *
 | |
|          * @return object Modified update array with custom plugin data.
 | |
|          */
 | |
|         function pre_set_site_transient_update_plugins_filter( $transient_data ) {
 | |
|             $this->_logger->entrance();
 | |
| 
 | |
|             /**
 | |
|              * "plugins" or "themes".
 | |
|              *
 | |
|              * @author Leo Fajardo (@leorw)
 | |
|              * @since  1.2.2
 | |
|              */
 | |
|             $module_type = $this->_fs->get_module_type() . 's';
 | |
| 
 | |
|             /**
 | |
|              * Ensure that we don't mix plugins update info with themes update info.
 | |
|              *
 | |
|              * @author Leo Fajardo (@leorw)
 | |
|              * @since  1.2.2
 | |
|              */
 | |
|             if ( "pre_set_site_transient_update_{$module_type}" !== current_filter() ) {
 | |
|                 return $transient_data;
 | |
|             }
 | |
| 
 | |
|             if ( empty( $transient_data ) ||
 | |
|                  defined( 'WP_FS__UNINSTALL_MODE' )
 | |
|             ) {
 | |
|                 return $transient_data;
 | |
|             }
 | |
| 
 | |
|             global $wp_current_filter;
 | |
| 
 | |
|             $current_plugin_version = $this->_fs->get_plugin_version();
 | |
| 
 | |
|             if ( ! empty( $wp_current_filter ) && 'upgrader_process_complete' === $wp_current_filter[0] ) {
 | |
|                 if (
 | |
|                     is_null( $this->_update_details ) ||
 | |
|                     ( is_object( $this->_update_details ) && $this->_update_details->new_version !== $current_plugin_version )
 | |
|                 ) {
 | |
|                     /**
 | |
|                      * After an update, clear the stored update details and reparse the plugin's main file in order to get
 | |
|                      * the updated version's information and prevent the previous update information from showing up on the
 | |
|                      * updates page.
 | |
|                      *
 | |
|                      * @author Leo Fajardo (@leorw)
 | |
|                      * @since 2.3.1
 | |
|                      */
 | |
|                     $this->_update_details  = null;
 | |
|                     $current_plugin_version = $this->_fs->get_plugin_version( true );
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if ( ! isset( $this->_update_details ) ) {
 | |
|                 // Get plugin's newest update.
 | |
|                 $new_version = $this->_fs->get_update(
 | |
|                     false,
 | |
|                     fs_request_get_bool( 'force-check' ),
 | |
|                     WP_FS__TIME_24_HOURS_IN_SEC / 24,
 | |
|                     $current_plugin_version
 | |
|                 );
 | |
| 
 | |
|                 $this->_update_details = false;
 | |
| 
 | |
|                 if ( is_object( $new_version ) && $this->is_new_version_premium( $new_version ) ) {
 | |
|                     $this->_logger->log( 'Found newer plugin version ' . $new_version->version );
 | |
| 
 | |
|                     /**
 | |
|                      * Cache plugin details locally since set_site_transient( 'update_plugins' )
 | |
|                      * called multiple times and the non wp.org plugins are filtered after the
 | |
|                      * call to .org.
 | |
|                      *
 | |
|                      * @since 1.1.8.1
 | |
|                      */
 | |
|                     $this->_update_details = $this->get_update_details( $new_version );
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Alias.
 | |
|             $basename = $this->_fs->premium_plugin_basename();
 | |
| 
 | |
|             if ( is_object( $this->_update_details ) ) {
 | |
|                 if ( isset( $transient_data->no_update ) ) {
 | |
|                     unset( $transient_data->no_update[ $basename ] );
 | |
|                 }
 | |
| 
 | |
|                 if ( ! isset( $transient_data->response ) ) {
 | |
|                     $transient_data->response = array();
 | |
|                 }
 | |
| 
 | |
|                 // Add plugin to transient data.
 | |
|                 $transient_data->response[ $basename ] = $this->_fs->is_plugin() ?
 | |
|                     $this->_update_details :
 | |
|                     (array) $this->_update_details;
 | |
|             } else {
 | |
|                 if ( isset( $transient_data->response ) ) {
 | |
|                     /**
 | |
|                      * Ensure that there's no update data for the plugin to prevent upgrading the premium version to the latest free version.
 | |
|                      *
 | |
|                      * @author Leo Fajardo (@leorw)
 | |
|                      * @since 2.3.0
 | |
|                      */
 | |
|                     unset( $transient_data->response[ $basename ] );
 | |
|                 }
 | |
| 
 | |
|                 if ( ! isset( $transient_data->no_update ) ) {
 | |
|                     $transient_data->no_update = array();
 | |
|                 }
 | |
| 
 | |
|                 /**
 | |
|                  * Add product to no_update transient data to properly integrate with WP 5.5 auto-updates UI.
 | |
|                  *
 | |
|                  * @since 2.4.1
 | |
|                  * @link https://make.wordpress.org/core/2020/07/30/recommended-usage-of-the-updates-api-to-support-the-auto-updates-ui-for-plugins-and-themes-in-wordpress-5-5/
 | |
|                  */
 | |
|                 $transient_data->no_update[ $basename ] = $this->_fs->is_plugin() ?
 | |
|                     (object) array(
 | |
|                         'id'            => $basename,
 | |
|                         'slug'          => $this->_fs->get_slug(),
 | |
|                         'plugin'        => $basename,
 | |
|                         'new_version'   => $this->_fs->get_plugin_version(),
 | |
|                         'url'           => '',
 | |
|                         'package'       => '',
 | |
|                         'icons'         => array(),
 | |
|                         'banners'       => array(),
 | |
|                         'banners_rtl'   => array(),
 | |
|                         'tested'        => '',
 | |
|                         'requires_php'  => '',
 | |
|                         'compatibility' => new stdClass(),
 | |
|                     ) :
 | |
|                     array(
 | |
|                         'theme'        => $basename,
 | |
|                         'new_version'  => $this->_fs->get_plugin_version(),
 | |
|                         'url'          => '',
 | |
|                         'package'      => '',
 | |
|                         'requires'     => '',
 | |
|                         'requires_php' => '',
 | |
|                     );
 | |
|             }
 | |
| 
 | |
|             $slug = $this->_fs->get_slug();
 | |
| 
 | |
|             if ( $this->_fs->is_org_repo_compliant() && $this->_fs->is_freemium() ) {
 | |
|                 if ( ! isset( $this->_translation_updates ) ) {
 | |
|                     $this->_translation_updates = array();
 | |
| 
 | |
|                     if ( current_user_can( 'update_languages' ) ) {
 | |
|                         $translation_updates = $this->fetch_wp_org_module_translation_updates( $module_type, $slug );
 | |
|                         if ( ! empty( $translation_updates ) ) {
 | |
|                             $this->_translation_updates = $translation_updates;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if ( ! empty( $this->_translation_updates ) ) {
 | |
|                     $all_translation_updates = ( isset( $transient_data->translations ) && is_array( $transient_data->translations ) ) ?
 | |
|                         $transient_data->translations :
 | |
|                         array();
 | |
| 
 | |
|                     $current_plugin_translation_updates_map = array();
 | |
|                     foreach ( $all_translation_updates as $key => $translation_update ) {
 | |
|                         if ( $module_type === ( $translation_update['type'] . 's' ) && $slug === $translation_update['slug'] ) {
 | |
|                             $current_plugin_translation_updates_map[ $translation_update['language'] ] = $translation_update;
 | |
|                             unset( $all_translation_updates[ $key ] );
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     foreach ( $this->_translation_updates as $translation_update ) {
 | |
|                         $lang = $translation_update['language'];
 | |
|                         if ( ! isset( $current_plugin_translation_updates_map[ $lang ] ) ||
 | |
|                             version_compare( $translation_update['version'], $current_plugin_translation_updates_map[ $lang ]['version'], '>' )
 | |
|                         ) {
 | |
|                             $current_plugin_translation_updates_map[ $lang ] = $translation_update;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     $transient_data->translations = array_merge( $all_translation_updates, array_values( $current_plugin_translation_updates_map ) );
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return $transient_data;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Get module's required data for the updates mechanism.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  2.0.0
 | |
|          *
 | |
|          * @param \FS_Plugin_Tag $new_version
 | |
|          *
 | |
|          * @return object
 | |
|          */
 | |
|         function get_update_details( FS_Plugin_Tag $new_version ) {
 | |
|             $update              = new stdClass();
 | |
|             $update->slug        = $this->_fs->get_slug();
 | |
|             $update->new_version = $new_version->version;
 | |
|             $update->url         = WP_FS__ADDRESS;
 | |
|             $update->package     = $new_version->url;
 | |
|             $update->tested      = $new_version->tested_up_to_version;
 | |
|             $update->requires    = $new_version->requires_platform_version;
 | |
| 
 | |
|             $icon = $this->_fs->get_local_icon_url();
 | |
| 
 | |
|             if ( ! empty( $icon ) ) {
 | |
|                 $update->icons = array(
 | |
| //                    '1x'      => $icon,
 | |
| //                    '2x'      => $icon,
 | |
|                     'default' => $icon,
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             if ( $this->_fs->is_premium() ) {
 | |
|                 $latest_tag = $this->_fs->_fetch_latest_version( $this->_fs->get_id(), false );
 | |
| 
 | |
|                 if (
 | |
|                     isset( $latest_tag->readme ) &&
 | |
|                     isset( $latest_tag->readme->upgrade_notice ) &&
 | |
|                     ! empty( $latest_tag->readme->upgrade_notice )
 | |
|                 ) {
 | |
|                     $update->upgrade_notice = $latest_tag->readme->upgrade_notice;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             $update->{$this->_fs->get_module_type()} = $this->_fs->get_plugin_basename();
 | |
| 
 | |
|             return $update;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since 2.3.0
 | |
|          *
 | |
|          * @param FS_Plugin_Tag $new_version
 | |
|          *
 | |
|          * @return bool
 | |
|          */
 | |
|         private function is_new_version_premium( FS_Plugin_Tag $new_version ) {
 | |
|             $query_str = parse_url( $new_version->url, PHP_URL_QUERY );
 | |
|             if ( empty( $query_str ) ) {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             parse_str( $query_str, $params );
 | |
| 
 | |
|             return ( isset( $params['is_premium'] ) && 'true' == $params['is_premium'] );
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Update the updates transient with the module's update information.
 | |
|          *
 | |
|          * This method is required for multisite environment.
 | |
|          * If a module is site activated (not network) and not on the main site,
 | |
|          * the module will NOT be executed on the network level, therefore, the
 | |
|          * custom updates logic will not be executed as well, so unless we force
 | |
|          * the injection of the update into the updates transient, premium updates
 | |
|          * will not work.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  2.0.0
 | |
|          *
 | |
|          * @param \FS_Plugin_Tag $new_version
 | |
|          */
 | |
|         function set_update_data( FS_Plugin_Tag $new_version ) {
 | |
|             $this->_logger->entrance();
 | |
| 
 | |
|             if ( ! $this->is_new_version_premium( $new_version ) ) {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             $transient_key = "update_{$this->_fs->get_module_type()}s";
 | |
| 
 | |
|             $transient_data = get_site_transient( $transient_key );
 | |
| 
 | |
|             $transient_data = is_object( $transient_data ) ?
 | |
|                 $transient_data :
 | |
|                 new stdClass();
 | |
| 
 | |
|             // Alias.
 | |
|             $basename  = $this->_fs->get_plugin_basename();
 | |
|             $is_plugin = $this->_fs->is_plugin();
 | |
| 
 | |
|             if ( ! isset( $transient_data->response ) ||
 | |
|                  ! is_array( $transient_data->response )
 | |
|             ) {
 | |
|                 $transient_data->response = array();
 | |
|             } else if ( ! empty( $transient_data->response[ $basename ] ) ) {
 | |
|                 $version = $is_plugin ?
 | |
|                     ( ! empty( $transient_data->response[ $basename ]->new_version ) ?
 | |
|                         $transient_data->response[ $basename ]->new_version :
 | |
|                         null
 | |
|                     ) : ( ! empty( $transient_data->response[ $basename ]['new_version'] ) ?
 | |
|                         $transient_data->response[ $basename ]['new_version'] :
 | |
|                         null
 | |
|                     );
 | |
| 
 | |
|                 if ( $version == $new_version->version ) {
 | |
|                     // The update data is already set.
 | |
|                     return;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Remove the added filters.
 | |
|             $this->remove_transient_filters();
 | |
| 
 | |
|             $this->_update_details = $this->get_update_details( $new_version );
 | |
| 
 | |
|             // Set update data in transient.
 | |
|             $transient_data->response[ $basename ] = $is_plugin ?
 | |
|                 $this->_update_details :
 | |
|                 (array) $this->_update_details;
 | |
| 
 | |
|             if ( ! isset( $transient_data->checked ) ||
 | |
|                  ! is_array( $transient_data->checked )
 | |
|             ) {
 | |
|                 $transient_data->checked = array();
 | |
|             }
 | |
| 
 | |
|             // Flag the module as if it was already checked.
 | |
|             $transient_data->checked[ $basename ] = $this->_fs->get_plugin_version();
 | |
|             $transient_data->last_checked         = time();
 | |
| 
 | |
|             set_site_transient( $transient_key, $transient_data );
 | |
| 
 | |
|             $this->add_transient_filters();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since 2.0.2
 | |
|          */
 | |
|         function delete_update_data() {
 | |
|             $this->_logger->entrance();
 | |
| 
 | |
|             $transient_key = "update_{$this->_fs->get_module_type()}s";
 | |
| 
 | |
|             $transient_data = get_site_transient( $transient_key );
 | |
| 
 | |
|             // Alias
 | |
|             $basename = $this->_fs->get_plugin_basename();
 | |
| 
 | |
|             if ( ! is_object( $transient_data ) ||
 | |
|                 ! isset( $transient_data->response ) ||
 | |
|                  ! is_array( $transient_data->response ) ||
 | |
|                 empty( $transient_data->response[ $basename ] )
 | |
|             ) {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             unset( $transient_data->response[ $basename ] );
 | |
| 
 | |
|             // Remove the added filters.
 | |
|             $this->remove_transient_filters();
 | |
| 
 | |
|             set_site_transient( $transient_key, $transient_data );
 | |
| 
 | |
|             $this->add_transient_filters();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Try to fetch plugin's info from .org repository.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.0.5
 | |
|          *
 | |
|          * @param string $action
 | |
|          * @param object $args
 | |
|          *
 | |
|          * @return bool|mixed
 | |
|          */
 | |
|         static function _fetch_plugin_info_from_repository( $action, $args ) {
 | |
|             $url = $http_url = 'http://api.wordpress.org/plugins/info/1.0/';
 | |
|             if ( $ssl = wp_http_supports( array( 'ssl' ) ) ) {
 | |
|                 $url = set_url_scheme( $url, 'https' );
 | |
|             }
 | |
| 
 | |
|             $args = array(
 | |
|                 'timeout' => 15,
 | |
|                 'body'    => array(
 | |
|                     'action'  => $action,
 | |
|                     'request' => serialize( $args )
 | |
|                 )
 | |
|             );
 | |
| 
 | |
|             $request = wp_remote_post( $url, $args );
 | |
| 
 | |
|             if ( is_wp_error( $request ) ) {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             $res = maybe_unserialize( wp_remote_retrieve_body( $request ) );
 | |
| 
 | |
|             if ( ! is_object( $res ) && ! is_array( $res ) ) {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             return $res;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Fetches module translation updates from wordpress.org.
 | |
|          *
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since  2.1.2
 | |
|          *
 | |
|          * @param string $module_type
 | |
|          * @param string $slug
 | |
|          *
 | |
|          * @return array|null
 | |
|          */
 | |
|         private function fetch_wp_org_module_translation_updates( $module_type, $slug ) {
 | |
|             $plugin_data = $this->_fs->get_plugin_data();
 | |
| 
 | |
|             $locales = array_values( get_available_languages() );
 | |
|             $locales = apply_filters( "{$module_type}_update_check_locales", $locales );
 | |
|             $locales = array_unique( $locales );
 | |
| 
 | |
|             $plugin_basename = $this->_fs->get_plugin_basename();
 | |
|             if ( 'themes' === $module_type ) {
 | |
|                 $plugin_basename = $slug;
 | |
|             }
 | |
| 
 | |
|             global $wp_version;
 | |
| 
 | |
|             $request_args = array(
 | |
|                 'timeout' => 15,
 | |
|                 'body'    => array(
 | |
|                     "{$module_type}" => json_encode(
 | |
|                         array(
 | |
|                             "{$module_type}" => array(
 | |
|                                 $plugin_basename => array(
 | |
|                                     'Name'   => trim( str_replace( $this->_fs->get_plugin()->premium_suffix, '', $plugin_data['Name'] ) ),
 | |
|                                     'Author' => $plugin_data['Author'],
 | |
|                                 )
 | |
|                             )
 | |
|                         )
 | |
|                     ),
 | |
|                     'translations'    => json_encode( $this->get_installed_translations( $module_type, $slug ) ),
 | |
|                     'locale'          => json_encode( $locales )
 | |
|                 ),
 | |
|                 'user-agent' => ( 'WordPress/' . $wp_version . '; ' . home_url( '/' ) )
 | |
|             );
 | |
| 
 | |
|             $url = "http://api.wordpress.org/{$module_type}/update-check/1.1/";
 | |
|             if ( $ssl = wp_http_supports( array( 'ssl' ) ) ) {
 | |
|                 $url = set_url_scheme( $url, 'https' );
 | |
|             }
 | |
| 
 | |
|             $raw_response = Freemius::safe_remote_post(
 | |
|                 $url,
 | |
|                 $request_args,
 | |
|                 WP_FS__TIME_24_HOURS_IN_SEC,
 | |
|                 WP_FS__TIME_12_HOURS_IN_SEC,
 | |
|                 false
 | |
|             );
 | |
| 
 | |
|             if ( is_wp_error( $raw_response ) ) {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             $response = json_decode( wp_remote_retrieve_body( $raw_response ), true );
 | |
| 
 | |
|             if ( ! is_array( $response ) ) {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             if ( ! isset( $response['translations'] ) || empty( $response['translations'] ) ) {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             return $response['translations'];
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since 2.1.2
 | |
|          *
 | |
|          * @param string $module_type
 | |
|          * @param string $slug
 | |
|          *
 | |
|          * @return array
 | |
|          */
 | |
|         private function get_installed_translations( $module_type, $slug ) {
 | |
|             if ( function_exists( 'wp_get_installed_translations' ) ) {
 | |
|                 return wp_get_installed_translations( $module_type );
 | |
|             }
 | |
| 
 | |
|             $dir = "/{$module_type}";
 | |
| 
 | |
|             if ( ! is_dir( WP_LANG_DIR . $dir ) )
 | |
|                 return array();
 | |
| 
 | |
|             $files = scandir( WP_LANG_DIR . $dir );
 | |
|             if ( ! $files )
 | |
|                 return array();
 | |
| 
 | |
|             $language_data = array();
 | |
| 
 | |
|             foreach ( $files as $file ) {
 | |
|                 if ( 0 !== strpos( $file, $slug ) ) {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 if ( '.' === $file[0] || is_dir( WP_LANG_DIR . "{$dir}/{$file}" ) ) {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 if ( substr( $file, -3 ) !== '.po' ) {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 if ( ! preg_match( '/(?:(.+)-)?([a-z]{2,3}(?:_[A-Z]{2})?(?:_[a-z0-9]+)?).po/', $file, $match ) ) {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 if ( ! in_array( substr( $file, 0, -3 ) . '.mo', $files ) )  {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 list( , $textdomain, $language ) = $match;
 | |
| 
 | |
|                 if ( '' === $textdomain ) {
 | |
|                     $textdomain = 'default';
 | |
|                 }
 | |
| 
 | |
|                 $language_data[ $textdomain ][ $language ] = wp_get_pomo_file_data( WP_LANG_DIR . "{$dir}/{$file}" );
 | |
|             }
 | |
| 
 | |
|             return $language_data;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Updates information on the "View version x.x details" page with custom data.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.0.4
 | |
|          *
 | |
|          * @uses   FS_Api
 | |
|          *
 | |
|          * @param object $data
 | |
|          * @param string $action
 | |
|          * @param mixed  $args
 | |
|          *
 | |
|          * @return object
 | |
|          */
 | |
|         function plugins_api_filter( $data, $action = '', $args = null ) {
 | |
|             $this->_logger->entrance();
 | |
| 
 | |
|             if ( ( 'plugin_information' !== $action ) ||
 | |
|                  ! isset( $args->slug )
 | |
|             ) {
 | |
|                 return $data;
 | |
|             }
 | |
| 
 | |
|             $addon         = false;
 | |
|             $is_addon      = false;
 | |
|             $addon_version = false;
 | |
| 
 | |
|             if ( $this->_fs->get_slug() !== $args->slug ) {
 | |
|                 $addon = $this->_fs->get_addon_by_slug( $args->slug );
 | |
| 
 | |
|                 if ( ! is_object( $addon ) ) {
 | |
|                     return $data;
 | |
|                 }
 | |
| 
 | |
|                 if ( $this->_fs->is_addon_activated( $addon->id ) ) {
 | |
|                     $addon_version = $this->_fs->get_addon_instance( $addon->id )->get_plugin_version();
 | |
|                 } else if ( $this->_fs->is_addon_installed( $addon->id ) ) {
 | |
|                     $addon_plugin_data = get_plugin_data(
 | |
|                         ( WP_PLUGIN_DIR . '/' . $this->_fs->get_addon_basename( $addon->id ) ),
 | |
|                         false,
 | |
|                         false
 | |
|                     );
 | |
| 
 | |
|                     if ( ! empty( $addon_plugin_data ) ) {
 | |
|                         $addon_version = $addon_plugin_data['Version'];
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 $is_addon = true;
 | |
|             }
 | |
| 
 | |
|             $plugin_in_repo = false;
 | |
|             if ( ! $is_addon ) {
 | |
|                 // Try to fetch info from .org repository.
 | |
|                 $data = self::_fetch_plugin_info_from_repository( $action, $args );
 | |
| 
 | |
|                 $plugin_in_repo = ( false !== $data );
 | |
|             }
 | |
| 
 | |
|             if ( ! $plugin_in_repo ) {
 | |
|                 $data = $args;
 | |
| 
 | |
|                 // Fetch as much as possible info from local files.
 | |
|                 $plugin_local_data = $this->_fs->get_plugin_data();
 | |
|                 $data->name        = $plugin_local_data['Name'];
 | |
|                 $data->author      = $plugin_local_data['Author'];
 | |
|                 $data->sections    = array(
 | |
|                     'description' => 'Upgrade ' . $plugin_local_data['Name'] . ' to latest.',
 | |
|                 );
 | |
| 
 | |
|                 // @todo Store extra plugin info on Freemius or parse readme.txt markup.
 | |
|                 /*$info = $this->_fs->get_api_site_scope()->call('/information.json');
 | |
| 
 | |
| if ( !isset($info->error) ) {
 | |
|     $data = $info;
 | |
| }*/
 | |
|             }
 | |
| 
 | |
|             $plugin_version = $is_addon ?
 | |
|                 $addon_version :
 | |
|                 $this->_fs->get_plugin_version();
 | |
| 
 | |
|             // Get plugin's newest update.
 | |
|             $new_version = $this->get_latest_download_details( $is_addon ? $addon->id : false, $plugin_version );
 | |
| 
 | |
|             if ( ! is_object( $new_version ) || empty( $new_version->version ) ) {
 | |
|                 $data->version = $plugin_version;
 | |
|             } else {
 | |
|                 if ( $is_addon ) {
 | |
|                     $data->name    = $addon->title . ' ' . $this->_fs->get_text_inline( 'Add-On', 'addon' );
 | |
|                     $data->slug    = $addon->slug;
 | |
|                     $data->url     = WP_FS__ADDRESS;
 | |
|                     $data->package = $new_version->url;
 | |
|                 }
 | |
| 
 | |
|                 if ( ! $plugin_in_repo ) {
 | |
|                     $data->last_updated = ! is_null( $new_version->updated ) ? $new_version->updated : $new_version->created;
 | |
|                     $data->requires     = $new_version->requires_platform_version;
 | |
|                     $data->tested       = $new_version->tested_up_to_version;
 | |
|                 }
 | |
| 
 | |
|                 $data->version       = $new_version->version;
 | |
|                 $data->download_link = $new_version->url;
 | |
| 
 | |
|                 if ( isset( $new_version->readme ) && is_object( $new_version->readme ) ) {
 | |
|                     $new_version_readme_data = $new_version->readme;
 | |
|                     if ( isset( $new_version_readme_data->sections ) ) {
 | |
|                         $new_version_readme_data->sections = (array) $new_version_readme_data->sections;
 | |
|                     } else {
 | |
|                         $new_version_readme_data->sections = array();
 | |
|                     }
 | |
| 
 | |
|                     if ( isset( $data->sections ) ) {
 | |
|                         if ( isset( $data->sections['screenshots'] ) ) {
 | |
|                             $new_version_readme_data->sections['screenshots'] = $data->sections['screenshots'];
 | |
|                         }
 | |
| 
 | |
|                         if ( isset( $data->sections['reviews'] ) ) {
 | |
|                             $new_version_readme_data->sections['reviews'] = $data->sections['reviews'];
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if ( isset( $new_version_readme_data->banners ) ) {
 | |
|                         $new_version_readme_data->banners = (array) $new_version_readme_data->banners;
 | |
|                     } else if ( isset( $data->banners ) ) {
 | |
|                         $new_version_readme_data->banners = $data->banners;
 | |
|                     }
 | |
| 
 | |
|                     $wp_org_sections = array(
 | |
|                         'author',
 | |
|                         'author_profile',
 | |
|                         'rating',
 | |
|                         'ratings',
 | |
|                         'num_ratings',
 | |
|                         'support_threads',
 | |
|                         'support_threads_resolved',
 | |
|                         'active_installs',
 | |
|                         'added',
 | |
|                         'homepage'
 | |
|                     );
 | |
| 
 | |
|                     foreach ( $wp_org_sections as $wp_org_section ) {
 | |
|                         if ( isset( $data->{$wp_org_section} ) ) {
 | |
|                             $new_version_readme_data->{$wp_org_section} = $data->{$wp_org_section};
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     $data = $new_version_readme_data;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return $data;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.2.1.7
 | |
|          *
 | |
|          * @param number|bool $addon_id
 | |
|          * @param bool|string $newer_than   Since 2.2.1
 | |
|          * @param bool|string $fetch_readme Since 2.2.1
 | |
|          *
 | |
|          * @return object
 | |
|          */
 | |
|         private function get_latest_download_details( $addon_id = false, $newer_than = false, $fetch_readme = true ) {
 | |
|             return $this->_fs->_fetch_latest_version( $addon_id, true, WP_FS__TIME_24_HOURS_IN_SEC, $newer_than, $fetch_readme );
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Checks if a given basename has a matching folder name
 | |
|          * with the current context plugin.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.2.1.6
 | |
|          *
 | |
|          * @return bool
 | |
|          */
 | |
|         private function is_correct_folder_name() {
 | |
|             return ( $this->_fs->get_target_folder_name() == trim( dirname( $this->_fs->get_plugin_basename() ), '/\\' ) );
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * This is a special after upgrade handler for migrating modules
 | |
|          * that didn't use the '-premium' suffix folder structure before
 | |
|          * the migration.
 | |
|          *
 | |
|          * @author Vova Feldman (@svovaf)
 | |
|          * @since  1.2.1.6
 | |
|          *
 | |
|          * @param bool  $response   Install response.
 | |
|          * @param array $hook_extra Extra arguments passed to hooked filters.
 | |
|          * @param array $result     Installation result data.
 | |
|          *
 | |
|          * @return bool
 | |
|          */
 | |
|         function _maybe_update_folder_name( $response, $hook_extra, $result ) {
 | |
|             $basename = $this->_fs->get_plugin_basename();
 | |
| 
 | |
|             if ( true !== $response ||
 | |
|                  empty( $hook_extra ) ||
 | |
|                  empty( $hook_extra['plugin'] ) ||
 | |
|                  $basename !== $hook_extra['plugin']
 | |
|             ) {
 | |
|                 return $response;
 | |
|             }
 | |
| 
 | |
|             $active_plugins_basenames = get_option( 'active_plugins' );
 | |
| 
 | |
|             foreach ( $active_plugins_basenames as $key => $active_plugin_basename ) {
 | |
|                 if ( $basename === $active_plugin_basename ) {
 | |
|                     // Get filename including extension.
 | |
|                     $filename = basename( $basename );
 | |
| 
 | |
|                     $new_basename = plugin_basename(
 | |
|                         trailingslashit( $this->_fs->is_premium() ? $this->_fs->get_premium_slug() : $this->_fs->get_slug() ) .
 | |
|                         $filename
 | |
|                     );
 | |
| 
 | |
|                     // Verify that the expected correct path exists.
 | |
|                     if ( file_exists( fs_normalize_path( WP_PLUGIN_DIR . '/' . $new_basename ) ) ) {
 | |
|                         // Override active plugin name.
 | |
|                         $active_plugins_basenames[ $key ] = $new_basename;
 | |
|                         update_option( 'active_plugins', $active_plugins_basenames );
 | |
|                     }
 | |
| 
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return $response;
 | |
|         }
 | |
| 
 | |
|         #----------------------------------------------------------------------------------
 | |
|         #region Auto Activation
 | |
|         #----------------------------------------------------------------------------------
 | |
| 
 | |
|         /**
 | |
|          * Installs and active a plugin when explicitly requested that from a 3rd party service.
 | |
|          *
 | |
|          * This logic was inspired by the TGMPA GPL licensed library by Thomas Griffin.
 | |
|          *
 | |
|          * @link   http://tgmpluginactivation.com/
 | |
|          *
 | |
|          * @author Vova Feldman
 | |
|          * @since  1.2.1.7
 | |
|          *
 | |
|          * @link   https://make.wordpress.org/plugins/2017/03/16/clarification-of-guideline-8-executable-code-and-installs/
 | |
|          *
 | |
|          * @uses   WP_Filesystem
 | |
|          * @uses   WP_Error
 | |
|          * @uses   WP_Upgrader
 | |
|          * @uses   Plugin_Upgrader
 | |
|          * @uses   Plugin_Installer_Skin
 | |
|          * @uses   Plugin_Upgrader_Skin
 | |
|          *
 | |
|          * @param number|bool $plugin_id
 | |
|          *
 | |
|          * @return array
 | |
|          */
 | |
|         function install_and_activate_plugin( $plugin_id = false ) {
 | |
|             if ( ! empty( $plugin_id ) && ! FS_Plugin::is_valid_id( $plugin_id ) ) {
 | |
|                 // Invalid plugin ID.
 | |
|                 return array(
 | |
|                     'message' => $this->_fs->get_text_inline( 'Invalid module ID.', 'auto-install-error-invalid-id' ),
 | |
|                     'code'    => 'invalid_module_id',
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             $is_addon = false;
 | |
|             if ( FS_Plugin::is_valid_id( $plugin_id ) &&
 | |
|                  $plugin_id != $this->_fs->get_id()
 | |
|             ) {
 | |
|                 $addon = $this->_fs->get_addon( $plugin_id );
 | |
| 
 | |
|                 if ( ! is_object( $addon ) ) {
 | |
|                     // Invalid add-on ID.
 | |
|                     return array(
 | |
|                         'message' => $this->_fs->get_text_inline( 'Invalid module ID.', 'auto-install-error-invalid-id' ),
 | |
|                         'code'    => 'invalid_module_id',
 | |
|                     );
 | |
|                 }
 | |
| 
 | |
|                 $slug          = $addon->slug;
 | |
|                 $premium_slug  = $addon->premium_slug;
 | |
|                 $title         = $addon->title . ' ' . $this->_fs->get_text_inline( 'Add-On', 'addon' );
 | |
| 
 | |
|                 $is_addon = true;
 | |
|             } else {
 | |
|                 $slug          = $this->_fs->get_slug();
 | |
|                 $premium_slug  = $this->_fs->get_premium_slug();
 | |
|                 $title         = $this->_fs->get_plugin_title() .
 | |
|                                  ( $this->_fs->is_addon() ? ' ' . $this->_fs->get_text_inline( 'Add-On', 'addon' ) : '' );
 | |
|             }
 | |
| 
 | |
|             if ( $this->is_premium_plugin_active( $plugin_id ) ) {
 | |
|                 // Premium version already activated.
 | |
|                 return array(
 | |
|                     'message' => $is_addon ?
 | |
|                         $this->_fs->get_text_inline( 'Premium add-on version already installed.', 'auto-install-error-premium-addon-activated' ) :
 | |
|                         $this->_fs->get_text_inline( 'Premium version already active.', 'auto-install-error-premium-activated' ),
 | |
|                     'code'    => 'premium_installed',
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             $latest_version = $this->get_latest_download_details( $plugin_id, false, false );
 | |
|             $target_folder  = $premium_slug;
 | |
| 
 | |
|             // Prep variables for Plugin_Installer_Skin class.
 | |
|             $extra         = array();
 | |
|             $extra['slug'] = $target_folder;
 | |
|             $source        = $latest_version->url;
 | |
|             $api           = null;
 | |
| 
 | |
|             $install_url = add_query_arg(
 | |
|                 array(
 | |
|                     'action' => 'install-plugin',
 | |
|                     'plugin' => urlencode( $slug ),
 | |
|                 ),
 | |
|                 'update.php'
 | |
|             );
 | |
| 
 | |
|             if ( ! class_exists( 'Plugin_Upgrader', false ) ) {
 | |
|                 // Include required resources for the installation.
 | |
|                 require_once ABSPATH . 'wp-admin/includes/class-wp-upgrader.php';
 | |
|             }
 | |
| 
 | |
|             $skin_args = array(
 | |
|                 'type'   => 'web',
 | |
|                 'title'  => sprintf( $this->_fs->get_text_inline( 'Installing plugin: %s', 'installing-plugin-x' ), $title ),
 | |
|                 'url'    => esc_url_raw( $install_url ),
 | |
|                 'nonce'  => 'install-plugin_' . $slug,
 | |
|                 'plugin' => '',
 | |
|                 'api'    => $api,
 | |
|                 'extra'  => $extra,
 | |
|             );
 | |
| 
 | |
| //			$skin = new Automatic_Upgrader_Skin( $skin_args );
 | |
| //			$skin = new Plugin_Installer_Skin( $skin_args );
 | |
|             $skin = new WP_Ajax_Upgrader_Skin( $skin_args );
 | |
| 
 | |
|             // Create a new instance of Plugin_Upgrader.
 | |
|             $upgrader = new Plugin_Upgrader( $skin );
 | |
| 
 | |
|             // Perform the action and install the plugin from the $source urldecode().
 | |
|             add_filter( 'upgrader_source_selection', array( 'FS_Plugin_Updater', '_maybe_adjust_source_dir' ), 1, 3 );
 | |
| 
 | |
|             $install_result = $upgrader->install( $source );
 | |
| 
 | |
|             remove_filter( 'upgrader_source_selection', array( 'FS_Plugin_Updater', '_maybe_adjust_source_dir' ), 1 );
 | |
| 
 | |
|             if ( is_wp_error( $install_result ) ) {
 | |
|                 return array(
 | |
|                     'message' => $install_result->get_error_message(),
 | |
|                     'code'    => $install_result->get_error_code(),
 | |
|                 );
 | |
|             } elseif ( is_wp_error( $skin->result ) ) {
 | |
|                 return array(
 | |
|                     'message' => $skin->result->get_error_message(),
 | |
|                     'code'    => $skin->result->get_error_code(),
 | |
|                 );
 | |
|             } elseif ( $skin->get_errors()->get_error_code() ) {
 | |
|                 return array(
 | |
|                     'message' => $skin->get_error_messages(),
 | |
|                     'code'    => 'unknown',
 | |
|                 );
 | |
|             } elseif ( is_null( $install_result ) ) {
 | |
|                 global $wp_filesystem;
 | |
| 
 | |
|                 $error_code    = 'unable_to_connect_to_filesystem';
 | |
|                 $error_message = $this->_fs->get_text_inline( 'Unable to connect to the filesystem. Please confirm your credentials.' );
 | |
| 
 | |
|                 // Pass through the error from WP_Filesystem if one was raised.
 | |
|                 if ( $wp_filesystem instanceof WP_Filesystem_Base &&
 | |
|                      is_wp_error( $wp_filesystem->errors ) &&
 | |
|                      $wp_filesystem->errors->get_error_code()
 | |
|                 ) {
 | |
|                     $error_message = $wp_filesystem->errors->get_error_message();
 | |
|                 }
 | |
| 
 | |
|                 return array(
 | |
|                     'message' => $error_message,
 | |
|                     'code'    => $error_code,
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             // Grab the full path to the main plugin's file.
 | |
|             $plugin_activate = $upgrader->plugin_info();
 | |
| 
 | |
|             // Try to activate the plugin.
 | |
|             $activation_result = $this->try_activate_plugin( $plugin_activate );
 | |
| 
 | |
|             if ( is_wp_error( $activation_result ) ) {
 | |
|                 return array(
 | |
|                     'message' => $activation_result->get_error_message(),
 | |
|                     'code'    => $activation_result->get_error_code(),
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             return $skin->get_upgrade_messages();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Tries to activate a plugin. If fails, returns the error.
 | |
|          *
 | |
|          * @author Vova Feldman
 | |
|          * @since  1.2.1.7
 | |
|          *
 | |
|          * @param string $file_path Path within wp-plugins/ to main plugin file.
 | |
|          *                          This determines the styling of the output messages.
 | |
|          *
 | |
|          * @return bool|WP_Error
 | |
|          */
 | |
|         protected function try_activate_plugin( $file_path ) {
 | |
|             $activate = activate_plugin( $file_path, '', $this->_fs->is_network_active() );
 | |
| 
 | |
|             return is_wp_error( $activate ) ?
 | |
|                 $activate :
 | |
|                 true;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Check if a premium module version is already active.
 | |
|          *
 | |
|          * @author Vova Feldman
 | |
|          * @since  1.2.1.7
 | |
|          *
 | |
|          * @param number|bool $plugin_id
 | |
|          *
 | |
|          * @return bool
 | |
|          */
 | |
|         private function is_premium_plugin_active( $plugin_id = false ) {
 | |
|             if ( $plugin_id != $this->_fs->get_id() ) {
 | |
|                 return $this->_fs->is_addon_activated( $plugin_id, true );
 | |
|             }
 | |
| 
 | |
|             return is_plugin_active( $this->_fs->premium_plugin_basename() );
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Store the basename since it's not always available in the `_maybe_adjust_source_dir` method below.
 | |
|          *
 | |
|          * @author Leo Fajardo (@leorw)
 | |
|          * @since 2.2.1
 | |
|          *
 | |
|          * @param bool|WP_Error $response   Response.
 | |
|          * @param array         $hook_extra Extra arguments passed to hooked filters.
 | |
|          *
 | |
|          * @return bool|WP_Error
 | |
|          */
 | |
|         static function _store_basename_for_source_adjustment( $response, $hook_extra ) {
 | |
|             if ( isset( $hook_extra['plugin'] ) ) {
 | |
|                 self::$_upgrade_basename = $hook_extra['plugin'];
 | |
|             } else if ( isset( $hook_extra['theme'] ) ) {
 | |
|                 self::$_upgrade_basename = $hook_extra['theme'];
 | |
|             } else {
 | |
|                 self::$_upgrade_basename = null;
 | |
|             }
 | |
| 
 | |
|             return $response;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Adjust the plugin directory name if necessary.
 | |
|          * Assumes plugin has a folder (not a single file plugin).
 | |
|          *
 | |
|          * The final destination directory of a plugin is based on the subdirectory name found in the
 | |
|          * (un)zipped source. In some cases this subdirectory name is not the same as the expected
 | |
|          * slug and the plugin will not be recognized as installed. This is fixed by adjusting
 | |
|          * the temporary unzipped source subdirectory name to the expected plugin slug.
 | |
|          *
 | |
|          * @author Vova Feldman
 | |
|          * @since  1.2.1.7
 | |
|          * @since  2.2.1 The method was converted to static since when the admin update bulk products via the Updates section, the logic applies the `upgrader_source_selection` filter for every product that is being updated.
 | |
|          *
 | |
|          * @param string       $source        Path to upgrade/zip-file-name.tmp/subdirectory/.
 | |
|          * @param string       $remote_source Path to upgrade/zip-file-name.tmp.
 | |
|          * @param \WP_Upgrader $upgrader      Instance of the upgrader which installs the plugin.
 | |
|          *
 | |
|          * @return string|WP_Error
 | |
|          */
 | |
|         static function _maybe_adjust_source_dir( $source, $remote_source, $upgrader ) {
 | |
|             if ( ! is_object( $GLOBALS['wp_filesystem'] ) ) {
 | |
|                 return $source;
 | |
|             }
 | |
| 
 | |
|             $basename = self::$_upgrade_basename;
 | |
|             $is_theme = false;
 | |
| 
 | |
|             // Figure out what the slug is supposed to be.
 | |
|             if ( isset( $upgrader->skin->options['extra'] ) ) {
 | |
|                 // Set by the auto-install logic.
 | |
|                 $desired_slug = $upgrader->skin->options['extra']['slug'];
 | |
|             } else if ( ! empty( $basename ) ) {
 | |
|                 /**
 | |
|                  * If it doesn't end with ".php", it's a theme.
 | |
|                  *
 | |
|                  * @author Leo Fajardo (@leorw)
 | |
|                  * @since 2.2.1
 | |
|                  */
 | |
|                 $is_theme = ( ! fs_ends_with( $basename, '.php' ) );
 | |
| 
 | |
|                 $desired_slug = ( ! $is_theme ) ?
 | |
|                     dirname( $basename ) :
 | |
|                     // Theme slug
 | |
|                     $basename;
 | |
|             } else {
 | |
|                 // Can't figure out the desired slug, stop the execution.
 | |
|                 return $source;
 | |
|             }
 | |
| 
 | |
|             if ( is_multisite() ) {
 | |
|                 /**
 | |
|                  * If we are running in a multisite environment and the product is not network activated,
 | |
|                  * the instance will not exist anyway. Therefore, try to update the source if necessary
 | |
|                  * regardless if the Freemius instance of the product exists or not.
 | |
|                  *
 | |
|                  * @author Vova Feldman
 | |
|                  */
 | |
|             } else if ( ! empty( $basename ) ) {
 | |
|                 $fs = Freemius::get_instance_by_file(
 | |
|                     $basename,
 | |
|                     $is_theme ?
 | |
|                         WP_FS__MODULE_TYPE_THEME :
 | |
|                         WP_FS__MODULE_TYPE_PLUGIN
 | |
|                 );
 | |
| 
 | |
|                 if ( ! is_object( $fs ) ) {
 | |
|                     /**
 | |
|                      * If the Freemius instance does not exist on a non-multisite network environment, it means that:
 | |
|                      *  1. The product is not powered by Freemius; OR
 | |
|                      *  2. The product is not activated, therefore, we don't mind if after the update the folder name will change.
 | |
|                      *
 | |
|                      * @author Leo Fajardo (@leorw)
 | |
|                      * @since  2.2.1
 | |
|                      */
 | |
|                     return $source;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             $subdir_name = untrailingslashit( str_replace( trailingslashit( $remote_source ), '', $source ) );
 | |
| 
 | |
|             if ( ! empty( $subdir_name ) && $subdir_name !== $desired_slug ) {
 | |
|                 $from_path = untrailingslashit( $source );
 | |
|                 $to_path   = trailingslashit( $remote_source ) . $desired_slug;
 | |
| 
 | |
|                 if ( true === $GLOBALS['wp_filesystem']->move( $from_path, $to_path ) ) {
 | |
|                     return trailingslashit( $to_path );
 | |
|                 }
 | |
| 
 | |
|                 return new WP_Error(
 | |
|                     'rename_failed',
 | |
|                     fs_text_inline( 'The remote plugin package does not contain a folder with the desired slug and renaming did not work.', 'module-package-rename-failure' ),
 | |
|                     array(
 | |
|                         'found'    => $subdir_name,
 | |
|                         'expected' => $desired_slug
 | |
|                     )
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             return $source;
 | |
|         }
 | |
| 
 | |
|         #endregion
 | |
|     }
 |