REST API: Support custom namespaces for custom post types.

While a custom post type can define a custom route by using the `rest_base` argument, a namespace of `wp/v2` was assumed. This commit introduces support for a `rest_namespace` argument. 

A new `rest_get_route_for_post_type_items` function has been introduced and the `rest_get_route_for_post` function updated to facilitate getting the correct route for custom post types.

While the WordPress Core Block Editor bootstrap code has been updated to use these API functions, for maximum compatibility sticking with the default `wp/v2` namespace is recommended until the API functions see wider use.

Props spacedmonkey, swissspidy.
Fixes #53656.

Built from https://develop.svn.wordpress.org/trunk@51962


git-svn-id: http://core.svn.wordpress.org/trunk@51551 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit is contained in:
TimothyBlynJacobs 2021-10-31 23:16:58 +00:00
parent c60a9d92e2
commit bac6e41c85
14 changed files with 95 additions and 48 deletions

View File

@ -49,7 +49,7 @@ add_filter( 'screen_options_show_screen', '__return_false' );
wp_enqueue_script( 'heartbeat' ); wp_enqueue_script( 'heartbeat' );
wp_enqueue_script( 'wp-edit-post' ); wp_enqueue_script( 'wp-edit-post' );
$rest_base = ! empty( $post_type_object->rest_base ) ? $post_type_object->rest_base : $post_type_object->name; $rest_path = rest_get_route_for_post( $post );
// Preload common data. // Preload common data.
$preload_paths = array( $preload_paths = array(
@ -57,12 +57,12 @@ $preload_paths = array(
'/wp/v2/types?context=edit', '/wp/v2/types?context=edit',
'/wp/v2/taxonomies?per_page=-1&context=edit', '/wp/v2/taxonomies?per_page=-1&context=edit',
'/wp/v2/themes?status=active', '/wp/v2/themes?status=active',
sprintf( '/wp/v2/%s/%s?context=edit', $rest_base, $post->ID ), add_query_arg( 'context', 'edit', $rest_path ),
sprintf( '/wp/v2/types/%s?context=edit', $post_type ), sprintf( '/wp/v2/types/%s?context=edit', $post_type ),
sprintf( '/wp/v2/users/me?post_type=%s&context=edit', $post_type ), sprintf( '/wp/v2/users/me?post_type=%s&context=edit', $post_type ),
array( '/wp/v2/media', 'OPTIONS' ), array( rest_get_route_for_post_type_items( 'attachment' ), 'OPTIONS' ),
array( '/wp/v2/blocks', 'OPTIONS' ), array( rest_get_route_for_post_type_items( 'wp_block' ), 'OPTIONS' ),
sprintf( '/wp/v2/%s/%d/autosaves?context=edit', $rest_base, $post->ID ), sprintf( '%s/autosaves?context=edit', $rest_path ),
); );
block_editor_rest_api_preload( $preload_paths, $block_editor_context ); block_editor_rest_api_preload( $preload_paths, $block_editor_context );

View File

@ -18,7 +18,7 @@ $current_screen->is_block_editor( true );
$block_editor_context = new WP_Block_Editor_Context(); $block_editor_context = new WP_Block_Editor_Context();
$preload_paths = array( $preload_paths = array(
array( '/wp/v2/media', 'OPTIONS' ), array( rest_get_route_for_post_type_items( 'attachment' ), 'OPTIONS' ),
'/wp/v2/sidebars?context=edit&per_page=-1', '/wp/v2/sidebars?context=edit&per_page=-1',
'/wp/v2/widgets?context=edit&per_page=-1&_embed=about', '/wp/v2/widgets?context=edit&per_page=-1&_embed=about',
); );

View File

@ -358,6 +358,14 @@ final class WP_Post_Type {
*/ */
public $rest_base; public $rest_base;
/**
* The namespace for this post type's REST API endpoints.
*
* @since 5.9
* @var string|bool $rest_namespace
*/
public $rest_namespace;
/** /**
* The controller for this post type's REST API endpoints. * The controller for this post type's REST API endpoints.
* *
@ -452,6 +460,7 @@ final class WP_Post_Type {
'delete_with_user' => null, 'delete_with_user' => null,
'show_in_rest' => false, 'show_in_rest' => false,
'rest_base' => false, 'rest_base' => false,
'rest_namespace' => false,
'rest_controller_class' => false, 'rest_controller_class' => false,
'template' => array(), 'template' => array(),
'template_lock' => false, 'template_lock' => false,
@ -473,6 +482,11 @@ final class WP_Post_Type {
$args['show_ui'] = $args['public']; $args['show_ui'] = $args['public'];
} }
// If not set, default rest_namespace to wp/v2 if show_in_rest is true.
if ( false === $args['rest_namespace'] && ! empty( $args['show_in_rest'] ) ) {
$args['rest_namespace'] = 'wp/v2';
}
// If not set, default to the setting for 'show_ui'. // If not set, default to the setting for 'show_ui'.
if ( null === $args['show_in_menu'] || ! $args['show_ui'] ) { if ( null === $args['show_in_menu'] || ! $args['show_ui'] ) {
$args['show_in_menu'] = $args['show_ui']; $args['show_in_menu'] = $args['show_ui'];

View File

@ -1418,6 +1418,7 @@ function get_post_types( $args = array(), $output = 'names', $operator = 'and' )
* @type bool $show_in_rest Whether to include the post type in the REST API. Set this to true * @type bool $show_in_rest Whether to include the post type in the REST API. Set this to true
* for the post type to be available in the block editor. * for the post type to be available in the block editor.
* @type string $rest_base To change the base URL of REST API route. Default is $post_type. * @type string $rest_base To change the base URL of REST API route. Default is $post_type.
* @type string $rest_namespace To change the namespace URL of REST API route. Default is wp/v2.
* @type string $rest_controller_class REST API controller class name. Default is 'WP_REST_Posts_Controller'. * @type string $rest_controller_class REST API controller class name. Default is 'WP_REST_Posts_Controller'.
* @type int $menu_position The position in the menu order the post type should appear. To work, * @type int $menu_position The position in the menu order the post type should appear. To work,
* $show_in_menu must be true. Default null (at the bottom). * $show_in_menu must be true. Default null (at the bottom).

View File

@ -3049,24 +3049,12 @@ function rest_get_route_for_post( $post ) {
return ''; return '';
} }
$post_type = get_post_type_object( $post->post_type ); $post_type_route = rest_get_route_for_post_type_items( $post->post_type );
if ( ! $post_type ) { if ( ! $post_type_route ) {
return ''; return '';
} }
$controller = $post_type->get_rest_controller(); $route = sprintf( '%s/%d', $post_type_route, $post->ID );
if ( ! $controller ) {
return '';
}
$route = '';
// The only two controllers that we can detect are the Attachments and Posts controllers.
if ( in_array( get_class( $controller ), array( 'WP_REST_Attachments_Controller', 'WP_REST_Posts_Controller' ), true ) ) {
$namespace = 'wp/v2';
$rest_base = ! empty( $post_type->rest_base ) ? $post_type->rest_base : $post_type->name;
$route = sprintf( '/%s/%s/%d', $namespace, $rest_base, $post->ID );
}
/** /**
* Filters the REST API route for a post. * Filters the REST API route for a post.
@ -3079,6 +3067,39 @@ function rest_get_route_for_post( $post ) {
return apply_filters( 'rest_route_for_post', $route, $post ); return apply_filters( 'rest_route_for_post', $route, $post );
} }
/**
* Gets the REST API route for a post type.
*
* @since 5.9.0
*
* @param string $post_type The name of a registered post type.
* @return string The route path with a leading slash for the given post type, or an empty string if there is not a route.
*/
function rest_get_route_for_post_type_items( $post_type ) {
$post_type = get_post_type_object( $post_type );
if ( ! $post_type ) {
return '';
}
if ( ! $post_type->show_in_rest ) {
return '';
}
$namespace = ! empty( $post_type->rest_namespace ) ? $post_type->rest_namespace : 'wp/v2';
$rest_base = ! empty( $post_type->rest_base ) ? $post_type->rest_base : $post_type->name;
$route = sprintf( '/%s/%s', $namespace, $rest_base );
/**
* Filters the REST API route for a post type.
*
* @since 5.9.0
*
* @param string $route The route path.
* @param WP_Post_Type $post_type The post type object.
*/
return apply_filters( 'rest_route_for_post_type_items', $route, $post_type );
}
/** /**
* Gets the REST API route for a term. * Gets the REST API route for a term.
* *

View File

@ -1288,7 +1288,7 @@ class WP_REST_Server {
if ( $site_logo_id ) { if ( $site_logo_id ) {
$response->add_link( $response->add_link(
'https://api.w.org/featuredmedia', 'https://api.w.org/featuredmedia',
rest_url( 'wp/v2/media/' . $site_logo_id ), rest_url( rest_get_route_for_post( $site_logo_id ) ),
array( array(
'embeddable' => true, 'embeddable' => true,
) )

View File

@ -67,8 +67,8 @@ class WP_REST_Autosaves_Controller extends WP_REST_Revisions_Controller {
$this->parent_controller = $parent_controller; $this->parent_controller = $parent_controller;
$this->revisions_controller = new WP_REST_Revisions_Controller( $parent_post_type ); $this->revisions_controller = new WP_REST_Revisions_Controller( $parent_post_type );
$this->namespace = 'wp/v2';
$this->rest_base = 'autosaves'; $this->rest_base = 'autosaves';
$this->namespace = ! empty( $post_type_object->rest_namespace ) ? $post_type_object->rest_namespace : 'wp/v2';
$this->parent_base = ! empty( $post_type_object->rest_base ) ? $post_type_object->rest_base : $post_type_object->name; $this->parent_base = ! empty( $post_type_object->rest_base ) ? $post_type_object->rest_base : $post_type_object->name;
} }

View File

@ -263,10 +263,11 @@ class WP_REST_Post_Statuses_Controller extends WP_REST_Controller {
$response = rest_ensure_response( $data ); $response = rest_ensure_response( $data );
$rest_url = rest_url( rest_get_route_for_post_type_items( 'post' ) );
if ( 'publish' === $status->name ) { if ( 'publish' === $status->name ) {
$response->add_link( 'archives', rest_url( 'wp/v2/posts' ) ); $response->add_link( 'archives', $rest_url );
} else { } else {
$response->add_link( 'archives', add_query_arg( 'status', $status->name, rest_url( 'wp/v2/posts' ) ) ); $response->add_link( 'archives', add_query_arg( 'status', $status->name, $rest_url ) );
} }
/** /**

View File

@ -180,6 +180,7 @@ class WP_REST_Post_Types_Controller extends WP_REST_Controller {
$taxonomies = wp_list_filter( get_object_taxonomies( $post_type->name, 'objects' ), array( 'show_in_rest' => true ) ); $taxonomies = wp_list_filter( get_object_taxonomies( $post_type->name, 'objects' ), array( 'show_in_rest' => true ) );
$taxonomies = wp_list_pluck( $taxonomies, 'name' ); $taxonomies = wp_list_pluck( $taxonomies, 'name' );
$base = ! empty( $post_type->rest_base ) ? $post_type->rest_base : $post_type->name; $base = ! empty( $post_type->rest_base ) ? $post_type->rest_base : $post_type->name;
$namespace = ! empty( $post_type->rest_namespace ) ? $post_type->rest_namespace : 'wp/v2';
$supports = get_all_post_type_supports( $post_type->name ); $supports = get_all_post_type_supports( $post_type->name );
$fields = $this->get_fields_for_response( $request ); $fields = $this->get_fields_for_response( $request );
@ -232,6 +233,10 @@ class WP_REST_Post_Types_Controller extends WP_REST_Controller {
$data['rest_base'] = $base; $data['rest_base'] = $base;
} }
if ( in_array( 'rest_namespace', $fields, true ) ) {
$data['rest_namespace'] = $namespace;
}
$context = ! empty( $request['context'] ) ? $request['context'] : 'view'; $context = ! empty( $request['context'] ) ? $request['context'] : 'view';
$data = $this->add_additional_fields_to_object( $data, $request ); $data = $this->add_additional_fields_to_object( $data, $request );
$data = $this->filter_response_by_context( $data, $context ); $data = $this->filter_response_by_context( $data, $context );
@ -245,7 +250,7 @@ class WP_REST_Post_Types_Controller extends WP_REST_Controller {
'href' => rest_url( sprintf( '%s/%s', $this->namespace, $this->rest_base ) ), 'href' => rest_url( sprintf( '%s/%s', $this->namespace, $this->rest_base ) ),
), ),
'https://api.w.org/items' => array( 'https://api.w.org/items' => array(
'href' => rest_url( sprintf( 'wp/v2/%s', $base ) ), 'href' => rest_url( rest_get_route_for_post_type_items( $post_type->name ) ),
), ),
) )
); );
@ -269,7 +274,7 @@ class WP_REST_Post_Types_Controller extends WP_REST_Controller {
* *
* @since 4.7.0 * @since 4.7.0
* @since 4.8.0 The `supports` property was added. * @since 4.8.0 The `supports` property was added.
* @since 5.9.0 The `visibility` property was added. * @since 5.9.0 The `visibility` and `rest_namespace` properties were added.
* *
* @return array Item schema data. * @return array Item schema data.
*/ */
@ -346,6 +351,12 @@ class WP_REST_Post_Types_Controller extends WP_REST_Controller {
'context' => array( 'view', 'edit', 'embed' ), 'context' => array( 'view', 'edit', 'embed' ),
'readonly' => true, 'readonly' => true,
), ),
'rest_namespace' => array(
'description' => __( 'REST route\'s namespace for the post type.' ),
'type' => 'string',
'context' => array( 'view', 'edit', 'embed' ),
'readonly' => true,
),
'visibility' => array( 'visibility' => array(
'description' => __( 'The visibility settings for the post type.' ), 'description' => __( 'The visibility settings for the post type.' ),
'type' => 'object', 'type' => 'object',

View File

@ -48,9 +48,9 @@ class WP_REST_Posts_Controller extends WP_REST_Controller {
*/ */
public function __construct( $post_type ) { public function __construct( $post_type ) {
$this->post_type = $post_type; $this->post_type = $post_type;
$this->namespace = 'wp/v2';
$obj = get_post_type_object( $post_type ); $obj = get_post_type_object( $post_type );
$this->rest_base = ! empty( $obj->rest_base ) ? $obj->rest_base : $obj->name; $this->rest_base = ! empty( $obj->rest_base ) ? $obj->rest_base : $obj->name;
$this->namespace = ! empty( $obj->rest_namespace ) ? $obj->rest_namespace : 'wp/v2';
$this->meta = new WP_REST_Post_Meta_Fields( $this->post_type ); $this->meta = new WP_REST_Post_Meta_Fields( $this->post_type );
} }
@ -2037,7 +2037,7 @@ class WP_REST_Posts_Controller extends WP_REST_Controller {
// If we have a featured media, add that. // If we have a featured media, add that.
$featured_media = get_post_thumbnail_id( $post->ID ); $featured_media = get_post_thumbnail_id( $post->ID );
if ( $featured_media ) { if ( $featured_media ) {
$image_url = rest_url( 'wp/v2/media/' . $featured_media ); $image_url = rest_url( rest_get_route_for_post( $featured_media ) );
$links['https://api.w.org/featuredmedia'] = array( $links['https://api.w.org/featuredmedia'] = array(
'href' => $image_url, 'href' => $image_url,
@ -2046,7 +2046,7 @@ class WP_REST_Posts_Controller extends WP_REST_Controller {
} }
if ( ! in_array( $post->post_type, array( 'attachment', 'nav_menu_item', 'revision' ), true ) ) { if ( ! in_array( $post->post_type, array( 'attachment', 'nav_menu_item', 'revision' ), true ) ) {
$attachments_url = rest_url( 'wp/v2/media' ); $attachments_url = rest_url( rest_get_route_for_post_type_items( 'attachment' ) );
$attachments_url = add_query_arg( 'parent', $post->ID, $attachments_url ); $attachments_url = add_query_arg( 'parent', $post->ID, $attachments_url );
$links['https://api.w.org/attachment'] = array( $links['https://api.w.org/attachment'] = array(

View File

@ -49,10 +49,10 @@ class WP_REST_Revisions_Controller extends WP_REST_Controller {
*/ */
public function __construct( $parent_post_type ) { public function __construct( $parent_post_type ) {
$this->parent_post_type = $parent_post_type; $this->parent_post_type = $parent_post_type;
$this->namespace = 'wp/v2';
$this->rest_base = 'revisions'; $this->rest_base = 'revisions';
$post_type_object = get_post_type_object( $parent_post_type ); $post_type_object = get_post_type_object( $parent_post_type );
$this->parent_base = ! empty( $post_type_object->rest_base ) ? $post_type_object->rest_base : $post_type_object->name; $this->parent_base = ! empty( $post_type_object->rest_base ) ? $post_type_object->rest_base : $post_type_object->name;
$this->namespace = ! empty( $post_type_object->rest_namespace ) ? $post_type_object->rest_namespace : 'wp/v2';
$this->parent_controller = $post_type_object->get_rest_controller(); $this->parent_controller = $post_type_object->get_rest_controller();
if ( ! $this->parent_controller ) { if ( ! $this->parent_controller ) {

View File

@ -33,9 +33,9 @@ class WP_REST_Templates_Controller extends WP_REST_Controller {
*/ */
public function __construct( $post_type ) { public function __construct( $post_type ) {
$this->post_type = $post_type; $this->post_type = $post_type;
$this->namespace = 'wp/v2';
$obj = get_post_type_object( $post_type ); $obj = get_post_type_object( $post_type );
$this->rest_base = ! empty( $obj->rest_base ) ? $obj->rest_base : $obj->name; $this->rest_base = ! empty( $obj->rest_base ) ? $obj->rest_base : $obj->name;
$this->namespace = ! empty( $obj->rest_namespace ) ? $obj->rest_namespace : 'wp/v2';
} }
/** /**

View File

@ -914,15 +914,14 @@ class WP_REST_Terms_Controller extends WP_REST_Controller {
$post_type_links = array(); $post_type_links = array();
foreach ( $taxonomy_obj->object_type as $type ) { foreach ( $taxonomy_obj->object_type as $type ) {
$post_type_object = get_post_type_object( $type ); $rest_path = rest_get_route_for_post_type_items( $type );
if ( empty( $post_type_object->show_in_rest ) ) { if ( empty( $rest_path ) ) {
continue; continue;
} }
$rest_base = ! empty( $post_type_object->rest_base ) ? $post_type_object->rest_base : $post_type_object->name;
$post_type_links[] = array( $post_type_links[] = array(
'href' => add_query_arg( $this->rest_base, $term->term_id, rest_url( sprintf( 'wp/v2/%s', $rest_base ) ) ), 'href' => add_query_arg( $this->rest_base, $term->term_id, rest_url( $rest_path ) ),
); );
} }

View File

@ -16,7 +16,7 @@
* *
* @global string $wp_version * @global string $wp_version
*/ */
$wp_version = '5.9-alpha-51961'; $wp_version = '5.9-alpha-51962';
/** /**
* Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema. * Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.