oont-contents/plugins/jetpack/jetpack_vendor/automattic/jetpack-stats-admin/src/class-rest-controller.php
2025-02-10 13:57:45 +01:00

1256 lines
34 KiB
PHP

<?php
/**
* The Stats Rest Controller class.
* Registers the REST routes for Odyssey Stats.
*
* @package automattic/jetpack-stats-admin
*/
namespace Automattic\Jetpack\Stats_Admin;
use Automattic\Jetpack\Constants;
use Automattic\Jetpack\Stats\WPCOM_Stats;
use Jetpack_Options;
use WP_Error;
use WP_REST_Request;
use WP_REST_Server;
/**
* Registers the REST routes for Stats.
* It bascially forwards the requests to the WordPress.com REST API.
*/
class REST_Controller {
const JETPACK_STATS_DASHBOARD_MODULES_CACHE_KEY = 'jetpack_stats_dashboard_modules_cache_key';
const JETPACK_STATS_DASHBOARD_MODULE_SETTINGS_CACHE_KEY = 'jetpack_stats_dashboard_module_settings_cache_key';
/**
* Namespace for the REST API.
*
* @var string
*/
public static $namespace = 'jetpack/v4/stats-app';
/**
* Hold an instance of WPCOM_Stats.
*
* @var WPCOM_Stats
*/
protected $wpcom_stats;
/**
* Constructor
*/
public function __construct() {
$this->wpcom_stats = new WPCOM_Stats();
}
/**
* Registers the REST routes for Odyssey Stats.
*
* Odyssey Stats is built from `wp-calypso`, which leverages the `public-api.wordpress.com` API.
* The current Site ID is added as part of the route, so that the front end doesn't have to handle the differences.
*
* @access public
* @static
*/
public function register_rest_routes() {
// Stats for single resource type.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/(?P<resource>[\-\w]+)/(?P<resource_id>[\d]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_single_resource_stats' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Stats for a resource type.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/(?P<resource>[\-\w]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_stats_resource' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Single post info.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/posts/(?P<resource_id>[\d]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_single_post' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Single post likes.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/posts/(?P<resource_id>[\d]+)/likes', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_single_post_likes' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// General stats for the site.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_site_stats' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Whether site has never published post / page.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/site-has-never-published-post', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'site_has_never_published_post' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// List posts.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/posts', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_site_posts' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Subscribers counts.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/subscribers/counts', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_site_subscribers_counts' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Stats Plan Usage.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats/usage', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_site_plan_usage' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// User feedback endpoint.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats/user-feedback', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::CREATABLE,
'callback' => array( $this, 'post_user_feedback' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// WordAds Earnings.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/wordads/earnings', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_wordads_earnings' ),
'permission_callback' => array( $this, 'can_user_view_wordads_stats_callback' ),
)
);
// WordAds Stats.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/wordads/stats', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_wordads_stats' ),
'permission_callback' => array( $this, 'can_user_view_wordads_stats_callback' ),
)
);
// Legacy: Update Stats notices.
// TODO: remove this in the next release.
register_rest_route(
static::$namespace,
'/stats/notices',
array(
'methods' => WP_REST_Server::EDITABLE,
'callback' => array( $this, 'update_notice_status' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
'args' => array(
'id' => array(
'required' => true,
'type' => 'string',
'description' => 'ID of the notice',
),
'status' => array(
'required' => true,
'type' => 'string',
'description' => 'Status of the notice',
),
'postponed_for' => array(
'type' => 'number',
'default' => null,
'description' => 'Postponed for (in seconds)',
'minimum' => 0,
),
),
)
);
// Update Stats notices.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats-dashboard/notices', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::EDITABLE,
'callback' => array( $this, 'update_notice_status' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
'args' => array(
'id' => array(
'required' => true,
'type' => 'string',
'description' => 'ID of the notice',
),
'status' => array(
'required' => true,
'type' => 'string',
'description' => 'Status of the notice',
),
'postponed_for' => array(
'type' => 'number',
'default' => null,
'description' => 'Postponed for (in seconds)',
'minimum' => 0,
),
),
)
);
// Get Stats notices.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats-dashboard/notices', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_notice_status' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Mark referrer spam.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/referrers/spam/new', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::EDITABLE,
'callback' => array( $this, 'mark_referrer_spam' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
'args' => array(
'domain' => array(
'required' => true,
'type' => 'string',
'description' => 'Domain of the referrer',
),
),
)
);
// Unmark referrer spam.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/referrers/spam/delete', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::EDITABLE,
'callback' => array( $this, 'unmark_referrer_spam' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
'args' => array(
'domain' => array(
'required' => true,
'type' => 'string',
'description' => 'Domain of the referrer',
),
),
)
);
// Update dashboard modules.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats-dashboard/modules', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::EDITABLE,
'callback' => array( $this, 'update_dashboard_modules' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get dashboard modules.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats-dashboard/modules', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_dashboard_modules' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Update dashboard module settings.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats-dashboard/module-settings', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::EDITABLE,
'callback' => array( $this, 'update_dashboard_module_settings' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get dashboard module settings.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/jetpack-stats-dashboard/module-settings', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_dashboard_module_settings' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get email stats as a list.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/emails/(?P<resource>[\-\w\d]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_email_stats_list' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get Email opens stats for a single post.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/opens/emails/(?P<post_id>[\d]+)/(?P<resource>[\-\w]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_email_opens_stats_single' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get Email clicks stats for a single post.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/clicks/emails/(?P<post_id>[\d]+)/(?P<resource>[\-\w]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_email_clicks_stats_single' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get Email stats time series.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/(?P<resource>[\-\w]+)/emails/(?P<post_id>[\d]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_email_stats_time_series' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get UTM stats time series.
register_rest_route(
static::$namespace,
// /stats/utm/utm_campaign,utm_source,utm_medium
sprintf( '/sites/%d/stats/utm/(?P<utm_params>[_,\-\w]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_utm_stats_time_series' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get Devices stats time series.
register_rest_route(
static::$namespace,
// /stats/devices/screensize
sprintf( '/sites/%d/stats/devices/(?P<device_property>[\w]+)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_devices_stats_time_series' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Rerun commercial classificiation.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/commercial-classification', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::EDITABLE,
'callback' => array( $this, 'run_commercial_classification' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Purchases endpoint.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/purchases', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_site_purchases' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
// Get Location stats.
register_rest_route(
static::$namespace,
sprintf( '/sites/%d/stats/location-views/(?P<geo_mode>country|region|city)', Jetpack_Options::get_option( 'id' ) ),
array(
'methods' => WP_REST_Server::READABLE,
'callback' => array( $this, 'get_location_stats' ),
'permission_callback' => array( $this, 'can_user_view_general_stats_callback' ),
)
);
}
/**
* Only administrators or users with capability `view_stats` can access the API.
*
* @return bool|WP_Error True if a blog token was used to sign the request, WP_Error otherwise.
*/
public function can_user_view_general_stats_callback() {
if ( current_user_can( 'manage_options' ) || current_user_can( 'view_stats' ) ) {
return true;
}
return $this->get_forbidden_error();
}
/**
* Only administrators or users with capability `activate_wordads` can access the API.
*/
public function can_user_view_wordads_stats_callback() {
// phpcs:ignore WordPress.WP.Capabilities.Unknown
if ( current_user_can( 'manage_options' ) || current_user_can( 'activate_wordads' ) ) {
return true;
}
return $this->get_forbidden_error();
}
/**
* Stats resource endpoint.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_stats_resource( $req ) {
switch ( $req->get_param( 'resource' ) ) {
case 'file-downloads':
return $this->wpcom_stats->get_file_downloads( $req->get_params() );
case 'video-plays':
return $this->wpcom_stats->get_video_plays( $req->get_params() );
case 'clicks':
return $this->wpcom_stats->get_clicks( $req->get_params() );
case 'search-terms':
return $this->wpcom_stats->get_search_terms( $req->get_params() );
case 'top-authors':
return $this->wpcom_stats->get_top_authors( $req->get_params() );
case 'country-views':
return $this->wpcom_stats->get_views_by_country( $req->get_params() );
case 'referrers':
return $this->wpcom_stats->get_referrers( $req->get_params() );
case 'top-posts':
return $this->wpcom_stats->get_top_posts( $req->get_params() );
case 'publicize':
return $this->wpcom_stats->get_publicize_followers( $req->get_params() );
case 'followers':
return $this->wpcom_stats->get_followers( $req->get_params() );
case 'tags':
return $this->wpcom_stats->get_tags( $req->get_params() );
case 'visits':
return $this->wpcom_stats->get_visits( $req->get_params() );
case 'comments':
return $this->wpcom_stats->get_top_comments( $req->get_params() );
case 'comment-followers':
return $this->wpcom_stats->get_comment_followers( $req->get_params() );
case 'streak':
return $this->wpcom_stats->get_streak( $req->get_params() );
case 'insights':
return $this->wpcom_stats->get_insights( $req->get_params() );
case 'highlights':
return $this->wpcom_stats->get_highlights( $req->get_params() );
case 'subscribers':
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/stats/subscribers?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v1.1',
array( 'timeout' => 5 )
);
default:
return $this->get_forbidden_error();
}
}
/**
* Return likes of a single post.
*
* @param WP_REST_Request $req The request object.
*/
public function get_single_post_likes( $req ) {
$response = wp_remote_get(
sprintf(
'%s/rest/v1.2/sites/%d/posts/%d/likes?%s',
Constants::get_constant( 'JETPACK__WPCOM_JSON_API_BASE' ),
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'resource_id' ),
$this->filter_and_build_query_string(
$req->get_params(),
array( 'resource_id' )
)
),
array( 'timeout' => 5 )
);
$response_code = wp_remote_retrieve_response_code( $response );
$response_body = json_decode( wp_remote_retrieve_body( $response ), true );
if ( is_wp_error( $response ) ) {
return $response;
}
if ( 200 !== $response_code ) {
return new WP_Error(
isset( $response_body['error'] ) ? 'remote-error-' . $response_body['error'] : 'remote-error',
isset( $response_body['message'] ) ? $response_body['message'] : 'unknown remote error',
array( 'status' => $response_code )
);
}
return $response_body;
}
/**
* Site Stats Resource endpoint.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_single_resource_stats( $req ) {
switch ( $req->get_param( 'resource' ) ) {
case 'post':
return $this->wpcom_stats->get_post_views(
intval( $req->get_param( 'resource_id' ) ),
$req->get_params()
);
case 'video':
return $this->wpcom_stats->get_video_details(
intval( $req->get_param( 'resource_id' ) ),
$req->get_params()
);
default:
return $this->get_forbidden_error();
}
}
/**
* Get brief information for a single post.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_single_post( $req ) {
$post = get_post( intval( $req->get_param( 'resource_id' ) ), 'OBJECT', 'display' );
if ( is_wp_error( $post ) || empty( $post ) ) {
return $post;
}
// The endpoint should be as compatible as possible with `/sites/$site_id/posts/$post_id`.
// The reason we are not forwarding the request is that `/sites/$site_id/posts/$post_id` might require user tokens for private posts/sites, which is not possible for users without a WordPress.com account.
// 'like_count' is not included in the response because it's available through another endpoint `/sites/$site_id/posts/$post_id/likes`.
return array(
'ID' => $post->ID,
'site_ID' => Jetpack_Options::get_option( 'id' ),
'title' => $post->post_title,
'URL' => get_permalink( $post->ID ),
'type' => $post->post_type,
'status' => $post->post_status,
'discussion' => array( 'comment_count' => intval( $post->comment_count ) ),
'date' => $post->post_date,
'post_thumbnail' => array( 'URL' => get_the_post_thumbnail_url( $post->ID ) ),
);
}
/**
* Get site stats.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_site_stats( $req ) {
return $this->wpcom_stats->get_stats( $req->get_params() );
}
/**
* List posts for the site.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_site_posts( $req ) {
// Force wpcom response.
$params = array_merge( array( 'force' => 'wpcom' ), $req->get_params() );
$response = wp_remote_get(
sprintf(
'%s/rest/v1.1/sites/%d/posts?%s',
Constants::get_constant( 'JETPACK__WPCOM_JSON_API_BASE' ),
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'resource_id' ),
$this->filter_and_build_query_string( $params, array( 'resource_id' ) )
),
array( 'timeout' => 5 )
);
$response_code = wp_remote_retrieve_response_code( $response );
$response_body = json_decode( wp_remote_retrieve_body( $response ), true );
if ( is_wp_error( $response ) ) {
return $response;
}
if ( 200 !== $response_code ) {
return new WP_Error(
isset( $response_body['error'] ) ? 'remote-error-' . $response_body['error'] : 'remote-error',
isset( $response_body['message'] ) ? $response_body['message'] : 'unknown remote error',
array( 'status' => $response_code )
);
}
return $response_body;
}
/**
* Get site subscribers counts.
*
* @param WP_REST_Request $req The request object.
*
* @return array
*/
public function get_site_subscribers_counts( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/subscribers/counts?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array( 'timeout' => 5 ),
null,
'wpcom'
);
}
/**
* Get site plan usage.
*
* @param WP_REST_Request $req The request object.
*
* @return array
*/
public function get_site_plan_usage( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/jetpack-stats/usage?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array( 'timeout' => 5 ),
null,
'wpcom',
false
);
}
/**
* Post user feedback for Jetpack Stats.
*
* @param WP_REST_Request $req The request object.
*
* @return array
*/
public function post_user_feedback( $req ) {
$current_user = wp_get_current_user();
$body_from_req = json_decode( $req->get_body(), true );
$body_data = is_array( $body_from_req ) ? $body_from_req : array();
$user_email = $current_user->user_email;
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/jetpack-stats/user-feedback?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array(
'timeout' => 5,
'method' => 'POST',
'headers' => array( 'Content-Type' => 'application/json' ),
),
wp_json_encode(
array_merge(
$body_data,
array(
'user_email' => $user_email,
)
)
),
'wpcom'
);
}
/**
* Whether site has never published post.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function site_has_never_published_post( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/site-has-never-published-post?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v2',
array( 'timeout' => 5 ),
null,
'wpcom'
);
}
/**
* Get detailed WordAds earnings information for the site.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_wordads_earnings( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/wordads/earnings?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 5 )
);
}
/**
* Get WordAds stats for the site.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_wordads_stats( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/wordads/stats?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 5 )
);
}
/**
* Get Email stats as a list.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_email_stats_list( $req ) {
switch ( $req->get_param( 'resource' ) ) {
case 'summary':
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/stats/emails/%s?%s',
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'resource' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 5 )
);
default:
return $this->get_forbidden_error();
}
}
/**
* Get Email opens stats for a single post.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_email_opens_stats_single( $req ) {
switch ( $req->get_param( 'resource' ) ) {
case 'client':
case 'device':
case 'country':
case 'rate':
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/stats/opens/emails/%d/%s?%s',
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'post_id' ),
$req->get_param( 'resource' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 5 )
);
default:
return $this->get_forbidden_error();
}
}
/**
* Get Email clicks stats for a single post.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_email_clicks_stats_single( $req ) {
switch ( $req->get_param( 'resource' ) ) {
case 'client':
case 'device':
case 'country':
case 'rate':
case 'link':
case 'user-content-link':
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/stats/clicks/emails/%d/%s?%s',
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'post_id' ),
$req->get_param( 'resource' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 5 )
);
default:
return $this->get_forbidden_error();
}
}
/**
* Get Email stats time series.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_email_stats_time_series( $req ) {
switch ( $req->get_param( 'resource' ) ) {
case 'opens':
case 'clicks':
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/stats/%s/emails/%d?%s',
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'resource' ),
$req->get_param( 'post_id' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 5 )
);
default:
return $this->get_forbidden_error();
}
}
/**
* Get UTM stats time series.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_utm_stats_time_series( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/stats/utm/%s?%s',
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'utm_params' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 10 )
);
}
/**
* Get Devices stats time series.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_devices_stats_time_series( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/stats/devices/%s?%s',
Jetpack_Options::get_option( 'id' ),
$req->get_param( 'device_property' ),
$this->filter_and_build_query_string(
$req->get_params()
)
),
'v1.1',
array( 'timeout' => 10 )
);
}
/**
* Get Location stats.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_location_stats( $req ) {
$params = $req->get_params();
$geo_mode = $params['geo_mode'];
unset( $params['geo_mode'] );
return $this->wpcom_stats->get_views_by_location( $geo_mode, $params );
}
/**
* Dismiss or delay stats notices.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function update_notice_status( $req ) {
return ( new Notices() )->update_notice( $req->get_param( 'id' ), $req->get_param( 'status' ), $req->get_param( 'postponed_for' ) );
}
/**
* Get stats notices.
*
* @return array
*/
public function get_notice_status() {
return ( new Notices() )->get_notices_to_show();
}
/**
* Mark a referrer as spam.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function mark_referrer_spam( $req ) {
return WPCOM_Client::request_as_blog(
sprintf(
'/sites/%d/stats/referrers/spam/new?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v1.1',
array(
'timeout' => 5,
'method' => 'POST',
)
);
}
/**
* Unmark a referrer as spam.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function unmark_referrer_spam( $req ) {
return WPCOM_Client::request_as_blog(
sprintf(
'/sites/%d/stats/referrers/spam/delete?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v1.1',
array(
'timeout' => 5,
'method' => 'POST',
)
);
}
/**
* Toggle modules on dashboard.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function update_dashboard_modules( $req ) {
// Clear dashboard modules cache.
delete_transient( static::JETPACK_STATS_DASHBOARD_MODULES_CACHE_KEY );
return WPCOM_Client::request_as_blog(
sprintf(
'/sites/%d/jetpack-stats-dashboard/modules?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array(
'timeout' => 5,
'method' => 'POST',
'headers' => array( 'Content-Type' => 'application/json' ),
),
$req->get_body(),
'wpcom'
);
}
/**
* Get modules on dashboard.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_dashboard_modules( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/jetpack-stats-dashboard/modules?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array(
'timeout' => 5,
),
null,
'wpcom',
true,
static::JETPACK_STATS_DASHBOARD_MODULES_CACHE_KEY
);
}
/**
* Update module settings on dashboard.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function update_dashboard_module_settings( $req ) {
// Clear dashboard modules cache.
delete_transient( static::JETPACK_STATS_DASHBOARD_MODULE_SETTINGS_CACHE_KEY );
return WPCOM_Client::request_as_blog(
sprintf(
'/sites/%d/jetpack-stats-dashboard/module-settings?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array(
'timeout' => 5,
'method' => 'POST',
'headers' => array( 'Content-Type' => 'application/json' ),
),
$req->get_body(),
'wpcom'
);
}
/**
* Get module settings on dashboard.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_dashboard_module_settings( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/jetpack-stats-dashboard/module-settings?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array(
'timeout' => 5,
),
null,
'wpcom',
true,
static::JETPACK_STATS_DASHBOARD_MODULE_SETTINGS_CACHE_KEY
);
}
/**
* Run commercial classification.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function run_commercial_classification( $req ) {
return WPCOM_Client::request_as_blog(
sprintf(
'/sites/%d/commercial-classification?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v2',
array(
'timeout' => 5,
'method' => 'POST',
),
null,
'wpcom'
);
}
/**
* Get purchases array; I don't see anything sensetive in there, so didn't sentinizie it.
* Plus it is the same case as Jetpack.
*
* @param WP_REST_Request $req The request object.
* @return array
*/
public function get_site_purchases( $req ) {
return WPCOM_Client::request_as_blog_cached(
sprintf(
'/sites/%d/purchases?%s',
Jetpack_Options::get_option( 'id' ),
$this->filter_and_build_query_string(
$req->get_query_params()
)
),
'v1.1',
array( 'timeout' => 10 ),
null,
'rest',
false
);
}
/**
* Return a WP_Error object with a forbidden error.
*/
protected function get_forbidden_error() {
$error_msg = esc_html__(
'You are not allowed to perform this action.',
'jetpack-stats-admin'
);
return new WP_Error( 'rest_forbidden', $error_msg, array( 'status' => rest_authorization_required_code() ) );
}
/**
* Filter and build query string from all the requested params.
*
* @param array $params The params to filter.
* @param array $keys_to_unset The keys to unset from the params array.
* @return string The filtered and built query string.
*/
protected function filter_and_build_query_string( $params, $keys_to_unset = array() ) {
if ( isset( $params['rest_route'] ) ) {
unset( $params['rest_route'] );
}
if ( ! empty( $keys_to_unset ) && is_array( $keys_to_unset ) ) {
foreach ( $keys_to_unset as $key ) {
if ( isset( $params[ $key ] ) ) {
unset( $params[ $key ] );
}
}
}
return http_build_query( $params );
}
}