diff options
Diffstat (limited to 'plugins/jetpack/class.jetpack-sync.php')
-rw-r--r-- | plugins/jetpack/class.jetpack-sync.php | 1104 |
1 files changed, 0 insertions, 1104 deletions
diff --git a/plugins/jetpack/class.jetpack-sync.php b/plugins/jetpack/class.jetpack-sync.php deleted file mode 100644 index 99b10004..00000000 --- a/plugins/jetpack/class.jetpack-sync.php +++ /dev/null @@ -1,1104 +0,0 @@ -<?php - -/** - * Request that a piece of data on this WordPress install be synced back to the - * Jetpack server for remote processing/notifications/etc - */ -class Jetpack_Sync { - // What modules want to sync what content - public $sync_conditions = array( 'posts' => array(), 'comments' => array() ); - - // We keep track of all the options registered for sync so that we can sync them all if needed - public $sync_options = array(); - - public $sync_constants = array(); - - // Keep trac of status transitions, which we wouldn't always know about on the Jetpack Servers but are important when deciding what to do with the sync. - public $post_transitions = array(); - public $comment_transitions = array(); - - // Objects to sync - public $sync = array(); - - function __construct() { - // WP Cron action. Only used on upgrade - add_action( 'jetpack_sync_all_registered_options', array( $this, 'sync_all_registered_options' ) ); - add_action( 'jetpack_heartbeat', array( $this, 'sync_all_registered_options' ) ); - - // Sync constants on heartbeat and plugin upgrade and connects - add_action( 'init', array( $this, 'register_constants_as_options' ) ); - add_action( 'jetpack_sync_all_registered_options', array( $this, 'sync_all_constants' ) ); - add_action( 'jetpack_heartbeat', array( $this, 'sync_all_constants' ) ); - - add_action( 'jetpack_activate_module', array( $this, 'sync_module_constants' ), 10, 1 ); - } - -/* Static Methods for Modules */ - - /** - * @param string $file __FILE__ - * @param array settings: - * post_types => array( post_type slugs ): The post types to sync. Default: post, page - * post_stati => array( post_status slugs ): The post stati to sync. Default: publish - */ - static function sync_posts( $file, array $settings = null ) { - if ( is_network_admin() ) return; - $jetpack = Jetpack::init(); - $args = func_get_args(); - return call_user_func_array( array( $jetpack->sync, 'posts' ), $args ); - } - - /** - * @param string $file __FILE__ - * @param array settings: - * post_types => array( post_type slugs ): The post types to sync. Default: post, page - * post_stati => array( post_status slugs ): The post stati to sync. Default: publish - * comment_types => array( comment_type slugs ): The comment types to sync. Default: '', comment, trackback, pingback - * comment_stati => array( comment_status slugs ): The comment stati to sync. Default: approved - */ - static function sync_comments( $file, array $settings = null ) { - if ( is_network_admin() ) return; - $jetpack = Jetpack::init(); - $args = func_get_args(); - return call_user_func_array( array( $jetpack->sync, 'comments' ), $args ); - } - - /** - * @param string $file __FILE__ - * @param string $option, Option name to sync - * @param string $option ... - */ - static function sync_options( $file, $option /*, $option, ... */ ) { - if ( is_network_admin() ) return; - $jetpack = Jetpack::init(); - $args = func_get_args(); - return call_user_func_array( array( $jetpack->sync, 'options' ), $args ); - } - /** - * @param string $file __FILE__ - * @param string $option, Option name to sync - * @param string $option ... - */ - static function sync_constant( $file, $constant ) { - if ( is_network_admin() ) return; - $jetpack = Jetpack::init(); - $args = func_get_args(); - return call_user_func_array( array( $jetpack->sync, 'constant' ), $args ); - } - -/* Internal Methods */ - - /** - * Create a sync object/request - * - * @param string $object Type of object to sync -- [ post | comment | option ] - * @param int $id Unique identifier - * @param array $settings - */ - function register( $object, $id = false, array $settings = null ) { - // Since we've registered something for sync, hook it up to execute on shutdown if we haven't already - if ( !$this->sync ) { - if ( function_exists( 'ignore_user_abort' ) ) { - ignore_user_abort( true ); - } - add_action( 'shutdown', array( $this, 'sync' ), 9 ); // Right before async XML-RPC - } - - $defaults = array( - 'on_behalf_of' => array(), // What modules want this data - ); - $settings = wp_parse_args( $settings, $defaults ); - - if ( !isset( $this->sync[$object] ) ) { - $this->sync[$object] = array(); - } - - // Store the settings for this object - if ( - // First time for this object - !isset( $this->sync[$object][$id] ) - ) { - // Easy: store the current settings - $this->sync[$object][$id] = $settings; - } else { - // Not as easy: we have to manually merge the settings from previous runs for this object with the settings for this run - - $this->sync[$object][$id]['on_behalf_of'] = array_unique( array_merge( $this->sync[$object][$id]['on_behalf_of'], $settings['on_behalf_of'] ) ); - } - - $delete_prefix = 'delete_'; - if ( 0 === strpos( $object, $delete_prefix ) ) { - $unset_object = substr( $object, strlen( $delete_prefix ) ); - } else { - $unset_object = "{$delete_prefix}{$object}"; - } - - // Ensure post ... delete_post yields a delete operation - // Ensure delete_post ... post yields a sync post operation - // Ensure update_option() ... delete_option() ends up as a delete - // Ensure delete_option() ... update_option() ends up as an update - // Etc. - unset( $this->sync[$unset_object][$id] ); - - return true; - } - - function get_common_sync_data() { - $available_modules = Jetpack::get_available_modules(); - $active_modules = Jetpack::get_active_modules(); - $modules = array(); - foreach ( $available_modules as $available_module ) { - $modules[$available_module] = in_array( $available_module, $active_modules ); - } - $modules['vaultpress'] = class_exists( 'VaultPress' ) || function_exists( 'vaultpress_contact_service' ); - - $sync_data = array( - 'modules' => $modules, - 'version' => JETPACK__VERSION, - 'is_multisite' => is_multisite(), - ); - - return $sync_data; - } - - /** - * Set up all the data and queue it for the outgoing XML-RPC request - */ - function sync() { - if ( !$this->sync ) { - return false; - } - - // Don't sync anything from a staging site. - if ( Jetpack::is_development_mode() || Jetpack::is_staging_site() ) { - return false; - } - - $sync_data = $this->get_common_sync_data(); - - $wp_importing = defined( 'WP_IMPORTING' ) && WP_IMPORTING; - - foreach ( $this->sync as $sync_operation_type => $sync_operations ) { - switch ( $sync_operation_type ) { - case 'post': - if ( $wp_importing ) { - break; - } - - $global_post = isset( $GLOBALS['post'] ) ? $GLOBALS['post'] : null; - $GLOBALS['post'] = null; - foreach ( $sync_operations as $post_id => $settings ) { - $sync_data['post'][$post_id] = $this->get_post( $post_id ); - if ( isset( $this->post_transitions[$post_id] ) ) { - $sync_data['post'][$post_id]['transitions'] = $this->post_transitions[$post_id]; - } else { - $sync_data['post'][$post_id]['transitions'] = array( false, false ); - } - $sync_data['post'][$post_id]['on_behalf_of'] = $settings['on_behalf_of']; - } - $GLOBALS['post'] = $global_post; - unset( $global_post ); - break; - case 'comment': - if ( $wp_importing ) { - break; - } - - $global_comment = isset( $GLOBALS['comment'] ) ? $GLOBALS['comment'] : null; - unset( $GLOBALS['comment'] ); - foreach ( $sync_operations as $comment_id => $settings ) { - $sync_data['comment'][$comment_id] = $this->get_comment( $comment_id ); - if ( isset( $this->comment_transitions[$comment_id] ) ) { - $sync_data['comment'][$comment_id]['transitions'] = $this->comment_transitions[$comment_id]; - } else { - $sync_data['comment'][$comment_id]['transitions'] = array( false, false ); - } - $sync_data['comment'][$comment_id]['on_behalf_of'] = $settings['on_behalf_of']; - } - $GLOBALS['comment'] = $global_comment; - unset( $global_comment ); - break; - case 'option' : - foreach ( $sync_operations as $option => $settings ) { - $sync_data['option'][ $option ] = array( 'value' => get_option( $option ) ); - } - break; - - case 'constant' : - foreach( $sync_operations as $constant => $settings ) { - $sync_data['constant'][ $constant ] = array( 'value' => $this->get_constant( $constant ) ); - } - break; - - case 'delete_post': - case 'delete_comment': - foreach ( $sync_operations as $object_id => $settings ) { - $sync_data[$sync_operation_type][$object_id] = array( 'on_behalf_of' => $settings['on_behalf_of'] ); - } - break; - case 'delete_option' : - foreach ( $sync_operations as $object_id => $settings ) { - $sync_data[$sync_operation_type][$object_id] = true; - } - break; - } - } - Jetpack::xmlrpc_async_call( 'jetpack.syncContent', $sync_data ); - } - - /** - * Format and return content data from a direct xmlrpc request for it. - * - * @param array $content_ids: array( 'posts' => array of ids, 'comments' => array of ids, 'options' => array of options ) - */ - function get_content( $content_ids ) { - $sync_data = $this->get_common_sync_data(); - - if ( isset( $content_ids['posts'] ) ) { - foreach ( $content_ids['posts'] as $id ) { - $sync_data['post'][$id] = $this->get_post( $id ); - } - } - - if ( isset( $content_ids['comments'] ) ) { - foreach ( $content_ids['comments'] as $id ) { - $sync_data['comment'][$id] = $this->get_post( $id ); - } - } - - if ( isset( $content_ids['options'] ) ) { - foreach ( $content_ids['options'] as $option ) { - $sync_data['option'][$option] = array( 'value' => get_option( $option ) ); - } - } - - return $sync_data; - } - - /** - * Helper method for registering a post for sync - * - * @param int $id wp_posts.ID - * @param array $settings Sync data - */ - function register_post( $id, array $settings = null ) { - $id = (int) $id; - if ( !$id ) { - return false; - } - - $post = get_post( $id ); - if ( !$post ) { - return false; - } - - $settings = wp_parse_args( $settings, array( - 'on_behalf_of' => array(), - ) ); - - return $this->register( 'post', $id, $settings ); - } - - /** - * Helper method for registering a comment for sync - * - * @param int $id wp_comments.comment_ID - * @param array $settings Sync data - */ - function register_comment( $id, array $settings = null ) { - $id = (int) $id; - if ( !$id ) { - return false; - } - - $comment = get_comment( $id ); - if ( !$comment || empty( $comment->comment_post_ID ) ) { - return false; - } - - $post = get_post( $comment->comment_post_ID ); - if ( !$post ) { - return false; - } - - $settings = wp_parse_args( $settings, array( - 'on_behalf_of' => array(), - ) ); - - return $this->register( 'comment', $id, $settings ); - } - -/* Posts Sync */ - - function posts( $file, array $settings = null ) { - $module_slug = Jetpack::get_module_slug( $file ); - - $defaults = array( - 'post_types' => array( 'post', 'page' ), - 'post_stati' => array( 'publish' ), - ); - - $this->sync_conditions['posts'][$module_slug] = wp_parse_args( $settings, $defaults ); - - add_action( 'transition_post_status', array( $this, 'transition_post_status_action' ), 10, 3 ); - add_action( 'delete_post', array( $this, 'delete_post_action' ) ); - } - - function delete_post_action( $post_id ) { - $post = get_post( $post_id ); - if ( !$post ) { - return $this->register( 'delete_post', (int) $post_id ); - } - - $this->transition_post_status_action( 'delete', $post->post_status, $post ); - } - - function transition_post_status_action( $new_status, $old_status, $post ) { - $sync = $this->get_post_sync_operation( $new_status, $old_status, $post, $this->sync_conditions['posts'] ); - if ( !$sync ) { - // No module wants to sync this post - return false; - } - - // Track post transitions - if ( isset( $this->post_transitions[$post->ID] ) ) { - // status changed more than once - keep tha most recent $new_status - $this->post_transitions[$post->ID][0] = $new_status; - } else { - $this->post_transitions[$post->ID] = array( $new_status, $old_status ); - } - - $operation = $sync['operation']; - unset( $sync['operation'] ); - - switch ( $operation ) { - case 'delete' : - return $this->register( 'delete_post', (int) $post->ID, $sync ); - case 'submit' : - return $this->register_post( (int) $post->ID, $sync ); - } - } - - function get_post_sync_operation( $new_status, $old_status, $post, $module_conditions ) { - $delete_on_behalf_of = array(); - $submit_on_behalf_of = array(); - $delete_stati = array( 'delete' ); - $cache_cleared = false; - - foreach ( $module_conditions as $module => $conditions ) { - if ( !in_array( $post->post_type, $conditions['post_types'] ) ) { - continue; - } - - $deleted_post = in_array( $new_status, $delete_stati ); - - if ( $deleted_post ) { - $delete_on_behalf_of[] = $module; - } else { - if ( ! $cache_cleared ) { - // inefficient to clear cache more than once - clean_post_cache( $post->ID ); - $cache_cleared = true; - } - $new_status = get_post_status( $post->ID ); // Inherited status is resolved here - } - - $old_status_in_stati = in_array( $old_status, $conditions['post_stati'] ); - $new_status_in_stati = in_array( $new_status, $conditions['post_stati'] ); - - if ( $old_status_in_stati && !$new_status_in_stati ) { - // Jetpack no longer needs the post - if ( !$deleted_post ) { - $delete_on_behalf_of[] = $module; - } // else, we've already flagged it above - continue; - } - - if ( !$new_status_in_stati ) { - continue; - } - - // At this point, we know we want to sync the post, not delete it - $submit_on_behalf_of[] = $module; - } - - if ( !empty( $submit_on_behalf_of ) ) { - return array( 'operation' => 'submit', 'on_behalf_of' => $submit_on_behalf_of ); - } - - if ( !empty( $delete_on_behalf_of ) ) { - return array( 'operation' => 'delete', 'on_behalf_of' => $delete_on_behalf_of ); - } - - return false; - } - - /** - * Get a post and associated data in the standard JP format. - * Cannot be called statically - * - * @param int $id Post ID - * @return Array containing full post details - */ - function get_post( $id ) { - $post_obj = get_post( $id ); - if ( !$post_obj ) - return false; - - if ( is_callable( $post_obj, 'to_array' ) ) { - // WP >= 3.5 - $post = $post_obj->to_array(); - } else { - // WP < 3.5 - $post = get_object_vars( $post_obj ); - } - - if ( 0 < strlen( $post['post_password'] ) ) { - $post['post_password'] = 'auto-' . wp_generate_password( 10, false ); // We don't want the real password. Just pass something random. - } - - // local optimizations - unset( - $post['filter'], - $post['ancestors'], - $post['post_content_filtered'], - $post['to_ping'], - $post['pinged'] - ); - - if ( $this->is_post_public( $post ) ) { - $post['post_is_public'] = Jetpack_Options::get_option( 'public' ); - } else { - //obscure content - $post['post_content'] = ''; - $post['post_excerpt'] = ''; - $post['post_is_public'] = false; - } - $post_type_obj = get_post_type_object( $post['post_type'] ); - $post['post_is_excluded_from_search'] = $post_type_obj->exclude_from_search; - - $post['tax'] = array(); - $taxonomies = get_object_taxonomies( $post_obj ); - foreach ( $taxonomies as $taxonomy ) { - $terms = get_object_term_cache( $post_obj->ID, $taxonomy ); - if ( empty( $terms ) ) - $terms = wp_get_object_terms( $post_obj->ID, $taxonomy ); - $term_names = array(); - foreach ( $terms as $term ) { - $term_names[] = $term->name; - } - $post['tax'][$taxonomy] = $term_names; - } - - $meta = get_post_meta( $post_obj->ID, false ); - $post['meta'] = array(); - foreach ( $meta as $key => $value ) { - $post['meta'][$key] = array_map( 'maybe_unserialize', $value ); - } - - $post['extra'] = array( - 'author' => get_the_author_meta( 'display_name', $post_obj->post_author ), - 'author_email' => get_the_author_meta( 'email', $post_obj->post_author ), - 'dont_email_post_to_subs' => get_post_meta( $post_obj->ID, '_jetpack_dont_email_post_to_subs', true ), - ); - - if ( $fid = get_post_thumbnail_id( $id ) ) { - $feature = wp_get_attachment_image_src( $fid, 'large' ); - if ( ! empty( $feature[0] ) ) { - $post['extra']['featured_image'] = $feature[0]; - } - - $attachment = get_post( $fid ); - if ( ! empty( $attachment ) ) { - $metadata = wp_get_attachment_metadata( $fid ); - - $post['extra']['post_thumbnail'] = array( - 'ID' => (int) $fid, - 'URL' => (string) wp_get_attachment_url( $fid ), - 'guid' => (string) $attachment->guid, - 'mime_type' => (string) $attachment->post_mime_type, - 'width' => (int) isset( $metadata['width'] ) ? $metadata['width'] : 0, - 'height' => (int) isset( $metadata['height'] ) ? $metadata['height'] : 0, - ); - - if ( isset( $metadata['duration'] ) ) { - $post['extra']['post_thumbnail'] = (int) $metadata['duration']; - } - - /** - * Filters the Post Thumbnail information returned for a specific post. - * - * @since 3.3.0 - * - * @param array $post['extra']['post_thumbnail'] { - * Array of details about the Post Thumbnail. - * @param int ID Post Thumbnail ID. - * @param string URL Post thumbnail URL. - * @param string guid Post thumbnail guid. - * @param string mime_type Post thumbnail mime type. - * @param int width Post thumbnail width. - * @param int height Post thumbnail height. - * } - */ - $post['extra']['post_thumbnail'] = (object) apply_filters( 'get_attachment', $post['extra']['post_thumbnail'] ); - } - } - - $post['permalink'] = get_permalink( $post_obj->ID ); - $post['shortlink'] = wp_get_shortlink( $post_obj->ID ); - /** - * Allow modules to send extra info on the sync post process. - * - * @since 2.8.0 - * - * @param array $args Array of custom data to attach to a post. - * @param Object $post_obj Object returned by get_post() for a given post ID. - */ - $post['module_custom_data'] = apply_filters( 'jetpack_sync_post_module_custom_data', array(), $post_obj ); - return $post; - } - - /** - * Decide whether a post/page/attachment is visible to the public. - * - * @param array $post - * @return bool - */ - function is_post_public( $post ) { - if ( !is_array( $post ) ) { - $post = (array) $post; - } - - if ( 0 < strlen( $post['post_password'] ) ) - return false; - if ( ! in_array( $post['post_type'], get_post_types( array( 'public' => true ) ) ) ) - return false; - $post_status = get_post_status( $post['ID'] ); // Inherited status is resolved here. - if ( ! in_array( $post_status, get_post_stati( array( 'public' => true ) ) ) ) - return false; - return true; - } - -/* Comments Sync */ - - function comments( $file, array $settings = null ) { - $module_slug = Jetpack::get_module_slug( $file ); - - $defaults = array( - 'post_types' => array( 'post', 'page' ), // For what post types will we sync comments? - 'post_stati' => array( 'publish' ), // For what post stati will we sync comments? - 'comment_types' => array( '', 'comment', 'trackback', 'pingback' ), // What comment types will we sync? - 'comment_stati' => array( 'approved' ), // What comment stati will we sync? - ); - - $settings = wp_parse_args( $settings, $defaults ); - - $this->sync_conditions['comments'][$module_slug] = $settings; - - add_action( 'wp_insert_comment', array( $this, 'wp_insert_comment_action' ), 10, 2 ); - add_action( 'transition_comment_status', array( $this, 'transition_comment_status_action' ), 10, 3 ); - add_action( 'edit_comment', array( $this, 'edit_comment_action' ) ); - } - - /* - * This is really annoying. If you edit a comment, but don't change the status, WordPress doesn't fire the transition_comment_status hook. - * That means we have to catch these comments on the edit_comment hook, but ignore comments on that hook when the transition_comment_status does fire. - */ - function edit_comment_action( $comment_id ) { - $comment = get_comment( $comment_id ); - $new_status = $this->translate_comment_status( $comment->comment_approved ); - add_action( "comment_{$new_status}_{$comment->comment_type}", array( $this, 'transition_comment_status_for_comments_whose_status_does_not_change' ), 10, 2 ); - } - - function wp_insert_comment_action( $comment_id, $comment ) { - $this->transition_comment_status_action( $comment->comment_approved, 'new', $comment ); - } - - function transition_comment_status_for_comments_whose_status_does_not_change( $comment_id, $comment ) { - if ( isset( $this->comment_transitions[$comment_id] ) ) { - return $this->transition_comment_status_action( $comment->comment_approved, $this->comment_transitions[$comment_id][1], $comment ); - } - - return $this->transition_comment_status_action( $comment->comment_approved, $comment->comment_approved, $comment ); - } - - function translate_comment_status( $status ) { - switch ( (string) $status ) { - case '0' : - case 'hold' : - return 'unapproved'; - case '1' : - case 'approve' : - return 'approved'; - } - - return $status; - } - - function transition_comment_status_action( $new_status, $old_status, $comment ) { - $post = get_post( $comment->comment_post_ID ); - if ( !$post ) { - return false; - } - - foreach ( array( 'new_status', 'old_status' ) as $_status ) { - $$_status = $this->translate_comment_status( $$_status ); - } - - // Track comment transitions - if ( isset( $this->comment_transitions[$comment->comment_ID] ) ) { - // status changed more than once - keep tha most recent $new_status - $this->comment_transitions[$comment->comment_ID][0] = $new_status; - } else { - $this->comment_transitions[$comment->comment_ID] = array( $new_status, $old_status ); - } - - $post_sync = $this->get_post_sync_operation( $post->post_status, '_jetpack_test_sync', $post, $this->sync_conditions['comments'] ); - - if ( !$post_sync ) { - // No module wants to sync this comment because its post doesn't match any sync conditions - return false; - } - - if ( 'delete' == $post_sync['operation'] ) { - // Had we been looking at post sync operations (instead of comment sync operations), - // this comment's post would have been deleted. Don't sync the comment. - return false; - } - - $delete_on_behalf_of = array(); - $submit_on_behalf_of = array(); - $delete_stati = array( 'delete' ); - - foreach ( $this->sync_conditions['comments'] as $module => $conditions ) { - if ( !in_array( $comment->comment_type, $conditions['comment_types'] ) ) { - continue; - } - - $deleted_comment = in_array( $new_status, $delete_stati ); - - if ( $deleted_comment ) { - $delete_on_behalf_of[] = $module; - } - - $old_status_in_stati = in_array( $old_status, $conditions['comment_stati'] ); - $new_status_in_stati = in_array( $new_status, $conditions['comment_stati'] ); - - if ( $old_status_in_stati && !$new_status_in_stati ) { - // Jetpack no longer needs the comment - if ( !$deleted_comment ) { - $delete_on_behalf_of[] = $module; - } // else, we've already flagged it above - continue; - } - - if ( !$new_status_in_stati ) { - continue; - } - - // At this point, we know we want to sync the comment, not delete it - $submit_on_behalf_of[] = $module; - } - - if ( ! empty( $submit_on_behalf_of ) ) { - $this->register_post( $comment->comment_post_ID, array( 'on_behalf_of' => $submit_on_behalf_of ) ); - return $this->register_comment( $comment->comment_ID, array( 'on_behalf_of' => $submit_on_behalf_of ) ); - } - - if ( !empty( $delete_on_behalf_of ) ) { - return $this->register( 'delete_comment', $comment->comment_ID, array( 'on_behalf_of' => $delete_on_behalf_of ) ); - } - - return false; - } - - /** - * Get a comment and associated data in the standard JP format. - * Cannot be called statically - * - * @param int $id Comment ID - * @return Array containing full comment details - */ - function get_comment( $id ) { - $comment_obj = get_comment( $id ); - if ( !$comment_obj ) - return false; - $comment = get_object_vars( $comment_obj ); - - $meta = get_comment_meta( $id, false ); - $comment['meta'] = array(); - foreach ( $meta as $key => $value ) { - $comment['meta'][$key] = array_map( 'maybe_unserialize', $value ); - } - - return $comment; - } - -/* Options Sync */ - - /* Ah... so much simpler than Posts and Comments :) */ - function options( $file, $option /*, $option, ... */ ) { - $options = func_get_args(); - $file = array_shift( $options ); - - $module_slug = Jetpack::get_module_slug( $file ); - - if ( !isset( $this->sync_options[$module_slug] ) ) { - $this->sync_options[$module_slug] = array(); - } - - foreach ( $options as $option ) { - $this->sync_options[$module_slug][] = $option; - add_action( "delete_option_{$option}", array( $this, 'deleted_option_action' ) ); - add_action( "update_option_{$option}", array( $this, 'updated_option_action' ) ); - add_action( "add_option_{$option}", array( $this, 'added_option_action' ) ); - } - - $this->sync_options[$module_slug] = array_unique( $this->sync_options[$module_slug] ); - } - - function deleted_option_action( $option ) { - $this->register( 'delete_option', $option ); - } - - function updated_option_action() { - // The value of $option isn't passed to the filter - // Calculate it - $option = current_filter(); - $prefix = 'update_option_'; - if ( 0 !== strpos( $option, $prefix ) ) { - return; - } - $option = substr( $option, strlen( $prefix ) ); - - $this->added_option_action( $option ); - } - - function added_option_action( $option ) { - $this->register( 'option', $option ); - } - - function sync_all_module_options( $module_slug ) { - if ( empty( $this->sync_options[$module_slug] ) ) { - return; - } - - foreach ( $this->sync_options[$module_slug] as $option ) { - $this->added_option_action( $option ); - } - } - - function sync_all_registered_options() { - if ( 'jetpack_sync_all_registered_options' == current_filter() ) { - add_action( 'shutdown', array( $this, 'register_all_options' ), 8 ); - } else { - wp_schedule_single_event( time(), 'jetpack_sync_all_registered_options', array( $this->sync_options ) ); - } - } - - /** - * All the options that are defined in modules as well as class.jetpack.php will get synced. - * Registers all options to be synced. - */ - function register_all_options() { - $all_registered_options = array_unique( call_user_func_array( 'array_merge', $this->sync_options ) ); - foreach ( $all_registered_options as $option ) { - $this->added_option_action( $option ); - } - } - -/* Constants Sync */ - - function get_all_constants() { - return array( - 'EMPTY_TRASH_DAYS', - 'WP_POST_REVISIONS', - 'AUTOMATIC_UPDATER_DISABLED', - 'ABSPATH', - 'WP_CONTENT_DIR', - 'FS_METHOD', - 'DISALLOW_FILE_EDIT', - 'DISALLOW_FILE_MODS', - 'WP_AUTO_UPDATE_CORE', - 'WP_HTTP_BLOCK_EXTERNAL', - 'WP_ACCESSIBLE_HOSTS', - ); - } - /** - * This lets us get the constant value like get_option( 'jetpack_constant_CONSTANT' ); - * Not the best way to get the constant value but necessery in some cases like in the API. - */ - function register_constants_as_options() { - foreach( $this->get_all_constants() as $constant ) { - add_filter( 'pre_option_jetpack_constant_'. $constant, array( $this, 'get_default_constant' ) ); - } - } - - function sync_all_constants() { - // add the constant to sync. - foreach( $this->get_all_constants() as $constant ) { - $this->register_constant( $constant ); - } - add_action( 'shutdown', array( $this, 'register_all_module_constants' ), 8 ); - } - - /** - * Returns default values of Constants - */ - function default_constant( $constant ) { - switch( $constant ) { - case 'WP_AUTO_UPDATE_CORE': - return 'minor'; - break; - - default: - return null; - break; - } - } - - function register_all_module_constants() { - // also add the contstants from each module to be synced. - foreach( $this->sync_constants as $module ) { - foreach( $module as $constant ) { - $this->register_constant( $constant ); - } - } - } - - /** - * Sync constants required by the module that was just activated. - * If you add Jetpack_Sync::sync_constant( __FILE__, 'HELLO_WORLD' ); - * to the module it will start syncing the constant after the constant has been updated. - * - * This function gets called on module activation. - */ - function sync_module_constants( $module ) { - - if ( isset( $this->sync_constants[ $module ] ) && is_array( $this->sync_constants[ $module ] ) ) { - // also add the contstants from each module to be synced. - foreach( $this->sync_constants[ $module ] as $constant ) { - $this->register_constant( $constant ); - } - } - } - - public function reindex_needed() { - return ( $this->_get_post_count_local() != $this->_get_post_count_cloud() ); - } - - public function reindex_trigger() { - $response = array( 'status' => 'ERROR' ); - - // Force a privacy check - Jetpack::check_privacy( JETPACK__PLUGIN_FILE ); - - Jetpack::load_xml_rpc_client(); - $client = new Jetpack_IXR_Client( array( - 'user_id' => JETPACK_MASTER_USER, - ) ); - - $client->query( 'jetpack.reindexTrigger' ); - - if ( !$client->isError() ) { - $response = $client->getResponse(); - Jetpack_Options::update_option( 'sync_bulk_reindexing', true ); - } - - return $response; - } - - public function reindex_status() { - $response = array( 'status' => 'ERROR' ); - - // Assume reindexing is done if it was not triggered in the first place - if ( false === Jetpack_Options::get_option( 'sync_bulk_reindexing' ) ) { - return array( 'status' => 'DONE' ); - } - - Jetpack::load_xml_rpc_client(); - $client = new Jetpack_IXR_Client( array( - 'user_id' => JETPACK_MASTER_USER, - ) ); - - $client->query( 'jetpack.reindexStatus' ); - - if ( !$client->isError() ) { - $response = $client->getResponse(); - if ( 'DONE' == $response['status'] ) { - Jetpack_Options::delete_option( 'sync_bulk_reindexing' ); - } - } - - return $response; - } - - public function reindex_ui() { - $strings = json_encode( array( - 'WAITING' => array( - 'action' => __( 'Refresh Status', 'jetpack' ), - 'status' => __( 'Indexing request queued and waiting…', 'jetpack' ), - ), - 'INDEXING' => array( - 'action' => __( 'Refresh Status', 'jetpack' ), - 'status' => __( 'Indexing posts', 'jetpack' ), - ), - 'DONE' => array( - 'action' => __( 'Reindex Posts', 'jetpack' ), - 'status' => __( 'Posts indexed.', 'jetpack' ), - ), - 'ERROR' => array( - 'action' => __( 'Refresh Status', 'jetpack' ), - 'status' => __( 'Status unknown.', 'jetpack' ), - ), - 'ERROR:LARGE' => array( - 'action' => __( 'Refresh Status', 'jetpack' ), - 'status' => __( 'This site is too large, please contact Jetpack support to sync.', 'jetpack' ), - ), - ) ); - - wp_enqueue_script( - 'jetpack_sync_reindex_control', - plugins_url( '_inc/jquery.jetpack-sync.js', JETPACK__PLUGIN_FILE ), - array( 'jquery' ), - JETPACK__VERSION - ); - - $template = <<<EOT - <p class="jetpack_sync_reindex_control" id="jetpack_sync_reindex_control" data-strings="%s"> - <input type="submit" class="jetpack_sync_reindex_control_action button" value="%s" disabled /> - <span class="jetpack_sync_reindex_control_status">…</span> - </p> -EOT; - - return sprintf( - $template, - esc_attr( $strings ), - esc_attr__( 'Refresh Status', 'jetpack' ) - ); - } - - private function _get_post_count_local() { - global $wpdb; - return (int) $wpdb->get_var( - "SELECT count(*) - FROM {$wpdb->posts} - WHERE post_status = 'publish' AND post_password = ''" - ); - } - - private function _get_post_count_cloud() { - $blog_id = Jetpack::init()->get_option( 'id' ); - - $body = array( - 'size' => 1, - ); - - $response = wp_remote_post( - "https://public-api.wordpress.com/rest/v1/sites/$blog_id/search", - array( - 'timeout' => 10, - 'user-agent' => 'jetpack_related_posts', - 'sslverify' => true, - 'body' => $body, - ) - ); - - if ( is_wp_error( $response ) ) { - return 0; - } - - $results = json_decode( wp_remote_retrieve_body( $response ), true ); - - return isset( $results['results'] ) && isset( $results['results']['total'] ) ? (int) $results['results']['total'] : 0; - } - - /** - * Sometimes we need to fake options to be able to sync data with .com - * This is a helper function. That will make it easier to do just that. - * - * It will make sure that the options are synced when do_action( 'jetpack_sync_all_registered_options' ); - * - * Which should happen everytime we update Jetpack to a new version or daily by Jetpack_Heartbeat. - * - * $callback is a function that is passed into a filter that returns the value of the option. - * This value should never be false. Since we want to short circuit the get_option function - * to return the value of the our callback. - * - * You can also trigger an update when a something else changes by calling the - * do_action( 'add_option_jetpack_' . $option, 'jetpack_'.$option, $callback_function ); - * on the action that should that would trigger the update. - * - * - * @param string $option Option will always be prefixed with Jetpack and be saved on .com side - * @param string or array $callback - */ - function mock_option( $option , $callback ) { - add_filter( 'pre_option_jetpack_'. $option, $callback ); - // This shouldn't happen but if it does we return the same as before. - add_filter( 'option_jetpack_'. $option, $callback ); - // Instead of passing a file we just pass in a string. - $this->options( 'mock-option' , 'jetpack_' . $option ); - - } - /** - * Sometimes you need to sync constants to .com - * Using the function will allow you to do just that. - * - * @param 'string' $constant Constants defined in code. - * - */ - function register_constant( $constant ) { - $this->register( 'constant', $constant ); - } - - function get_default_constant() { - $filter = current_filter(); - // We don't know what the constant is so we get it from the current filter. - if ( 'pre_option_jetpack_constant_' === substr( $filter, 0, 28 ) ) { - $constant = substr( $filter, 28 ); - if ( defined( $constant ) ) { - // If constant is set to false we will not shortcut the get_option function and will return the default value. - // Hance we set it to null. Which in most cases would produce the same result. - return false === constant( $constant ) ? null : constant( $constant ); - } - return $this->default_constant( $constant ); - } - } - /** - * Simular to $this->options() function. - * Add the constant to be synced to .com when we activate the module. - * As well as on heartbeat and plugin upgrade and connection to .com. - * - * @param string $file - * @param string $constant - */ - function constant( $file, $constant ) { - $constants = func_get_args(); - $file = array_shift( $constants ); - - $module_slug = Jetpack::get_module_slug( $file ); - - if ( ! isset( $this->sync_constants[ $module_slug ] ) ) { - $this->sync_constants[ $module_slug ] = array(); - } - - foreach ( $constants as $constant ) { - $this->sync_constants[ $module_slug ][] = $constant; - } - } - - /** - * Helper function to return the constants value. - * - * @param string $constant - * @return value of the constant or null if the constant is set to false or doesn't exits. - */ - static function get_constant( $constant ) { - if ( defined( $constant ) ) { - return constant( $constant ); - } - - return null; - } -} |