true, 'wpcom_product_slug' => static::get_wpcom_product_slug(), ), Wpcom_Products::get_product_pricing( static::get_wpcom_product_slug() ) ); $record_count = intval( Search_Stats::estimate_count() ); $search_pricing = static::get_pricing_from_wpcom( $record_count ); if ( is_wp_error( $search_pricing ) ) { return $pricing; } return array_merge( $pricing, $search_pricing ); } /** * Get the WPCOM product slug used to make the purchase * * @return ?string */ public static function get_wpcom_product_slug() { return 'jetpack_search'; } /** * Use centralized Search pricing API. * * The function is also used by the search package, as a result it could be called before site connection - i.e. blog token might not be available. * * @param int $record_count Record count to estimate pricing. * * @return array|WP_Error */ public static function get_pricing_from_wpcom( $record_count ) { static $pricings = array(); if ( isset( $pricings[ $record_count ] ) ) { return $pricings[ $record_count ]; } $response = wp_remote_get( sprintf( Constants::get_constant( 'JETPACK__WPCOM_JSON_API_BASE' ) . '/wpcom/v2/jetpack-search/pricing?record_count=%1$d&locale=%2$s', $record_count, get_user_locale() ), array( 'timeout' => 5 ) ); if ( is_wp_error( $response ) || 200 !== wp_remote_retrieve_response_code( $response ) ) { return new WP_Error( 'search_pricing_fetch_failed' ); } $body = wp_remote_retrieve_body( $response ); $pricings[ $record_count ] = json_decode( $body, true ); return $pricings[ $record_count ]; } /** * Hits the wpcom api to check Search status. * * @todo Maybe add caching. * * @return Object|WP_Error */ private static function get_state_from_wpcom() { static $status = null; if ( $status !== null ) { return $status; } $blog_id = Jetpack_Options::get_option( 'id' ); $response = Client::wpcom_json_api_request_as_blog( '/sites/' . $blog_id . '/jetpack-search/plan', '2', array( 'timeout' => 5 ), null, 'wpcom' ); if ( is_wp_error( $response ) || 200 !== wp_remote_retrieve_response_code( $response ) ) { return new WP_Error( 'search_state_fetch_failed' ); } $body = wp_remote_retrieve_body( $response ); $status = json_decode( $body ); return $status; } /** * Checks whether the current plan of the site already supports the product * * Returns true if it supports. Return false if a purchase is still required. * * Free products will always return true. * * @return boolean */ public static function has_required_plan() { $search_state = static::get_state_from_wpcom(); return ! empty( $search_state->supports_search ) || ! empty( $search_state->supports_instant_search ); } /** * Activates the product. Try to enable instant search after the Search module was enabled. * * @param bool|WP_Error $product_activation Is the result of the top level activation actions. You probably won't do anything if it is an WP_Error. * @return bool|WP_Error */ public static function do_product_specific_activation( $product_activation ) { $product_activation = parent::do_product_specific_activation( $product_activation ); if ( is_wp_error( $product_activation ) ) { return $product_activation; } if ( class_exists( 'Automattic\Jetpack\Search\Module_Control' ) ) { ( new Search_Module_Control() )->enable_instant_search(); } // we don't want to change the success of the activation if we fail to activate instant search. That's not mandatory. return $product_activation; } /** * Get the URL the user is taken after activating the product * * @return ?string */ public static function get_post_activation_url() { return ''; // stay in My Jetpack page or continue the purchase flow if needed. } /** * Get the URL where the user manages the product * * @return ?string */ public static function get_manage_url() { return admin_url( 'admin.php?page=jetpack-search' ); } }