259 lines
8.1 KiB
PHP
259 lines
8.1 KiB
PHP
<?php
|
|
/**
|
|
* Class to handle the Protect Status of Jetpack Protect
|
|
*
|
|
* @package automattic/jetpack-protect-plugin
|
|
*/
|
|
|
|
namespace Automattic\Jetpack\Protect;
|
|
|
|
use Automattic\Jetpack\Connection\Client;
|
|
use Automattic\Jetpack\Connection\Manager as Connection_Manager;
|
|
use Automattic\Jetpack\Plugins_Installer;
|
|
use Automattic\Jetpack\Redirect;
|
|
use Automattic\Jetpack\Sync\Functions as Sync_Functions;
|
|
use Jetpack_Options;
|
|
use WP_Error;
|
|
|
|
/**
|
|
* Class that handles fetching and caching the Status of vulnerabilities check from the WPCOM servers
|
|
*/
|
|
class Protect_Status extends Status {
|
|
|
|
/**
|
|
* WPCOM endpoint
|
|
*
|
|
* @var string
|
|
*/
|
|
const REST_API_BASE = '/sites/%d/jetpack-protect-status';
|
|
|
|
/**
|
|
* Name of the option where status is stored
|
|
*
|
|
* @var string
|
|
*/
|
|
const OPTION_NAME = 'jetpack_protect_status';
|
|
|
|
/**
|
|
* Name of the option where the timestamp of the status is stored
|
|
*
|
|
* @var string
|
|
*/
|
|
const OPTION_TIMESTAMP_NAME = 'jetpack_protect_status_time';
|
|
|
|
/**
|
|
* Gets the current status of the Jetpack Protect checks
|
|
*
|
|
* @param bool $refresh_from_wpcom Refresh the local plan and status cache from wpcom.
|
|
* @return Status_Model
|
|
*/
|
|
public static function get_status( $refresh_from_wpcom = false ) {
|
|
if ( self::$status !== null ) {
|
|
return self::$status;
|
|
}
|
|
|
|
if ( $refresh_from_wpcom || ! self::should_use_cache() || self::is_cache_expired() ) {
|
|
$status = self::fetch_from_server();
|
|
} else {
|
|
$status = self::get_from_options();
|
|
}
|
|
|
|
if ( is_wp_error( $status ) ) {
|
|
$status = new Status_Model(
|
|
array(
|
|
'error' => 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;
|
|
}
|
|
}
|