REST API: Optimize rest_filter_response_by_context performance.
In [47758] a new function `rest_filter_response_by_context` was introduced to expand the JSON schema features supported by the context filtering mechanism. This commit improves the performance of that function by eliminating repetitive comparisons and loops. Additionally, it improves multi-type support for object + array types. Fixes #50700. Props dlh. git-svn-id: https://develop.svn.wordpress.org/trunk@48555 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
parent
b11757e4ed
commit
df8399dce5
@ -2052,15 +2052,28 @@ function rest_filter_response_by_context( $data, $schema, $context ) {
|
|||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$is_array_type = 'array' === $type || ( is_array( $type ) && in_array( 'array', $type, true ) );
|
||||||
|
$is_object_type = 'object' === $type || ( is_array( $type ) && in_array( 'object', $type, true ) );
|
||||||
|
|
||||||
|
if ( $is_array_type && $is_object_type ) {
|
||||||
|
if ( rest_is_array( $data ) ) {
|
||||||
|
$is_object_type = false;
|
||||||
|
} else {
|
||||||
|
$is_array_type = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$has_additional_properties = $is_object_type && isset( $schema['additionalProperties'] ) && is_array( $schema['additionalProperties'] );
|
||||||
|
|
||||||
foreach ( $data as $key => $value ) {
|
foreach ( $data as $key => $value ) {
|
||||||
$check = array();
|
$check = array();
|
||||||
|
|
||||||
if ( 'array' === $type || ( is_array( $type ) && in_array( 'array', $type, true ) ) ) {
|
if ( $is_array_type ) {
|
||||||
$check = isset( $schema['items'] ) ? $schema['items'] : array();
|
$check = isset( $schema['items'] ) ? $schema['items'] : array();
|
||||||
} elseif ( 'object' === $type || ( is_array( $type ) && in_array( 'object', $type, true ) ) ) {
|
} elseif ( $is_object_type ) {
|
||||||
if ( isset( $schema['properties'][ $key ] ) ) {
|
if ( isset( $schema['properties'][ $key ] ) ) {
|
||||||
$check = $schema['properties'][ $key ];
|
$check = $schema['properties'][ $key ];
|
||||||
} elseif ( isset( $schema['additionalProperties'] ) && is_array( $schema['additionalProperties'] ) ) {
|
} elseif ( $has_additional_properties ) {
|
||||||
$check = $schema['additionalProperties'];
|
$check = $schema['additionalProperties'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2070,6 +2083,12 @@ function rest_filter_response_by_context( $data, $schema, $context ) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ( ! in_array( $context, $check['context'], true ) ) {
|
if ( ! in_array( $context, $check['context'], true ) ) {
|
||||||
|
if ( $is_array_type ) {
|
||||||
|
// All array items share schema, so there's no need to check each one.
|
||||||
|
$data = array();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
if ( is_object( $data ) ) {
|
if ( is_object( $data ) ) {
|
||||||
unset( $data->$key );
|
unset( $data->$key );
|
||||||
} else {
|
} else {
|
||||||
|
@ -1348,7 +1348,7 @@ class Tests_REST_API extends WP_UnitTestCase {
|
|||||||
),
|
),
|
||||||
array( 'multi' => array( array( 'visible' => '1' ) ) ),
|
array( 'multi' => array( array( 'visible' => '1' ) ) ),
|
||||||
),
|
),
|
||||||
'grand child properties does not traverses missing context' => array(
|
'does not traverse missing context' => array(
|
||||||
array(
|
array(
|
||||||
'$schema' => 'http://json-schema.org/draft-04/schema#',
|
'$schema' => 'http://json-schema.org/draft-04/schema#',
|
||||||
'type' => 'object',
|
'type' => 'object',
|
||||||
@ -1391,6 +1391,126 @@ class Tests_REST_API extends WP_UnitTestCase {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
'object with no matching properties' => array(
|
||||||
|
array(
|
||||||
|
'$schema' => 'http://json-schema.org/draft-04/schema#',
|
||||||
|
'type' => 'object',
|
||||||
|
'properties' => array(
|
||||||
|
'a' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'edit' ),
|
||||||
|
),
|
||||||
|
'b' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'edit' ),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'a' => 'hi',
|
||||||
|
'b' => 'hello',
|
||||||
|
),
|
||||||
|
array(),
|
||||||
|
),
|
||||||
|
'array whose type does not match' => array(
|
||||||
|
array(
|
||||||
|
'$schema' => 'http://json-schema.org/draft-04/schema#',
|
||||||
|
'type' => 'object',
|
||||||
|
'properties' => array(
|
||||||
|
'arr' => array(
|
||||||
|
'type' => 'array',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
'items' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'edit' ),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'arr' => array( 'foo', 'bar', 'baz' ),
|
||||||
|
),
|
||||||
|
array( 'arr' => array() ),
|
||||||
|
),
|
||||||
|
'array and object type passed object' => array(
|
||||||
|
array(
|
||||||
|
'$schema' => 'http://json-schema.org/draft-04/schema#',
|
||||||
|
'type' => array( 'array', 'object' ),
|
||||||
|
'properties' => array(
|
||||||
|
'a' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
'b' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'items' => array(
|
||||||
|
'type' => 'object',
|
||||||
|
'context' => array( 'edit' ),
|
||||||
|
'properties' => array(
|
||||||
|
'a' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
'b' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'a' => 'foo',
|
||||||
|
'b' => 'bar',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'a' => 'foo',
|
||||||
|
'b' => 'bar',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'array and object type passed array' => array(
|
||||||
|
array(
|
||||||
|
'$schema' => 'http://json-schema.org/draft-04/schema#',
|
||||||
|
'type' => array( 'array', 'object' ),
|
||||||
|
'properties' => array(
|
||||||
|
'a' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
'b' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'items' => array(
|
||||||
|
'type' => 'object',
|
||||||
|
'context' => array( 'edit' ),
|
||||||
|
'properties' => array(
|
||||||
|
'a' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
'b' => array(
|
||||||
|
'type' => 'string',
|
||||||
|
'context' => array( 'view' ),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'a' => 'foo',
|
||||||
|
'b' => 'bar',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'a' => 'foo',
|
||||||
|
'b' => 'bar',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
array(),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user