Adding Navigation Favorites REST API Endpoints (https://github.com/woocommerce/woocommerce-admin/pull/6282)

This commit is contained in:
Joel Thiessen 2021-02-09 18:35:47 -08:00 committed by GitHub
parent 4cd14f5137
commit fda46f9920
3 changed files with 360 additions and 10 deletions

View File

@ -97,6 +97,7 @@ class Init {
'Automattic\WooCommerce\Admin\API\OnboardingProfile', 'Automattic\WooCommerce\Admin\API\OnboardingProfile',
'Automattic\WooCommerce\Admin\API\OnboardingTasks', 'Automattic\WooCommerce\Admin\API\OnboardingTasks',
'Automattic\WooCommerce\Admin\API\OnboardingThemes', 'Automattic\WooCommerce\Admin\API\OnboardingThemes',
'Automattic\WooCommerce\Admin\API\NavigationFavorites',
); );
// The performance indicators controller must be registered last, after other /stats endpoints have been registered. // The performance indicators controller must be registered last, after other /stats endpoints have been registered.

View File

@ -0,0 +1,327 @@
<?php
/**
* REST API Navigation Favorites controller
*
* Handles requests to the navigation favorites endpoint
*/
namespace Automattic\WooCommerce\Admin\API;
defined( 'ABSPATH' ) || exit;
use Automattic\WooCommerce\Admin\Features\Navigation\Favorites;
/**
* REST API Favorites controller class.
*
* @extends WC_REST_CRUD_Controller
*/
class NavigationFavorites extends \WC_REST_Data_Controller {
/**
* Endpoint namespace.
*
* @var string
*/
protected $namespace = 'wc-admin';
/**
* Route base.
*
* @var string
*/
protected $rest_base = 'navigation/favorites';
/**
* Error code to status code mapping.
*
* @var array
*/
protected $error_to_status_map = array(
'woocommerce_favorites_invalid_request' => 400,
'woocommerce_favorites_already_exists' => 409,
'woocommerce_favorites_does_not_exist' => 404,
'woocommerce_favorites_invalid_user' => 400,
'woocommerce_favorites_unauthenticated' => 401,
);
/**
* Register the routes
*/
public function register_routes() {
register_rest_route(
$this->namespace,
'/' . $this->rest_base,
array(
array(
'methods' => \WP_REST_Server::READABLE,
'callback' => array( $this, 'get_items' ),
'permission_callback' => array( $this, 'get_items_permissions_check' ),
'args' => array(
'user_id' => array(
'required' => true,
'validate_callback' => function( $param, $request, $key ) {
return is_numeric( $param );
},
),
),
),
array(
'methods' => \WP_REST_Server::CREATABLE,
'callback' => array( $this, 'add_item' ),
'permission_callback' => array( $this, 'add_item_permissions_check' ),
'args' => array(
'item_id' => array(
'required' => true,
),
'user_id' => array(
'required' => true,
'validate_callback' => function( $param, $request, $key ) {
return is_numeric( $param );
},
),
),
),
array(
'methods' => \WP_REST_Server::DELETABLE,
'callback' => array( $this, 'delete_item' ),
'permission_callback' => array( $this, 'delete_item_permissions_check' ),
'args' => array(
'item_id' => array(
'required' => true,
),
'user_id' => array(
'required' => true,
'validate_callback' => function( $param, $request, $key ) {
return is_numeric( $param );
},
),
),
),
'schema' => array( $this, 'get_public_item_schema' ),
)
);
register_rest_route(
$this->namespace,
'/' . $this->rest_base . '/me',
array(
array(
'methods' => \WP_REST_Server::READABLE,
'callback' => array( $this, 'get_items_by_current_user' ),
'permission_callback' => array( $this, 'current_user_permissions_check' ),
),
array(
'methods' => \WP_REST_Server::CREATABLE,
'callback' => array( $this, 'add_item_by_current_user' ),
'permission_callback' => array( $this, 'current_user_permissions_check' ),
'args' => array(
'item_id' => array(
'required' => true,
),
),
),
array(
'methods' => \WP_REST_Server::DELETABLE,
'callback' => array( $this, 'delete_item_by_current_user' ),
'permission_callback' => array( $this, 'current_user_permissions_check' ),
'args' => array(
'item_id' => array(
'required' => true,
),
),
),
'schema' => array( $this, 'get_public_item_schema' ),
)
);
}
/**
* Get all favorites.
*
* @param WP_REST_Request $request Request data.
* @return WP_REST_Response
*/
public function get_items( $request ) {
$user_id = $request->get_param( 'user_id' );
$response = Favorites::get_all( $user_id );
if ( is_wp_error( $response ) || ! $response ) {
return rest_ensure_response( $this->prepare_error( $response ) );
}
return rest_ensure_response(
array_map( 'stripslashes', $response )
);
}
/**
* Get all favorites of current user.
*
* @param WP_REST_Request $request Request data.
* @return WP_REST_Response
*/
public function get_items_by_current_user( $request ) {
$current_user = get_current_user_id();
if ( ! $current_user ) {
return $this->prepare_error(
new \WP_Error(
'woocommerce_favorites_unauthenticated',
__( 'You must be authenticated to use this endpoint', 'woocommerce-admin' )
)
);
}
$request->set_param( 'user_id', $current_user );
return $this->get_items( $request );
}
/**
* Add a favorite.
*
* @param WP_REST_Request $request Request data.
* @return WP_REST_Response
*/
public function add_item( $request ) {
$user_id = $request->get_param( 'user_id' );
$fav_id = $request->get_param( 'item_id' );
$user = get_userdata( $user_id );
if ( false === $user ) {
return $this->prepare_error(
new \WP_Error(
'woocommerce_favorites_invalid_user',
__( 'Invalid user_id provided', 'woocommerce-admin' )
)
);
}
$response = Favorites::add_item( $fav_id, $user_id );
if ( is_wp_error( $response ) || ! $response ) {
return rest_ensure_response( $this->prepare_error( $response ) );
}
return rest_ensure_response( Favorites::get_all( $user_id ) );
}
/**
* Add a favorite for current user.
*
* @param WP_REST_Request $request Request data.
* @return WP_REST_Response
*/
public function add_item_by_current_user( $request ) {
$current_user = get_current_user_id();
if ( ! $current_user ) {
return $this->prepare_error(
new \WP_Error(
'woocommerce_favorites_unauthenticated',
__( 'You must be authenticated to use this endpoint', 'woocommerce-admin' )
)
);
}
$request->set_param( 'user_id', get_current_user_id() );
return $this->add_item( $request );
}
/**
* Delete a favorite.
*
* @param WP_REST_Request $request Request data.
* @return WP_REST_Response
*/
public function delete_item( $request ) {
$user_id = $request->get_param( 'user_id' );
$fav_id = $request->get_param( 'item_id' );
$response = Favorites::remove_item( $fav_id, $user_id );
if ( is_wp_error( $response ) || ! $response ) {
return rest_ensure_response( $this->prepare_error( $response ) );
}
return rest_ensure_response( Favorites::get_all( $user_id ) );
}
/**
* Delete a favorite for current user.
*
* @param WP_REST_Request $request Request data.
* @return WP_REST_Response
*/
public function delete_item_by_current_user( $request ) {
$current_user = get_current_user_id();
if ( ! $current_user ) {
return $this->prepare_error(
new \WP_Error(
'woocommerce_favorites_unauthenticated',
__( 'You must be authenticated to use this endpoint', 'woocommerce-admin' )
)
);
}
$request->set_param( 'user_id', $current_user );
return $this->delete_item( $request );
}
/**
* Check whether a given request has permission to create favorites.
*
* @param WP_REST_Request $request Full details about the request.
* @return WP_Error|boolean
*/
public function add_item_permissions_check( $request ) {
return current_user_can( 'edit_users' );
}
/**
* Check whether a given request has permission to delete notes.
*
* @param WP_REST_Request $request Full details about the request.
* @return WP_Error|boolean
*/
public function delete_item_permissions_check( $request ) {
return current_user_can( 'edit_users' );
}
/**
* Always allow for operations that only impact current user
*
* @param WP_REST_Request $request Full details about the request.
* @return WP_Error|boolean
*/
public function current_user_permissions_check( $request ) {
return true;
}
/**
* Accept an instance of WP_Error and add the appropriate data for REST transit.
*
* @param WP_Error $error Error to prepare.
* @return WP_Error
*/
protected function prepare_error( $error ) {
if ( ! is_wp_error( $error ) ) {
return $error;
}
$error->add_data(
array(
'status' => $this->error_to_status_map[ $error->get_error_code() ] ?? 500,
)
);
return $error;
}
}

