Query: Expand the list of operators available to compare_key in WP_Meta_Query.

`compare_key`, introduced in #42409, previously supported only `=` and `LIKE`
operators. This changeset introduces a number of other operators: `!=`, `IN`,
`NOT IN`, `NOT LIKE`, `RLIKE`, `REGEXP`, `NOT REGEXP`, `EXISTS`, and `NOT EXISTS`
(the latter two aliased to `=` and `!=`, respectively). To support case-sensitive
regular expression key comparisons, the new `type_key` parameter will force
a MySQL `CAST` when 'BINARY' is passed.

Props soulseekah.
Fixes #43346.

git-svn-id: https://develop.svn.wordpress.org/trunk@46188 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
Boone Gorges 2019-09-19 15:02:20 +00:00
parent 7e69921a5a
commit 6eabc83b81
4 changed files with 392 additions and 30 deletions

View File

@ -100,6 +100,8 @@ class WP_Meta_Query {
* @since 3.2.0 * @since 3.2.0
* @since 4.2.0 Introduced support for naming query clauses by associative array keys. * @since 4.2.0 Introduced support for naming query clauses by associative array keys.
* @since 5.1.0 Introduced $compare_key clause parameter, which enables LIKE key matches. * @since 5.1.0 Introduced $compare_key clause parameter, which enables LIKE key matches.
* @since 5.3.0 Increased the number of operators available to $compare_key. Introduced $type_key,
* which enables the $key to be cast to a new data type for comparisons.
* *
* @param array $meta_query { * @param array $meta_query {
* Array of meta query clauses. When first-order clauses or sub-clauses use strings as * Array of meta query clauses. When first-order clauses or sub-clauses use strings as
@ -111,8 +113,13 @@ class WP_Meta_Query {
* Optional. An array of first-order clause parameters, or another fully-formed meta query. * Optional. An array of first-order clause parameters, or another fully-formed meta query.
* *
* @type string $key Meta key to filter by. * @type string $key Meta key to filter by.
* @type string $compare_key MySQL operator used for comparing the $key. Accepts '=' and 'LIKE'. * @type string $compare_key MySQL operator used for comparing the $key. Accepts '=', '!='
* Default '='. * 'LIKE', 'NOT LIKE', 'IN', 'NOT IN', 'REGEXP', 'NOT REGEXP', 'RLIKE',
* 'EXISTS' (alias of '=') or 'NOT EXISTS' (alias of '!=').
* Default is 'IN' when `$key` is an array, '=' otherwise.
* @type string $type_key MySQL data type that the meta_key column will be CAST to for
* comparisons. Accepts 'BINARY' for case-sensitive regular expression
* comparisons. Default is ''.
* @type string $value Meta value to filter by. * @type string $value Meta value to filter by.
* @type string $compare MySQL operator used for comparing the $value. Accepts '=', * @type string $compare MySQL operator used for comparing the $value. Accepts '=',
* '!=', '>', '>=', '<', '<=', 'LIKE', 'NOT LIKE', * '!=', '>', '>=', '<', '<=', 'LIKE', 'NOT LIKE',
@ -239,7 +246,7 @@ class WP_Meta_Query {
* the rest of the meta_query). * the rest of the meta_query).
*/ */
$primary_meta_query = array(); $primary_meta_query = array();
foreach ( array( 'key', 'compare', 'type', 'compare_key' ) as $key ) { foreach ( array( 'key', 'compare', 'type', 'compare_key', 'type_key' ) as $key ) {
if ( ! empty( $qv[ "meta_$key" ] ) ) { if ( ! empty( $qv[ "meta_$key" ] ) ) {
$primary_meta_query[ $key ] = $qv[ "meta_$key" ]; $primary_meta_query[ $key ] = $qv[ "meta_$key" ];
} }
@ -498,34 +505,40 @@ class WP_Meta_Query {
$clause['compare'] = isset( $clause['value'] ) && is_array( $clause['value'] ) ? 'IN' : '='; $clause['compare'] = isset( $clause['value'] ) && is_array( $clause['value'] ) ? 'IN' : '=';
} }
if ( ! in_array( $non_numeric_operators = array(
$clause['compare'], '=',
array( '!=',
'=', 'LIKE',
'!=', 'NOT LIKE',
'>', 'IN',
'>=', 'NOT IN',
'<', 'EXISTS',
'<=', 'NOT EXISTS',
'LIKE', 'RLIKE',
'NOT LIKE', 'REGEXP',
'IN', 'NOT REGEXP',
'NOT IN', );
'BETWEEN',
'NOT BETWEEN', $numeric_operators = array(
'EXISTS', '>',
'NOT EXISTS', '>=',
'REGEXP', '<',
'NOT REGEXP', '<=',
'RLIKE', 'BETWEEN',
) 'NOT BETWEEN',
) ) { );
if ( ! in_array( $clause['compare'], $non_numeric_operators, true ) && ! in_array( $clause['compare'], $numeric_operators, true ) ) {
$clause['compare'] = '='; $clause['compare'] = '=';
} }
if ( isset( $clause['compare_key'] ) && 'LIKE' === strtoupper( $clause['compare_key'] ) ) { if ( isset( $clause['compare_key'] ) ) {
$clause['compare_key'] = strtoupper( $clause['compare_key'] ); $clause['compare_key'] = strtoupper( $clause['compare_key'] );
} else { } else {
$clause['compare_key'] = isset( $clause['key'] ) && is_array( $clause['key'] ) ? 'IN' : '=';
}
if ( ! in_array( $clause['compare_key'], $non_numeric_operators, true ) ) {
$clause['compare_key'] = '='; $clause['compare_key'] = '=';
} }
@ -594,11 +607,79 @@ class WP_Meta_Query {
if ( 'NOT EXISTS' === $meta_compare ) { if ( 'NOT EXISTS' === $meta_compare ) {
$sql_chunks['where'][] = $alias . '.' . $this->meta_id_column . ' IS NULL'; $sql_chunks['where'][] = $alias . '.' . $this->meta_id_column . ' IS NULL';
} else { } else {
if ( 'LIKE' === $meta_compare_key ) { /**
$sql_chunks['where'][] = $wpdb->prepare( "$alias.meta_key LIKE %s", '%' . $wpdb->esc_like( trim( $clause['key'] ) ) . '%' ); * In joined clauses negative operators have to be nested into a
} else { * NOT EXISTS clause and flipped, to avoid returning records with
$sql_chunks['where'][] = $wpdb->prepare( "$alias.meta_key = %s", trim( $clause['key'] ) ); * matching post IDs but different meta keys. Here we prepare the
* nested clause.
*/
if ( in_array( $meta_compare_key, array( '!=', 'NOT IN', 'NOT LIKE', 'NOT EXISTS', 'NOT REGEXP' ), true ) ) {
// Negative clauses may be reused.
$i = count( $this->table_aliases );
$subquery_alias = $i ? 'mt' . $i : $this->meta_table;
$this->table_aliases[] = $subquery_alias;
$meta_compare_string_start = 'NOT EXISTS (';
$meta_compare_string_start .= "SELECT 1 FROM $wpdb->postmeta $subquery_alias ";
$meta_compare_string_start .= "WHERE $subquery_alias.post_ID = $alias.post_ID ";
$meta_compare_string_end = 'LIMIT 1';
$meta_compare_string_end .= ')';
} }
switch ( $meta_compare_key ) {
case '=':
case 'EXISTS':
$where = $wpdb->prepare( "$alias.meta_key = %s", trim( $clause['key'] ) ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared
break;
case 'LIKE':
$meta_compare_value = '%' . $wpdb->esc_like( trim( $clause['key'] ) ) . '%';
$where = $wpdb->prepare( "$alias.meta_key LIKE %s", $meta_compare_value ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared
break;
case 'IN':
$meta_compare_string = "$alias.meta_key IN (" . substr( str_repeat( ',%s', count( $clause['key'] ) ), 1 ) . ')';
$where = $wpdb->prepare( $meta_compare_string, $clause['key'] ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
break;
case 'RLIKE':
case 'REGEXP':
$operator = $meta_compare_key;
if ( isset( $clause['type_key'] ) && 'BINARY' === strtoupper( $clause['type_key'] ) ) {
$cast = 'BINARY';
} else {
$cast = '';
}
$where = $wpdb->prepare( "$alias.meta_key $operator $cast %s", trim( $clause['key'] ) ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared
break;
case '!=':
case 'NOT EXISTS':
$meta_compare_string = $meta_compare_string_start . "AND $subquery_alias.meta_key = %s " . $meta_compare_string_end;
$where = $wpdb->prepare( $meta_compare_string, $clause['key'] ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
break;
case 'NOT LIKE':
$meta_compare_string = $meta_compare_string_start . "AND $subquery_alias.meta_key LIKE %s " . $meta_compare_string_end;
$meta_compare_value = '%' . $wpdb->esc_like( trim( $clause['key'] ) ) . '%';
$where = $wpdb->prepare( $meta_compare_string, $meta_compare_value ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
break;
case 'NOT IN':
$array_subclause = '(' . substr( str_repeat( ',%s', count( $clause['key'] ) ), 1 ) . ') ';
$meta_compare_string = $meta_compare_string_start . "AND $subquery_alias.meta_key IN " . $array_subclause . $meta_compare_string_end;
$where = $wpdb->prepare( $meta_compare_string, $clause['key'] ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
break;
case 'NOT REGEXP':
$operator = $meta_compare_key;
if ( isset( $clause['type_key'] ) && 'BINARY' === strtoupper( $clause['type_key'] ) ) {
$cast = 'BINARY';
} else {
$cast = '';
}
$meta_compare_string = $meta_compare_string_start . "AND $subquery_alias.meta_key REGEXP $cast %s " . $meta_compare_string_end;
$where = $wpdb->prepare( $meta_compare_string, $clause['key'] ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
break;
}
$sql_chunks['where'][] = $where;
} }
} }

View File

@ -614,6 +614,7 @@ class WP_Query {
* @since 4.6.0 Added 'post_name__in' support for `$orderby`. Introduced the `$lazy_load_term_meta` argument. * @since 4.6.0 Added 'post_name__in' support for `$orderby`. Introduced the `$lazy_load_term_meta` argument.
* @since 4.9.0 Introduced the `$comment_count` parameter. * @since 4.9.0 Introduced the `$comment_count` parameter.
* @since 5.1.0 Introduced the `$meta_compare_key` parameter. * @since 5.1.0 Introduced the `$meta_compare_key` parameter.
* @since 5.3.0 Introduced the `$meta_type_key` parameter.
* *
* @param string|array $query { * @param string|array $query {
* Optional. Array or string of Query parameters. * Optional. Array or string of Query parameters.
@ -655,6 +656,7 @@ class WP_Query {
* @type array $meta_query An associative array of WP_Meta_Query arguments. See WP_Meta_Query. * @type array $meta_query An associative array of WP_Meta_Query arguments. See WP_Meta_Query.
* @type string $meta_value Custom field value. * @type string $meta_value Custom field value.
* @type int $meta_value_num Custom field value number. * @type int $meta_value_num Custom field value number.
* @type string $meta_type_key Cast for 'meta_key'. See WP_Meta_Query::construct().
* @type int $menu_order The menu order of the posts. * @type int $menu_order The menu order of the posts.
* @type int $monthnum The two-digit month. Default empty. Accepts numbers 1-12. * @type int $monthnum The two-digit month. Default empty. Accepts numbers 1-12.
* @type string $name Post slug. * @type string $name Post slug.

View File

@ -737,6 +737,42 @@ class Tests_Meta_Query extends WP_UnitTestCase {
$this->assertSame( 1, substr_count( $sql['where'], "$wpdb->postmeta.meta_value =" ) ); $this->assertSame( 1, substr_count( $sql['where'], "$wpdb->postmeta.meta_value =" ) );
} }
/**
* Verifies only that meta_type_key is passed. See query/metaQuery.php for more complete tests.
*
* @ticket 43446
*/
public function test_meta_type_key_should_be_passed_to_meta_query() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'AAA_FOO_AAA', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_aaa', 'abc' );
$q = new WP_Query(
array(
'meta_key' => 'AAA_foo_.*',
'meta_compare_key' => 'REGEXP',
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[0], $posts[2] ), $q->posts );
$q = new WP_Query(
array(
'meta_key' => 'AAA_FOO_.*',
'meta_compare_key' => 'REGEXP',
'meta_type_key' => 'BINARY',
'fields' => 'ids',
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[0] ), $q->posts );
}
/** /**
* This is the clause that ensures that empty arrays are not valid queries. * This is the clause that ensures that empty arrays are not valid queries.
*/ */

View File

@ -1922,4 +1922,247 @@ class Tests_Query_MetaQuery extends WP_UnitTestCase {
$this->assertEqualSets( array( $posts[0] ), $q->posts ); $this->assertEqualSets( array( $posts[0] ), $q->posts );
} }
/**
* @ticket 43446
*/
public function test_compare_key_not_equals() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'aaa_foo_aaa', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_ccc', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => '!=',
'key' => 'aaa_foo_bbb',
'value' => 'abc',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[0], $posts[1] ), $q->posts );
}
/**
* @ticket 43446
*/
public function test_compare_key_not_like() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'aaa_foo_aaa', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_ccc', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'NOT LIKE',
'key' => 'aaa_bar',
'value' => 'abc',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[0], $posts[2] ), $q->posts );
}
/**
* @ticket 43446
*/
public function test_compare_key_in() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'aaa_foo_aaa', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'IN',
'key' => array( 'aaa_foo_bbb', 'aaa_bar_aaa' ),
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[1], $posts[2] ), $q->posts );
}
/**
* @ticket 43446
*/
public function test_compare_key_not_in() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'aaa_foo_aaa', 'abc' );
add_post_meta( $posts[0], 'aaa_foo_ddd', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_ccc', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'NOT IN',
'key' => array( 'aaa_foo_bbb', 'aaa_foo_ddd' ),
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[1] ), $q->posts );
}
/**
* @ticket 43446
*/
public function test_compare_key_not_exists() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'aaa_foo_aaa', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_ccc', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'NOT EXISTS',
'key' => 'aaa_foo_bbb',
'value' => 'abc',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[0], $posts[1] ), $q->posts );
}
/**
* @ticket 43446
*/
public function test_compare_key_exists() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'aaa_foo_aaa', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_ccc', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'EXISTS',
'key' => 'aaa_foo_bbb',
'value' => 'abc',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[2] ), $q->posts );
}
/**
* @ticket 43446
*/
public function test_compare_key_regexp_rlike() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'AAA_FOO_AAA', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_aaa', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'REGEXP',
'key' => 'AAA_foo_.*',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[0], $posts[2] ), $q->posts );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'RLIKE',
'key' => 'AAA_FOO_.*',
'type_key' => 'BINARY',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[0] ), $q->posts );
}
/**
* @ticket 43446
*/
public function test_compare_key_not_regexp() {
$posts = self::factory()->post->create_many( 3 );
add_post_meta( $posts[0], 'AAA_FOO_AAA', 'abc' );
add_post_meta( $posts[0], 'AAA_foo_AAA', 'abc' );
add_post_meta( $posts[1], 'aaa_bar_aaa', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_bbb', 'abc' );
add_post_meta( $posts[2], 'aaa_foo_aaa', 'abc' );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'NOT REGEXP',
'key' => 'AAA_foo_.*',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[1] ), $q->posts );
$q = new WP_Query(
array(
'meta_query' => array(
array(
'compare_key' => 'NOT REGEXP',
'key' => 'AAA_FOO_.*',
'type_key' => 'BINARY',
),
),
'fields' => 'ids',
)
);
$this->assertEqualSets( array( $posts[1], $posts[2] ), $q->posts );
}
} }