REST API: Reuse previously-generated embedded objects when building collection response.
Store each generated embedded object in a temporary cache when querying for linked resources so that repeated links to the same resource do not trigger repeated queries or processing. Props TimothyBlynJacobs. Fixes #48838. git-svn-id: https://develop.svn.wordpress.org/trunk@47138 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
parent
10a5077524
commit
547a08ee03
@ -78,6 +78,14 @@ class WP_REST_Server {
|
|||||||
*/
|
*/
|
||||||
protected $route_options = array();
|
protected $route_options = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Caches embedded requests.
|
||||||
|
*
|
||||||
|
* @since 5.4.0
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
protected $embed_cache = array();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Instantiates the REST server.
|
* Instantiates the REST server.
|
||||||
*
|
*
|
||||||
@ -462,12 +470,14 @@ class WP_REST_Server {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ( $embed ) {
|
if ( $embed ) {
|
||||||
|
$this->embed_cache = array();
|
||||||
// Determine if this is a numeric array.
|
// Determine if this is a numeric array.
|
||||||
if ( wp_is_numeric_array( $data ) ) {
|
if ( wp_is_numeric_array( $data ) ) {
|
||||||
$data = array_map( array( $this, 'embed_links' ), $data );
|
$data = array_map( array( $this, 'embed_links' ), $data );
|
||||||
} else {
|
} else {
|
||||||
$data = $this->embed_links( $data );
|
$data = $this->embed_links( $data );
|
||||||
}
|
}
|
||||||
|
$this->embed_cache = array();
|
||||||
}
|
}
|
||||||
|
|
||||||
return $data;
|
return $data;
|
||||||
@ -588,24 +598,28 @@ class WP_REST_Server {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Run through our internal routing and serve.
|
if ( ! array_key_exists( $item['href'], $this->embed_cache ) ) {
|
||||||
$request = WP_REST_Request::from_url( $item['href'] );
|
// Run through our internal routing and serve.
|
||||||
if ( ! $request ) {
|
$request = WP_REST_Request::from_url( $item['href'] );
|
||||||
$embeds[] = array();
|
if ( ! $request ) {
|
||||||
continue;
|
$embeds[] = array();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Embedded resources get passed context=embed.
|
||||||
|
if ( empty( $request['context'] ) ) {
|
||||||
|
$request['context'] = 'embed';
|
||||||
|
}
|
||||||
|
|
||||||
|
$response = $this->dispatch( $request );
|
||||||
|
|
||||||
|
/** This filter is documented in wp-includes/rest-api/class-wp-rest-server.php */
|
||||||
|
$response = apply_filters( 'rest_post_dispatch', rest_ensure_response( $response ), $this, $request );
|
||||||
|
|
||||||
|
$this->embed_cache[ $item['href'] ] = $this->response_to_data( $response, false );
|
||||||
}
|
}
|
||||||
|
|
||||||
// Embedded resources get passed context=embed.
|
$embeds[] = $this->embed_cache[ $item['href'] ];
|
||||||
if ( empty( $request['context'] ) ) {
|
|
||||||
$request['context'] = 'embed';
|
|
||||||
}
|
|
||||||
|
|
||||||
$response = $this->dispatch( $request );
|
|
||||||
|
|
||||||
/** This filter is documented in wp-includes/rest-api/class-wp-rest-server.php */
|
|
||||||
$response = apply_filters( 'rest_post_dispatch', rest_ensure_response( $response ), $this, $request );
|
|
||||||
|
|
||||||
$embeds[] = $this->response_to_data( $response, false );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine if any real links were found.
|
// Determine if any real links were found.
|
||||||
|
@ -677,6 +677,108 @@ class Tests_REST_Server extends WP_Test_REST_TestCase {
|
|||||||
$this->assertEquals( 403, $up_data['data']['status'] );
|
$this->assertEquals( 403, $up_data['data']['status'] );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @ticket 48838
|
||||||
|
*/
|
||||||
|
public function test_link_embedding_clears_cache() {
|
||||||
|
$post_id = self::factory()->post->create();
|
||||||
|
|
||||||
|
$response = new WP_REST_Response();
|
||||||
|
$response->add_link( 'post', rest_url( 'wp/v2/posts/' . $post_id ), array( 'embeddable' => true ) );
|
||||||
|
|
||||||
|
$data = rest_get_server()->response_to_data( $response, true );
|
||||||
|
$this->assertArrayHasKey( 'post', $data['_embedded'] );
|
||||||
|
$this->assertCount( 1, $data['_embedded']['post'] );
|
||||||
|
|
||||||
|
wp_update_post(
|
||||||
|
array(
|
||||||
|
'ID' => $post_id,
|
||||||
|
'post_title' => 'My Awesome Title',
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
$data = rest_get_server()->response_to_data( $response, true );
|
||||||
|
$this->assertArrayHasKey( 'post', $data['_embedded'] );
|
||||||
|
$this->assertCount( 1, $data['_embedded']['post'] );
|
||||||
|
$this->assertEquals( 'My Awesome Title', $data['_embedded']['post'][0]['title']['rendered'] );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @ticket 48838
|
||||||
|
*/
|
||||||
|
public function test_link_embedding_cache() {
|
||||||
|
$response = new WP_REST_Response(
|
||||||
|
array(
|
||||||
|
'id' => 1,
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$response->add_link(
|
||||||
|
'author',
|
||||||
|
rest_url( 'wp/v2/users/1' ),
|
||||||
|
array( 'embeddable' => true )
|
||||||
|
);
|
||||||
|
$response->add_link(
|
||||||
|
'author',
|
||||||
|
rest_url( 'wp/v2/users/1' ),
|
||||||
|
array( 'embeddable' => true )
|
||||||
|
);
|
||||||
|
|
||||||
|
$mock = new MockAction();
|
||||||
|
add_filter( 'rest_post_dispatch', array( $mock, 'filter' ) );
|
||||||
|
|
||||||
|
$data = rest_get_server()->response_to_data( $response, true );
|
||||||
|
|
||||||
|
$this->assertArrayHasKey( '_embedded', $data );
|
||||||
|
$this->assertArrayHasKey( 'author', $data['_embedded'] );
|
||||||
|
$this->assertCount( 2, $data['_embedded']['author'] );
|
||||||
|
|
||||||
|
$this->assertCount( 1, $mock->get_events() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @ticket 48838
|
||||||
|
*/
|
||||||
|
public function test_link_embedding_cache_collection() {
|
||||||
|
$response = new WP_REST_Response(
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'id' => 1,
|
||||||
|
'_links' => array(
|
||||||
|
'author' => array(
|
||||||
|
array(
|
||||||
|
'href' => rest_url( 'wp/v2/users/1' ),
|
||||||
|
'embeddable' => true,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'id' => 2,
|
||||||
|
'_links' => array(
|
||||||
|
'author' => array(
|
||||||
|
array(
|
||||||
|
'href' => rest_url( 'wp/v2/users/1' ),
|
||||||
|
'embeddable' => true,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
$mock = new MockAction();
|
||||||
|
add_filter( 'rest_post_dispatch', array( $mock, 'filter' ) );
|
||||||
|
|
||||||
|
$data = rest_get_server()->response_to_data( $response, true );
|
||||||
|
|
||||||
|
$embeds = wp_list_pluck( $data, '_embedded' );
|
||||||
|
$this->assertCount( 2, $embeds );
|
||||||
|
$this->assertArrayHasKey( 'author', $embeds[0] );
|
||||||
|
$this->assertArrayHasKey( 'author', $embeds[1] );
|
||||||
|
|
||||||
|
$this->assertCount( 1, $mock->get_events() );
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Ensure embedding is a no-op without links in the data.
|
* Ensure embedding is a no-op without links in the data.
|
||||||
*/
|
*/
|
||||||
|
Loading…
Reference in New Issue
Block a user