View File

@ -46,23 +46,32 @@ class Favorites {
* *
* @param string $item_id Identifier of item to add. * @param string $item_id Identifier of item to add.
* @param string|number $user_id Identifier of user to add to. * @param string|number $user_id Identifier of user to add to.
* @return WP_Error|Boolean Throws exception if item already exists.
*/ */
public static function add_item( $item_id, $user_id = null ) { public static function add_item( $item_id, $user_id = null ) {
$user = $user_id ? $user_id : get_current_user_id(); $user = $user_id ?? get_current_user_id();
if ( ! $user || ! $item_id ) { if ( ! $user || ! $item_id ) {
return; return new \WP_Error(
'woocommerce_favorites_invalid_request',
__( 'Sorry, invalid request', 'woocommerce-admin' )
);
} }
$all_favorites = self::get_all( $user ); $all_favorites = self::get_all( $user );
if ( in_array( $item_id, $all_favorites, true ) ) { if ( in_array( $item_id, $all_favorites, true ) ) {
return; return new \WP_Error(
'woocommerce_favorites_already_exists',
__( 'Favorite already exists', 'woocommerce-admin' )
);
} }
$all_favorites[] = $item_id; $all_favorites[] = $item_id;
self::set_meta_value( $user, $all_favorites ); self::set_meta_value( $user, $all_favorites );
return true;
} }
/** /**
@ -70,35 +79,48 @@ class Favorites {
* *
* @param string $item_id Identifier of item to remove. * @param string $item_id Identifier of item to remove.
* @param string|number $user_id Identifier of user to remove from. * @param string|number $user_id Identifier of user to remove from.
* @return \WP_Error|Boolean Throws exception if item does not exist.
*/ */
public static function remove_item( $item_id, $user_id = null ) { public static function remove_item( $item_id, $user_id = null ) {
$user = $user_id ? $user_id : get_current_user_id(); $user = $user_id ?? get_current_user_id();
if ( ! $user || ! $item_id ) { if ( ! $user || ! $item_id ) {
return; return new \WP_Error(
'woocommerce_favorites_invalid_request',
__( 'Sorry, invalid request', 'woocommerce-admin' )
);
} }
$all_favorites = self::get_all( $user ); $all_favorites = self::get_all( $user );
if ( ! in_array( $item_id, $all_favorites, true ) ) { if ( ! in_array( $item_id, $all_favorites, true ) ) {
return; return new \WP_Error(
'woocommerce_favorites_does_not_exist',
__( 'Favorite item not found', 'woocommerce-admin' )
);
} }
$remaining = array_diff( $all_favorites, [ $item_id ] ); $remaining = array_values( array_diff( $all_favorites, [ $item_id ] ) );
self::set_meta_value( $user, array_values( $remaining ) ); self::set_meta_value( $user, $remaining );
return true;
} }
/** /**
* Get all registered favorites. * Get all registered favorites.
* *
* @param string|number $user_id Identifier of user to query. * @param string|number $user_id Identifier of user to query.
* @return WP_Error|Array
*/ */
public static function get_all( $user_id = null ) { public static function get_all( $user_id = null ) {
$user = $user_id ? $user_id : get_current_user_id(); $user = $user_id ?? get_current_user_id();
if ( ! $user ) { if ( ! $user ) {
return; return new \WP_Error(
'woocommerce_favorites_invalid_request',
__( 'Sorry, invalid request', 'woocommerce-admin' )
);
} }
$response = Loader::get_user_data_field( $user, self::META_NAME ); $response = Loader::get_user_data_field( $user, self::META_NAME );