rray( 'methods' => WP_REST_Server::READABLE, 'callback' => array( $this, 'get_search_results' ), 'permission_callback' => 'is_user_logged_in', ) ); register_rest_route( static::$namespace, '/search/local-stats', array( 'methods' => WP_REST_Server::READABLE, 'callback' => array( $this, 'get_local_stats' ), 'permission_callback' => array( $this, 'require_valid_blog_token_callback' ), ) ); } /** * Routes only existing in WPCOM. * * We currently don't have any. */ protected function register_wpcom_only_rest_routes() { return true; } /** * Only administrators can access the API. * * @return bool|WP_Error True if a blog token was used to sign the request, WP_Error otherwise. */ public function require_admin_privilege_callback() { if ( current_user_can( 'manage_options' ) ) { return true; } return $this->get_forbidden_error(); } /** * The corresponding endpoints can only be accessible from WPCOM. * * @access public * @static * * @return bool|WP_Error True if a blog token was used to sign the request, WP_Error otherwise. */ public function require_valid_blog_token_callback() { if ( Rest_Authentication::is_signed_with_blog_token() ) { return true; } return $this->get_forbidden_error(); } /** * 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-search-pkg' ); return new WP_Error( 'rest_forbidden', $error_msg, array( 'status' => rest_authorization_required_code() ) ); } /** * Proxy the request to WPCOM and return the response. * * GET `jetpack/v4/search/plan` */ public function get_search_plan() { $response = ( new Plan() )->get_plan_info_from_wpcom(); return $this->make_proper_response( $response ); } /** * POST `jetpack/v4/search/settings` * * @param WP_REST_Request $request - REST request. */ public function update_settings( $request ) { $request_body = $request->get_json_params(); $module_active = isset( $request_body['module_active'] ) ? (bool) $request_body['module_active'] : null; $instant_search_enabled = isset( $request_body['instant_search_enabled'] ) ? (bool) $request_body['instant_search_enabled'] : null; $error = $this->validate_search_settings( $module_active, $instant_search_enabled ); if ( is_wp_error( $error ) ) { return $error; } // Enabling instant search should enable the module too. if ( true === $instant_search_enabled && true !== $module_active ) { $module_active = true; } $errors = array(); if ( $module_active !== null ) { $module_active_updated = $this->search_module->update_status( $module_active ); if ( is_wp_error( $module_active_updated ) ) { $errors['module_active'] = $module_active_updated; } } if ( $instant_search_enabled !== null ) { $instant_search_enabled_updated = $this->search_module->update_instant_search_status( $instant_search_enabled ); if ( is_wp_error( $instant_search_enabled_updated ) ) { $errors['instant_search_enabled'] = $instant_search_enabled_updated; } } if ( ! empty( $errors ) ) { return new WP_Error( 'some_updated', sprintf( /* translators: %s are the setting name that not updated. */ __( 'Some settings ( %s ) not updated.', 'jetpack-search-pkg' ), implode( ',', array_keys( $errors ) ) ), array( 'status' => 400 ) ); } return rest_ensure_response( $this->get_settings() ); } /** * Validate $module_active and $instant_search_enabled. Returns an WP_Error instance if invalid. * * @param boolean $module_active - Module status. * @param boolean $instant_search_enabled - Instant Search status. */ protected function validate_search_settings( $module_active, $instant_search_enabled ) { if ( ( true === $instant_search_enabled && false === $module_active ) || ( $module_active === null && $instant_search_enabled === null ) ) { return new WP_Error( 'rest_invalid_arguments', esc_html__( 'The arguments passed in are invalid.', 'jetpack-search-pkg' ), array( 'status' => 400 ) ); } return true; } /** * GET `jetpack/v4/search/settings` */ public function get_settings() { return rest_ensure_response( array( 'module_active' => $this->search_module->is_active(), 'instant_search_enabled' => $this->search_module->is_instant_search_enabled(), ) ); } /** * Proxy the request to WPCOM and return the response. * * GET `jetpack/v4/search/stats` */ public function get_stats() { $response = ( new Stats() )->get_stats_from_wpcom(); return $this->make_proper_response( $response ); } /** * Search Endpoint for private sites. * * GET `jetpack/v4/search` * * @param WP_REST_Request $request - REST request. */ public function get_search_results( $request ) { $blog_id = $this->get_blog_id(); $path = sprintf( '/sites/%d/search', absint( $blog_id ) ); $path = add_query_arg( $request->get_query_params(), sprintf( '/sites/%d/search', absint( $blog_id ) ) ); $response = Client::wpcom_json_api_request_as_blog( $path, '1.3', array(), null, 'rest' ); return rest_ensure_response( $this->make_proper_response( $response ) ); } /** * Activate plan: activate the search module, instant search and do initial configuration. * Typically called from WPCOM. * * POST `jetpack/v4/search/plan/activate` * * @param WP_REST_Request $request - REST request. */ public function activate_plan( $request ) { $default_options = array( 'search_plan_info' => null, 'enable_search' => true, 'enable_instant_search' => true, 'auto_config_search' => true, ); $payload = $request->get_json_params(); $payload = wp_parse_args( $payload, $default_options ); // Update plan data, plan info is in the request body. // We do this to avoid another call to WPCOM and reduce latency. if ( $payload['search_plan_info'] === null || ! $this->plan->set_plan_options( $payload['search_plan_info'] ) ) { $this->plan->get_plan_info_from_wpcom(); } // Enable search module by default, unless `enable_search` is explicitly set to boolean `false`. if ( false !== $payload['enable_search'] ) { // Eligibility is checked in `activate` function. $ret = $this->search_module->activate(); if ( is_wp_error( $ret ) ) { return $ret; } } // Enable instant search by default, unless `enable_instant_search` is explicitly set to boolean `false`. if ( false !== $payload['enable_instant_search'] ) { // Eligibility is checked in `enable_instant_search` function. $ret = $this->search_module->enable_instant_search(); if ( is_wp_error( $ret ) ) { return $ret; } } // Automatically configure necessary settings for instant search, unless `auto_config_search` is explicitly set to boolean `false`. if ( false !== $payload['auto_config_search'] ) { Instant_Search::instance( $this->get_blog_id() )->auto_config_search(); } return rest_ensure_response( array( 'code' => 'success', ) ); } /** * Deactivate plan: turn off search module and instant search. * If the plan is still valid then the function would simply deactivate the search module. * Typically called from WPCOM. * * POST `jetpack/v4/search/plan/deactivate` */ public function deactivate_plan() { // Instant Search would be disabled along with search module. $this->search_module->deactivate(); return rest_ensure_response( array( 'code' => 'success', ) ); } /** * Return post type breakdown for the site. */ public function get_local_stats() { return array( 'post_count' => Search_Product_Stats::estimate_count(), 'post_type_breakdown' => Search_Product_Stats::get_post_type_breakdown(), ); } /** * Pricing for record count of the site */ public function product_pricing() { $tier_pricing = Search_Product::get_pricing_for_ui(); // we can force the plugin to use the new pricing by appending `new_pricing_202208=1` to URL. if ( Helper::is_forced_new_pricing_202208() ) { $tier_pricing['pricing_version'] = Plan::JETPACK_SEARCH_NEW_PRICING_VERSION; } return rest_ensure_response( $tier_pricing ); } /** * Forward remote response to client with error handling. * * @param array|WP_Error $response - Response from WPCOM. */ protected function make_proper_response( $response ) { if ( is_wp_error( $response ) ) { return $response; } $body = json_decode( wp_remote_retrieve_body( $response ), true ); $status_code = wp_remote_retrieve_response_code( $response ); if ( 200 === $status_code ) { return $body; } return new WP_Error( isset( $body['error'] ) ? 'remote-error-' . $body['error'] : 'remote-error', isset( $body['message'] ) ? $body['message'] : 'unknown remote error', array( 'status' => $status_code ) ); } /** * Get blog id */ protected function get_blog_id() { return $this->is_wpcom ? get_current_blog_id() : Jetpack_Options::get_option( 'id' ); } }
Fatal error: Uncaught Error: Class "Automattic\Jetpack\Search\REST_Controller" not found in /htdocs/wp-content/plugins/jetpack/jetpack_vendor/automattic/jetpack-search/src/initializers/class-initializer.php:102 Stack trace: #0 /htdocs/wp-content/plugins/jetpack/jetpack_vendor/automattic/jetpack-search/src/initializers/class-initializer.php(48): Automattic\Jetpack\Search\Initializer::init_before_connection() #1 /htdocs/wp-content/plugins/jetpack/jetpack_vendor/automattic/jetpack-config/src/class-config.php(282): Automattic\Jetpack\Search\Initializer::init() #2 /htdocs/wp-content/plugins/jetpack/jetpack_vendor/automattic/jetpack-config/src/class-config.php(217): Automattic\Jetpack\Config->enable_search() #3 /htdocs/wp-content/plugins/jetpack/jetpack_vendor/automattic/jetpack-config/src/class-config.php(134): Automattic\Jetpack\Config->ensure_feature('search') #4 /htdocs/wp-includes/class-wp-hook.php(324): Automattic\Jetpack\Config->on_plugins_loaded('') #5 /htdocs/wp-includes/class-wp-hook.php(348): WP_Hook->apply_filters(NULL, Array) #6 /htdocs/wp-includes/plugin.php(517): WP_Hook->do_action(Array) #7 /htdocs/wp-settings.php(559): do_action('plugins_loaded') #8 /htdocs/wp-config.php(85): require_once('/htdocs/wp-sett...') #9 /htdocs/wp-load.php(50): require_once('/htdocs/wp-conf...') #10 /htdocs/wp-blog-header.php(13): require_once('/htdocs/wp-load...') #11 /htdocs/index.php(17): require('/htdocs/wp-blog...') #12 {main} thrown in /htdocs/wp-content/plugins/jetpack/jetpack_vendor/automattic/jetpack-search/src/initializers/class-initializer.php on line 102