/*
* Copyright (c) 2008-2017, Hazelcast, Inc. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.hazelcast.query.impl.extractor.predicates;
import com.hazelcast.config.InMemoryFormat;
import com.hazelcast.query.Predicates;
import com.hazelcast.query.impl.extractor.AbstractExtractionTest;
import com.hazelcast.test.annotation.ParallelTest;
import com.hazelcast.test.annotation.QuickTest;
import org.junit.Test;
import org.junit.experimental.categories.Category;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import java.util.Collection;
import static com.hazelcast.config.InMemoryFormat.BINARY;
import static com.hazelcast.config.InMemoryFormat.OBJECT;
import static com.hazelcast.query.impl.extractor.AbstractExtractionSpecification.Index.NO_INDEX;
import static com.hazelcast.query.impl.extractor.AbstractExtractionSpecification.Index.ORDERED;
import static com.hazelcast.query.impl.extractor.AbstractExtractionSpecification.Index.UNORDERED;
import static com.hazelcast.query.impl.extractor.AbstractExtractionSpecification.Multivalue.ARRAY;
import static com.hazelcast.query.impl.extractor.AbstractExtractionSpecification.Multivalue.LIST;
import static com.hazelcast.query.impl.extractor.predicates.CollectionDataStructure.Person;
import static com.hazelcast.query.impl.extractor.predicates.CollectionDataStructure.limb;
import static com.hazelcast.query.impl.extractor.predicates.CollectionDataStructure.person;
import static java.util.Arrays.asList;
/**
* Tests whether all predicates work with the extraction in collections.
* Each predicate is tested with and without the extraction including "any" operator.
* Extraction with the "any" operator may return multiple results, thus each predicate has to handle it.
*
* Extraction mechanism: IN-BUILT REFLECTION EXTRACTION
*
* The trick here is that each multi-value attribute in the used data structure is present as both an array and as
* a collection. For more details have a look at CollectionDataStructure.
* This test is parametrised, so that it's executed for both options too (arrays and collections).
* Let's have a look at the following path: 'limbs_[1].power'.
* limbs_[1].power is unfolded to: limbs_array[1].power and limbs_list[1].power in two separate tests run.
* In this way we are testing extraction in collection and arrays making sure that the default behavior is consistent
* for both extraction sources!
*
* It's not the only parametrisation in this test; the other ones are:
* - each test is executed separately for BINARY and OBJECT in memory format
* - each test is executed separately having each query using NO_INDEX, UNORDERED_INDEX and ORDERED_INDEX.
* In this way we are spec-testing most of the reasonable combinations of the configuration of map & extraction.
*/
@RunWith(Parameterized.class)
@Category({QuickTest.class, ParallelTest.class})
public class CollectionAllPredicatesReflectionTest extends AbstractExtractionTest {
private static final Person BOND = person(limb("left", 49), limb("right", 51));
private static final Person KRUEGER = person(limb("links", 27), limb("rechts", 29));
public CollectionAllPredicatesReflectionTest(InMemoryFormat inMemoryFormat, Index index, Multivalue multivalue) {
super(inMemoryFormat, index, multivalue);
}
@Test
public void equals_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.equal("limbs_[1].power", 51), mv),
Expected.of(BOND));
}
@Test
public void equals_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.equal("limbs_[any].power", 51), mv),
Expected.of(BOND));
}
@Test
public void between_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.between("limbs_[1].power", 40, 60), mv),
Expected.of(BOND));
}
@Test
public void between_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.between("limbs_[any].power", 20, 40), mv),
Expected.of(KRUEGER));
}
@Test
public void greater_less_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.lessEqual("limbs_[1].power", 29), mv),
Expected.of(KRUEGER));
}
@Test
public void greater_less_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.lessEqual("limbs_[any].power", 27), mv),
Expected.of(KRUEGER));
}
@Test
public void in_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.in("limbs_[1].power", 28, 29, 30), mv),
Expected.of(KRUEGER));
}
@Test
public void in_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.in("limbs_[any].power", 26, 27, 51), mv),
Expected.of(BOND, KRUEGER));
}
@Test
public void notEqual_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.notEqual("limbs_[1].power", 29), mv),
Expected.of(BOND));
}
@Test
public void notEqual_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.notEqual("limbs_[any].power", 27), mv),
Expected.of(BOND));
}
@Test
public void like_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.like("limbs_[1].name", "recht_"), mv),
Expected.of(KRUEGER));
}
@Test
public void like_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.like("limbs_[any].name", "le%"), mv),
Expected.of(BOND));
}
@Test
public void ilike_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.ilike("limbs_[1].name", "REcht_"), mv),
Expected.of(KRUEGER));
}
@Test
public void ilike_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.ilike("limbs_[any].name", "LE%"), mv),
Expected.of(BOND));
}
@Test
public void regex_predicate() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.regex("limbs_[1].name", "ri.*"), mv),
Expected.of(BOND));
}
@Test
public void regex_predicate_reduced() {
execute(Input.of(BOND, KRUEGER),
Query.of(Predicates.regex("limbs_[any].name", "li.*"), mv),
Expected.of(KRUEGER));
}
@Parameterized.Parameters(name = "{index}: {0}, {1}, {2}")
public static Collection<Object[]> data() {
return axes(
asList(BINARY, OBJECT),
asList(NO_INDEX, UNORDERED, ORDERED),
asList(ARRAY, LIST)
);
}
}