View file File name : class-initializer.php Content :<?php /** * WP Admin page with information and configuration shared among all Jetpack stand-alone plugins * * @package automattic/my-jetpack */ namespace Automattic\Jetpack\My_Jetpack; use Automattic\Jetpack\Admin_UI\Admin_Menu; use Automattic\Jetpack\Assets; use Automattic\Jetpack\Boost_Speed_Score\Speed_Score; use Automattic\Jetpack\Boost_Speed_Score\Speed_Score_History; use Automattic\Jetpack\Connection\Client; use Automattic\Jetpack\Connection\Initial_State as Connection_Initial_State; use Automattic\Jetpack\Connection\Manager as Connection_Manager; use Automattic\Jetpack\Connection\Rest_Authentication as Connection_Rest_Authentication; use Automattic\Jetpack\Constants as Jetpack_Constants; use Automattic\Jetpack\ExPlat; use Automattic\Jetpack\JITMS\JITM; use Automattic\Jetpack\Licensing; use Automattic\Jetpack\Modules; use Automattic\Jetpack\Plugins_Installer; use Automattic\Jetpack\Protect_Status\Status as Protect_Status; use Automattic\Jetpack\Status; use Automattic\Jetpack\Status\Host as Status_Host; use Automattic\Jetpack\Sync\Functions as Sync_Functions; use Automattic\Jetpack\Terms_Of_Service; use Automattic\Jetpack\Tracking; use Automattic\Jetpack\VideoPress\Stats as VideoPress_Stats; use Automattic\Jetpack\Waf\Waf_Runner; use Jetpack; use WP_Error; /** * The main Initializer class that registers the admin menu and eneuque the assets. */ class Initializer { /** * My Jetpack package version * * @var string */ const PACKAGE_VERSION = '4.36.0'; /** * HTML container ID for the IDC screen on My Jetpack page. */ const IDC_CONTAINER_ID = 'my-jetpack-identity-crisis-container'; const JETPACK_PLUGIN_SLUGS = array( 'jetpack-backup', 'jetpack-boost', 'zerobscrm', 'jetpack', 'jetpack-protect', 'jetpack-social', 'jetpack-videopress', 'jetpack-search', ); const MY_JETPACK_SITE_INFO_TRANSIENT_KEY = 'my-jetpack-site-info'; const UPDATE_HISTORICALLY_ACTIVE_JETPACK_MODULES_KEY = 'update-historically-active-jetpack-modules'; const MISSING_CONNECTION_NOTIFICATION_KEY = 'missing-connection'; const VIDEOPRESS_STATS_KEY = 'my-jetpack-videopress-stats'; const VIDEOPRESS_PERIOD_KEY = 'my-jetpack-videopress-period'; /** * Holds info/data about the site (from the /sites/%d endpoint) * * @var object */ public static $site_info; /** * Initialize My Jetpack * * @return void */ public static function init() { if ( ! self::should_initialize() || did_action( 'my_jetpack_init' ) ) { return; } // Extend jetpack plugins action links. Products::extend_plugins_action_links(); // Set up the REST authentication hooks. Connection_Rest_Authentication::init(); if ( self::is_licensing_ui_enabled() ) { Licensing::instance()->initialize(); } // Initialize Boost Speed Score new Speed_Score( array(), 'jetpack-my-jetpack' ); // Add custom WP REST API endoints. add_action( 'rest_api_init', array( __CLASS__, 'register_rest_endpoints' ) ); add_action( 'admin_menu', array( __CLASS__, 'add_my_jetpack_menu_item' ) ); add_action( 'admin_init', array( __CLASS__, 'setup_historically_active_jetpack_modules_sync' ) ); // This is later than the admin-ui package, which runs on 1000 add_action( 'admin_init', array( __CLASS__, 'maybe_show_red_bubble' ), 1001 ); // Set up the ExPlat package endpoints ExPlat::init(); // Sets up JITMS. JITM::configure(); // Add "Activity Log" menu item. Activitylog::init(); // Add "Jetpack Manage" menu item. Jetpack_Manage::init(); /** * Fires after the My Jetpack package is initialized * * @since 0.1.0 */ do_action( 'my_jetpack_init' ); } /** * Acts as a feature flag, returning a boolean for whether we should show the licensing UI. * * @since 1.2.0 * * @return boolean */ public static function is_licensing_ui_enabled() { // Default changed to true in 1.5.0. $is_enabled = true; /* * Bail if My Jetpack is not enabled, * and thus the licensing UI shouldn't be enabled either. */ if ( ! self::should_initialize() ) { $is_enabled = false; } /** * Acts as a feature flag, returning a boolean for whether we should show the licensing UI. * * @param bool $is_enabled Defaults to true. * * @since 1.2.0 * @since 1.5.0 Update default value to true. */ return apply_filters( 'jetpack_my_jetpack_should_enable_add_license_screen', $is_enabled ); } /** * Add My Jetpack menu item to the admin menu. * * @return void */ public static function add_my_jetpack_menu_item() { $page_suffix = Admin_Menu::add_menu( __( 'My Jetpack', 'jetpack-my-jetpack' ), __( 'My Jetpack', 'jetpack-my-jetpack' ), 'edit_posts', 'my-jetpack', array( __CLASS__, 'admin_page' ), -1 ); add_action( 'load-' . $page_suffix, array( __CLASS__, 'admin_init' ) ); } /** * Callback for the load my jetpack page hook. * * @return void */ public static function admin_init() { self::$site_info = self::get_site_info(); add_filter( 'identity_crisis_container_id', array( static::class, 'get_idc_container_id' ) ); add_action( 'admin_enqueue_scripts', array( __CLASS__, 'enqueue_scripts' ) ); // Product statuses are constantly changing, so we never want to cache the page. header( 'Cache-Control: no-cache, no-store, must-revalidate' ); header( 'Pragma: no-cache' ); header( 'Expires: 0' ); } /** * Returns whether we are in condition to track to use * Analytics functionality like Tracks, MC, or GA. */ public static function can_use_analytics() { $status = new Status(); $connection = new Connection_Manager(); $tracking = new Tracking( 'jetpack', $connection ); return $tracking->should_enable_tracking( new Terms_Of_Service(), $status ); } /** * Enqueue admin page assets. * * @return void */ public static function enqueue_scripts() { /** * Fires after the My Jetpack page is initialized. * Allows for enqueuing additional scripts only on the My Jetpack page. * * @since 4.35.7 */ do_action( 'myjetpack_enqueue_scripts' ); Assets::register_script( 'my_jetpack_main_app', '../build/index.js', __FILE__, array( 'enqueue' => true, 'in_footer' => true, 'textdomain' => 'jetpack-my-jetpack', ) ); $modules = new Modules(); $connection = new Connection_Manager(); $speed_score_history = new Speed_Score_History( get_site_url() ); $latest_score = $speed_score_history->latest(); $previous_score = array(); if ( $speed_score_history->count() > 1 ) { $previous_score = $speed_score_history->latest( 1 ); } $latest_score['previousScores'] = $previous_score['scores'] ?? array(); $scan_data = Protect_Status::get_status(); self::update_historically_active_jetpack_modules(); $waf_config = array(); if ( class_exists( 'Automattic\Jetpack\Waf\Waf_Runner' ) ) { $waf_config = Waf_Runner::get_config(); } wp_localize_script( 'my_jetpack_main_app', 'myJetpackInitialState', array( 'products' => array( 'items' => Products::get_products(), ), 'purchases' => array( 'items' => array(), ), 'plugins' => Plugins_Installer::get_plugins(), 'themes' => Sync_Functions::get_themes(), 'myJetpackUrl' => admin_url( 'admin.php?page=my-jetpack' ), 'myJetpackCheckoutUri' => admin_url( 'admin.php?page=my-jetpack' ), 'topJetpackMenuItemUrl' => Admin_Menu::get_top_level_menu_item_url(), 'siteSuffix' => ( new Status() )->get_site_suffix(), 'siteUrl' => esc_url( get_site_url() ), 'blogID' => Connection_Manager::get_site_id( true ), 'myJetpackVersion' => self::PACKAGE_VERSION, 'myJetpackFlags' => self::get_my_jetpack_flags(), 'fileSystemWriteAccess' => self::has_file_system_write_access(), 'loadAddLicenseScreen' => self::is_licensing_ui_enabled(), 'adminUrl' => esc_url( admin_url() ), 'IDCContainerID' => static::get_idc_container_id(), 'userIsAdmin' => current_user_can( 'manage_options' ), 'userIsNewToJetpack' => self::is_jetpack_user_new(), 'lifecycleStats' => array( 'jetpackPlugins' => self::get_installed_jetpack_plugins(), 'historicallyActiveModules' => \Jetpack_Options::get_option( 'historically_active_modules', array() ), 'ownedProducts' => Products::get_products_by_ownership( 'owned' ), 'unownedProducts' => Products::get_products_by_ownership( 'unowned' ), 'brokenModules' => self::check_for_broken_modules(), 'isSiteConnected' => $connection->is_connected(), 'isUserConnected' => $connection->is_user_connected(), 'purchases' => self::get_purchases(), 'modules' => self::get_active_modules(), ), 'redBubbleAlerts' => self::get_red_bubble_alerts(), 'recommendedModules' => array( 'modules' => self::get_recommended_modules(), 'isFirstRun' => \Jetpack_Options::get_option( 'recommendations_first_run', true ), 'dismissed' => \Jetpack_Options::get_option( 'dismissed_recommendations', false ), ), 'isStatsModuleActive' => $modules->is_active( 'stats' ), 'isUserFromKnownHost' => self::is_user_from_known_host(), 'isCommercial' => self::is_commercial_site(), 'isAtomic' => ( new Status_Host() )->is_woa_site(), 'jetpackManage' => array( 'isEnabled' => Jetpack_Manage::could_use_jp_manage(), 'isAgencyAccount' => Jetpack_Manage::is_agency_account(), ), 'latestBoostSpeedScores' => $latest_score, 'protect' => array( 'scanData' => $scan_data, 'wafConfig' => array_merge( $waf_config, array( 'blocked_logins' => (int) get_site_option( 'jetpack_protect_blocked_attempts', 0 ) ) ), ), 'videopress' => self::get_videopress_stats(), ) ); wp_localize_script( 'my_jetpack_main_app', 'myJetpackRest', array( 'apiRoot' => esc_url_raw( rest_url() ), 'apiNonce' => wp_create_nonce( 'wp_rest' ), ) ); // Connection Initial State. Connection_Initial_State::render_script( 'my_jetpack_main_app' ); // Required for Analytics. if ( self::can_use_analytics() ) { Tracking::register_tracks_functions_scripts( true ); } } /** * Get stats for VideoPress * * @return array|WP_Error */ public static function get_videopress_stats() { $video_count = array_sum( (array) wp_count_attachments( 'video' ) ); if ( ! class_exists( 'Automattic\Jetpack\VideoPress\Stats' ) ) { return array( 'videoCount' => $video_count, ); } $featured_stats = get_transient( self::VIDEOPRESS_STATS_KEY ); if ( $featured_stats ) { return array( 'featuredStats' => $featured_stats, 'videoCount' => $video_count, ); } $stats_period = get_transient( self::VIDEOPRESS_PERIOD_KEY ); $videopress_stats = new VideoPress_Stats(); // If the stats period exists, retrieve that information without checking the view count. // If it does not, check the view count of monthly stats and determine if we want to show yearly or monthly stats. if ( $stats_period ) { if ( $stats_period === 'day' ) { $featured_stats = $videopress_stats->get_featured_stats( 60, 'day' ); } else { $featured_stats = $videopress_stats->get_featured_stats( 2, 'year' ); } } else { $featured_stats = $videopress_stats->get_featured_stats( 60, 'day' ); if ( ! is_wp_error( $featured_stats ) && $featured_stats && ( $featured_stats['data']['views']['current'] < 500 || $featured_stats['data']['views']['previous'] < 500 ) ) { $featured_stats = $videopress_stats->get_featured_stats( 2, 'year' ); } } if ( is_wp_error( $featured_stats ) || ! $featured_stats ) { return array( 'videoCount' => $video_count, ); } set_transient( self::VIDEOPRESS_PERIOD_KEY, $featured_stats['period'], WEEK_IN_SECONDS ); set_transient( self::VIDEOPRESS_STATS_KEY, $featured_stats, DAY_IN_SECONDS ); return array( 'featuredStats' => $featured_stats, 'videoCount' => $video_count, ); } /** * Get product slugs of the active purchases * * @return array */ public static function get_purchases() { $purchases = Wpcom_Products::get_site_current_purchases(); if ( is_wp_error( $purchases ) ) { return array(); } return array_map( function ( $purchase ) { return $purchase->product_slug; }, (array) $purchases ); } /** * Get installed Jetpack plugins * * @return array */ public static function get_installed_jetpack_plugins() { $plugin_slugs = array_keys( Plugins_Installer::get_plugins() ); $plugin_slugs = array_map( static function ( $slug ) { $parts = explode( '/', $slug ); if ( empty( $parts ) ) { return ''; } // Return the last segment of the filepath without the PHP extension return str_replace( '.php', '', $parts[ count( $parts ) - 1 ] ); }, $plugin_slugs ); return array_values( array_intersect( self::JETPACK_PLUGIN_SLUGS, $plugin_slugs ) ); } /** * Get active modules (except ones enabled by default) * * @return array */ public static function get_active_modules() { $modules = new Modules(); $active_modules = $modules->get_active(); // if the Jetpack plugin is active, filter out the modules that are active by default if ( class_exists( 'Jetpack' ) && ! empty( $active_modules ) ) { $active_modules = array_diff( $active_modules, Jetpack::get_default_modules() ); } return array_values( $active_modules ); } /** * Determine if the current user is "new" to Jetpack * This is used to vary some messaging in My Jetpack * * On the front-end, purchases are also taken into account * * @return bool */ public static function is_jetpack_user_new() { // is the user connected? $connection = new Connection_Manager(); if ( $connection->is_user_connected() ) { return false; } // TODO: add a data point for the last known connection/ disconnection time // are any modules active? $active_modules = self::get_active_modules(); if ( ! empty( $active_modules ) ) { return false; } // check for other Jetpack plugins that are installed on the site (active or not) // If there's more than one Jetpack plugin active, this user is not "new" $plugin_slugs = array_keys( Plugins_Installer::get_plugins() ); $plugin_slugs = array_map( static function ( $slug ) { $parts = explode( '/', $slug ); if ( empty( $parts ) ) { return ''; } // Return the last segment of the filepath without the PHP extension return str_replace( '.php', '', $parts[ count( $parts ) - 1 ] ); }, $plugin_slugs ); $installed_jetpack_plugins = array_intersect( self::JETPACK_PLUGIN_SLUGS, $plugin_slugs ); if ( is_countable( $installed_jetpack_plugins ) && count( $installed_jetpack_plugins ) >= 2 ) { return false; } // Does the site have any purchases? $purchases = Wpcom_Products::get_site_current_purchases(); if ( ! empty( $purchases ) && ! is_wp_error( $purchases ) ) { return false; } return true; } /** * Determines whether the user has come from a host we can recognize. * * @return string */ public static function is_user_from_known_host() { // Known (external) host is the one that has been determined and is not dotcom. return ! in_array( ( new Status_Host() )->get_known_host_guess(), array( 'unknown', 'wpcom' ), true ); } /** * Build flags for My Jetpack UI * * @return array */ public static function get_my_jetpack_flags() { $flags = array( 'videoPressStats' => Jetpack_Constants::is_true( 'JETPACK_MY_JETPACK_VIDEOPRESS_STATS_ENABLED' ), 'showFullJetpackStatsCard' => class_exists( 'Jetpack' ), ); return $flags; } /** * Echoes the admin page content. * * @return void */ public static function admin_page() { echo '<div id="my-jetpack-container"></div>'; } /** * Register the REST API routes. * * @return void */ public static function register_rest_endpoints() { new REST_Products(); new REST_Purchases(); new REST_Zendesk_Chat(); new REST_Product_Data(); new REST_AI(); new REST_Recommendations_Evaluation(); register_rest_route( 'my-jetpack/v1', 'site', array( 'methods' => \WP_REST_Server::READABLE, 'callback' => __CLASS__ . '::get_site', 'permission_callback' => __CLASS__ . '::permissions_callback', ) ); register_rest_route( 'my-jetpack/v1', 'site/dismiss-welcome-banner', array( 'methods' => \WP_REST_Server::EDITABLE, 'callback' => __CLASS__ . '::dismiss_welcome_banner', 'permission_callback' => __CLASS__ . '::permissions_callback', ) ); } /** * Check user capability to access the endpoint. * * @access public * @static * * @return true|WP_Error */ public static function permissions_callback() { return current_user_can( 'manage_options' ); } /** * Return true if we should initialize the My Jetpack admin page. */ public static function should_initialize() { $should = true; if ( is_multisite() ) { $should = false; } // All options presented in My Jetpack require a connection to WordPress.com. if ( ( new Status() )->is_offline_mode() ) { $should = false; } /** * Allows filtering whether My Jetpack should be initialized. * * @since 0.5.0-alpha * * @param bool $shoud_initialize Should we initialize My Jetpack? */ return apply_filters( 'jetpack_my_jetpack_should_initialize', $should ); } /** * Set transient to queue an update to the historically active Jetpack modules on the next wp-admin load * * @param string $plugin The plugin that triggered the update. This will be present if the function was queued by a plugin activation. * * @return void */ public static function queue_historically_active_jetpack_modules_update( $plugin = null ) { $plugin_filenames = Products::get_all_plugin_filenames(); if ( ! $plugin || in_array( $plugin, $plugin_filenames, true ) ) { set_transient( self::UPDATE_HISTORICALLY_ACTIVE_JETPACK_MODULES_KEY, true ); } } /** * Hook into several connection-based actions to update the historically active Jetpack modules * If the transient that indicates the list needs to be synced, update it and delete the transient * * @return void */ public static function setup_historically_active_jetpack_modules_sync() { if ( get_transient( self::UPDATE_HISTORICALLY_ACTIVE_JETPACK_MODULES_KEY ) && ! wp_doing_ajax() ) { self::update_historically_active_jetpack_modules(); delete_transient( self::UPDATE_HISTORICALLY_ACTIVE_JETPACK_MODULES_KEY ); } $actions = array( 'jetpack_site_registered', 'jetpack_user_authorized', 'activated_plugin', ); foreach ( $actions as $action ) { add_action( $action, array( __CLASS__, 'queue_historically_active_jetpack_modules_update' ), 5 ); } // Modules are often updated async, so we need to update them right away as there will sometimes be no page reload. add_action( 'jetpack_activate_module', array( __CLASS__, 'update_historically_active_jetpack_modules' ), 5 ); } /** * Update historically active Jetpack plugins * Historically active is defined as the Jetpack plugins that are installed and active with the required connections * This array will consist of any plugins that were active at one point in time and are still enabled on the site * * @return void */ public static function update_historically_active_jetpack_modules() { $historically_active_modules = \Jetpack_Options::get_option( 'historically_active_modules', array() ); $products = Products::get_products(); foreach ( $products as $product ) { $status = $product['status']; $product_slug = $product['slug']; // We want to leave modules in the array if they've been active in the past // and were not manually disabled by the user. if ( in_array( $status, Products::$broken_module_statuses, true ) ) { continue; } // If the module is active and not already in the array, add it if ( in_array( $status, Products::$active_module_statuses, true ) && ! in_array( $product_slug, $historically_active_modules, true ) ) { $historically_active_modules[] = $product_slug; } // If the module has been disabled due to a manual user action, // or because of a missing plan error, remove it from the array if ( in_array( $status, Products::$disabled_module_statuses, true ) ) { $historically_active_modules = array_values( array_diff( $historically_active_modules, array( $product_slug ) ) ); } } \Jetpack_Options::update_option( 'historically_active_modules', array_unique( $historically_active_modules ) ); } /** * Site full-data endpoint. * * @return object Site data. */ public static function get_site() { $site_id = \Jetpack_Options::get_option( 'id' ); $wpcom_endpoint = sprintf( '/sites/%d?force=wpcom', $site_id ); $wpcom_api_version = '1.1'; $response = Client::wpcom_json_api_request_as_blog( $wpcom_endpoint, $wpcom_api_version ); $response_code = wp_remote_retrieve_response_code( $response ); $body = json_decode( wp_remote_retrieve_body( $response ) ); if ( is_wp_error( $response ) || empty( $response['body'] ) ) { return new WP_Error( 'site_data_fetch_failed', 'Site data fetch failed', array( 'status' => $response_code ) ); } return rest_ensure_response( $body ); } /** * Populates the self::$site_info var with site data from the /sites/%d endpoint * * @return object|WP_Error */ public static function get_site_info() { static $site_info = null; if ( $site_info !== null ) { return $site_info; } // Check for a cached value before doing lookup $stored_site_info = get_transient( self::MY_JETPACK_SITE_INFO_TRANSIENT_KEY ); if ( $stored_site_info !== false ) { return $stored_site_info; } $response = self::get_site(); if ( is_wp_error( $response ) ) { return $response; } $site_info = $response->data; set_transient( self::MY_JETPACK_SITE_INFO_TRANSIENT_KEY, $site_info, DAY_IN_SECONDS ); return $site_info; } /** * Returns whether a site has been determined "commercial" or not. * * @return bool|null */ public static function is_commercial_site() { if ( is_wp_error( self::$site_info ) ) { return null; } return empty( self::$site_info->options->is_commercial ) ? false : self::$site_info->options->is_commercial; } /** * Check if site is registered (has been connected before). * * @return bool */ public static function is_registered() { return (bool) \Jetpack_Options::get_option( 'id' ); } /** * Dismiss the welcome banner. * * @return \WP_REST_Response */ public static function dismiss_welcome_banner() { \Jetpack_Options::update_option( 'dismissed_welcome_banner', true ); return rest_ensure_response( array( 'success' => true ) ); } /** * Returns true if the site has file write access to the plugins folder, false otherwise. * * @return string **/ public static function has_file_system_write_access() { $cache = get_transient( 'my_jetpack_write_access' ); if ( false !== $cache ) { return $cache; } if ( ! function_exists( 'get_filesystem_method' ) ) { require_once ABSPATH . 'wp-admin/includes/file.php'; } require_once ABSPATH . 'wp-admin/includes/template.php'; $write_access = 'no'; $filesystem_method = get_filesystem_method( array(), WP_PLUGIN_DIR ); if ( 'direct' === $filesystem_method ) { $write_access = 'yes'; } if ( ! $write_access ) { ob_start(); $filesystem_credentials_are_stored = request_filesystem_credentials( self_admin_url() ); ob_end_clean(); if ( $filesystem_credentials_are_stored ) { $write_access = 'yes'; } } set_transient( 'my_jetpack_write_access', $write_access, 30 * MINUTE_IN_SECONDS ); return $write_access; } /** * Get container IDC for the IDC screen. * * @return string */ public static function get_idc_container_id() { return static::IDC_CONTAINER_ID; } /** * Conditionally append the red bubble notification to the "Jetpack" menu item if there are alerts to show * * @return void */ public static function maybe_show_red_bubble() { global $menu; // filters for the items in this file add_filter( 'my_jetpack_red_bubble_notification_slugs', array( __CLASS__, 'add_red_bubble_alerts' ) ); $red_bubble_alerts = array_filter( self::get_red_bubble_alerts(), function ( $alert ) { // We don't want to show silent alerts return empty( $alert['is_silent'] ); } ); // The Jetpack menu item should be on index 3 if ( ! empty( $red_bubble_alerts ) && is_countable( $red_bubble_alerts ) && isset( $menu[3] ) && $menu[3][0] === 'Jetpack' ) { // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited $menu[3][0] .= sprintf( ' <span class="awaiting-mod">%d</span>', count( $red_bubble_alerts ) ); } } /** * Collect all possible alerts that we might use a red bubble notification for * * @return array */ public static function get_red_bubble_alerts() { static $red_bubble_alerts = array(); // using a static cache since we call this function more than once in the class if ( ! empty( $red_bubble_alerts ) ) { return $red_bubble_alerts; } // go find the alerts $red_bubble_alerts = apply_filters( 'my_jetpack_red_bubble_notification_slugs', $red_bubble_alerts ); return $red_bubble_alerts; } /** * Get list of module names sorted by their recommendation score * * @return array|null */ public static function get_recommended_modules() { $recommendations_evaluation = \Jetpack_Options::get_option( 'recommendations_evaluation', null ); if ( ! $recommendations_evaluation ) { return null; } arsort( $recommendations_evaluation ); // Sort by scores in descending order return array_keys( $recommendations_evaluation ); // Get only module names } /** * Check for features broken by a disconnected user or site * * @return array */ public static function check_for_broken_modules() { $connection = new Connection_Manager(); $is_user_connected = $connection->is_user_connected() || $connection->has_connected_owner(); $is_site_connected = $connection->is_connected(); $broken_modules = array( 'needs_site_connection' => array(), 'needs_user_connection' => array(), ); if ( $is_user_connected && $is_site_connected ) { return $broken_modules; } $products = Products::get_products_classes(); $historically_active_modules = \Jetpack_Options::get_option( 'historically_active_modules', array() ); foreach ( $products as $product ) { if ( ! in_array( $product::$slug, $historically_active_modules, true ) ) { continue; } if ( $product::$requires_user_connection && ! $is_user_connected ) { if ( ! in_array( $product::$slug, $broken_modules['needs_user_connection'], true ) ) { $broken_modules['needs_user_connection'][] = $product::$slug; } } elseif ( ! $is_site_connected ) { if ( ! in_array( $product::$slug, $broken_modules['needs_site_connection'], true ) ) { $broken_modules['needs_site_connection'][] = $product::$slug; } } } return $broken_modules; } /** * Add relevant red bubble notifications * * @param array $red_bubble_slugs - slugs that describe the reasons the red bubble is showing. * @return array */ public static function add_red_bubble_alerts( array $red_bubble_slugs ) { if ( wp_doing_ajax() ) { return array(); } $connection = new Connection_Manager(); $welcome_banner_dismissed = \Jetpack_Options::get_option( 'dismissed_welcome_banner', false ); if ( self::is_jetpack_user_new() && ! $welcome_banner_dismissed ) { $red_bubble_slugs['welcome-banner-active'] = array( 'is_silent' => $connection->is_connected(), // we don't display the red bubble if the user is connected ); return $red_bubble_slugs; } else { return self::alert_if_missing_connection( $red_bubble_slugs ); } } /** * Add an alert slug if the site is missing a site connection * * @param array $red_bubble_slugs - slugs that describe the reasons the red bubble is showing. * @return array */ public static function alert_if_missing_connection( array $red_bubble_slugs ) { $broken_modules = self::check_for_broken_modules(); $connection = new Connection_Manager(); // Checking for site connection issues first. if ( ! empty( $broken_modules['needs_site_connection'] ) ) { $red_bubble_slugs[ self::MISSING_CONNECTION_NOTIFICATION_KEY ] = array( 'type' => 'site', 'is_error' => true, ); return $red_bubble_slugs; } if ( ! empty( $broken_modules['needs_user_connection'] ) ) { $red_bubble_slugs[ self::MISSING_CONNECTION_NOTIFICATION_KEY ] = array( 'type' => 'user', 'is_error' => true, ); return $red_bubble_slugs; } if ( ! $connection->is_connected() ) { $red_bubble_slugs[ self::MISSING_CONNECTION_NOTIFICATION_KEY ] = array( 'type' => 'site', 'is_error' => false, ); return $red_bubble_slugs; } return $red_bubble_slugs; } }