true, 'error_code' => $status->get_error_code(), 'error_message' => $status->get_error_message(), ) ); } else { $status = self::normalize_protect_report_data( $status ); } self::$status = $status; return $status; } /** * Gets the WPCOM API endpoint * * @return WP_Error|string */ public static function get_api_url() { $blog_id = Jetpack_Options::get_option( 'id' ); $is_connected = ( new Connection_Manager() )->is_connected(); if ( ! $blog_id || ! $is_connected ) { return new WP_Error( 'site_not_connected' ); } $api_url = sprintf( self::REST_API_BASE, $blog_id ); return $api_url; } /** * Fetches the status from WPCOM servers * * @return WP_Error|array */ public static function fetch_from_server() { $api_url = self::get_api_url(); if ( is_wp_error( $api_url ) ) { return $api_url; } $response = Client::wpcom_json_api_request_as_blog( self::get_api_url(), '2', array( 'method' => 'GET' ), null, 'wpcom' ); $response_code = wp_remote_retrieve_response_code( $response ); if ( is_wp_error( $response ) || 200 !== $response_code || empty( $response['body'] ) ) { return new WP_Error( 'failed_fetching_status', 'Failed to fetch Protect Status data from server', array( 'status' => $response_code ) ); } $body = json_decode( wp_remote_retrieve_body( $response ) ); self::update_status_option( $body ); return $body; } /** * Normalize data from the Protect Report data source. * * @param object $report_data Data from the Protect Report. * @return Status_Model */ protected static function normalize_protect_report_data( $report_data ) { $status = new Status_Model(); $status->data_source = 'protect_report'; // map report data properties directly into the Status_Model $status->status = isset( $report_data->status ) ? $report_data->status : null; $status->last_checked = isset( $report_data->last_checked ) ? $report_data->last_checked : null; $status->num_threats = isset( $report_data->num_vulnerabilities ) ? $report_data->num_vulnerabilities : null; $status->num_themes_threats = isset( $report_data->num_themes_vulnerabilities ) ? $report_data->num_themes_vulnerabilities : null; $status->num_plugins_threats = isset( $report_data->num_plugins_vulnerabilities ) ? $report_data->num_plugins_vulnerabilities : null; // merge plugins from report with all installed plugins before mapping into the Status_Model $installed_plugins = Plugins_Installer::get_plugins(); $last_report_plugins = isset( $report_data->plugins ) ? $report_data->plugins : new \stdClass(); $status->plugins = self::merge_installed_and_checked_lists( $installed_plugins, $last_report_plugins, array( 'type' => 'plugins' ) ); // merge themes from report with all installed plugins before mapping into the Status_Model $installed_themes = Sync_Functions::get_themes(); $last_report_themes = isset( $report_data->themes ) ? $report_data->themes : new \stdClass(); $status->themes = self::merge_installed_and_checked_lists( $installed_themes, $last_report_themes, array( 'type' => 'themes' ) ); // normalize WordPress core report data and map into Status_Model $status->core = self::normalize_core_information( isset( $report_data->core ) ? $report_data->core : new \stdClass() ); // check if any installed items (themes, plugins, or core) have not been checked in the report $all_items = array_merge( $status->plugins, $status->themes, array( $status->core ) ); $unchecked_items = array_filter( $all_items, function ( $item ) { return ! isset( $item->checked ) || ! $item->checked; } ); $status->has_unchecked_items = ! empty( $unchecked_items ); return $status; } /** * Merges the list of installed extensions with the list of extensions that were checked for known vulnerabilities and return a normalized list to be used in the UI * * @param array $installed The list of installed extensions, where each attribute key is the extension slug. * @param object $checked The list of checked extensions. * @param array $append Additional data to append to each result in the list. * @return array Normalized list of extensions. */ protected static function merge_installed_and_checked_lists( $installed, $checked, $append ) { $new_list = array(); foreach ( array_keys( $installed ) as $slug ) { $checked = (object) $checked; $extension = new Extension_Model( array_merge( array( 'name' => $installed[ $slug ]['Name'], 'version' => $installed[ $slug ]['Version'], 'slug' => $slug, 'threats' => array(), 'checked' => false, ), $append ) ); if ( isset( $checked->{ $slug } ) && $checked->{ $slug }->version === $installed[ $slug ]['Version'] ) { $extension->version = $checked->{ $slug }->version; $extension->checked = true; if ( is_array( $checked->{ $slug }->vulnerabilities ) ) { foreach ( $checked->{ $slug }->vulnerabilities as $threat ) { $extension->threats[] = new Threat_Model( array( 'id' => $threat->id, 'title' => $threat->title, 'fixed_in' => $threat->fixed_in, 'description' => isset( $threat->description ) ? $threat->description : null, 'source' => isset( $threat->id ) ? Redirect::get_url( 'jetpack-protect-vul-info', array( 'path' => $threat->id ) ) : null, ) ); } } } $new_list[] = $extension; } $new_list = parent::sort_threats( $new_list ); return $new_list; } /** * Check if the WordPress version that was checked matches the current installed version. * * @param object $core_check The object returned by Protect wpcom endpoint. * @return object The object representing the current status of core checks. */ protected static function normalize_core_information( $core_check ) { global $wp_version; $core = new Extension_Model( array( 'type' => 'core', 'name' => 'WordPress', 'version' => $wp_version, 'checked' => false, ) ); if ( isset( $core_check->version ) && $core_check->version === $wp_version ) { if ( is_array( $core_check->vulnerabilities ) ) { $core->checked = true; $core->set_threats( array_map( function ( $vulnerability ) { $vulnerability->source = isset( $vulnerability->id ) ? Redirect::get_url( 'jetpack-protect-vul-info', array( 'path' => $vulnerability->id ) ) : null; return $vulnerability; }, $core_check->vulnerabilities ) ); } } return $core; } }