Skip to content

Commit d3dfb5b

Browse files
committed
Add new function for arrays: PgArrayILikeFunction
1 parent 1b90d71 commit d3dfb5b

File tree

4 files changed

+117
-0
lines changed

4 files changed

+117
-0
lines changed

README.md

+15
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ Currently the plugin supports array, hstore and json fields as well as some quer
2222
* [Is Empty or Contains](#is-empty-or-contains)
2323
* [Equals](#equals)
2424
* [Not Equals](#not-equals)
25+
* [Ilike](#ilike)
2526
* [Hstore](#hstore)
2627
* [Grails 2.2.5 and 2.3.1+](#grails-225-and-231)
2728
* [Old Grails versions](#old-grails-versions)
@@ -291,6 +292,20 @@ def result = Like.withCriteria {
291292
}
292293
```
293294

295+
#### ILike
296+
297+
With this criteria you can get all the rows that are ilike to a value. To use it just use the new criteria `pgArrayIlike`.
298+
299+
It only can be used on arrays of string.
300+
301+
It uses the ilike syntaxis, so you can do for example:
302+
303+
```groovy
304+
def result = Like.withCriteria {
305+
pgArrayIlike 'favoriteMovies', "%tarwar%"
306+
}
307+
```
308+
294309

295310
### Hstore
296311

src/groovy/net/kaleidos/hibernate/postgresql/criteria/ArrayCriterias.groovy

+22
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ package net.kaleidos.hibernate.postgresql.criteria
22

33
import grails.orm.HibernateCriteriaBuilder
44
import net.kaleidos.hibernate.criterion.array.PgArrayExpression
5+
import net.kaleidos.hibernate.criterion.array.PgArrayILikeFunction
56
import net.kaleidos.hibernate.criterion.array.PgEmptinessExpression
67

78
class ArrayCriterias {
@@ -15,6 +16,7 @@ class ArrayCriterias {
1516
addIsEmptyOrContainsOperator()
1617
addEqualsOperator()
1718
addNotEqualsOperator()
19+
addILikeOperator()
1820
}
1921

2022
private void addContainsOperator() {
@@ -160,6 +162,26 @@ class ArrayCriterias {
160162
}
161163
}
162164

165+
private void addILikeOperator() {
166+
/**
167+
* Creates a "ilike in native array" Criterion based on the specified property name and value
168+
* @param propertyName The property name
169+
* @param propertyValue The property value
170+
* @return A Criterion instance
171+
*/
172+
HibernateCriteriaBuilder.metaClass.pgArrayIlike = { String propertyName, String propertyValue ->
173+
if (!validateSimpleExpression()) {
174+
throwRuntimeException(new IllegalArgumentException("Call to [pgArrayIlike] with propertyName [" +
175+
propertyName + "] and value [" + propertyValue + "] not allowed here."))
176+
}
177+
178+
propertyName = calculatePropertyName(propertyName)
179+
propertyValue = calculatePropertyValue(propertyValue)
180+
181+
return addToCriteria(new PgArrayILikeFunction(propertyName, propertyValue))
182+
}
183+
}
184+
163185
private void addNotEqualsOperator() {
164186
/**
165187
* Creates a "not equals in native array" Criterion based on the specified property name and value
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
package net.kaleidos.hibernate.criterion.array;
2+
3+
import net.kaleidos.hibernate.usertype.ArrayType;
4+
import net.kaleidos.hibernate.utils.PgArrayUtils;
5+
import org.hibernate.Criteria;
6+
import org.hibernate.HibernateException;
7+
import org.hibernate.annotations.common.util.StringHelper;
8+
import org.hibernate.criterion.CriteriaQuery;
9+
import org.hibernate.criterion.Criterion;
10+
import org.hibernate.engine.spi.TypedValue;
11+
import org.hibernate.type.CustomType;
12+
import org.hibernate.type.Type;
13+
import org.hibernate.type.StringType;
14+
15+
/**
16+
* Constrains a property in an array
17+
*/
18+
public class PgArrayILikeFunction implements Criterion {
19+
20+
private static final long serialVersionUID = 7475136611436979257L;
21+
22+
private final String propertyName;
23+
private final String value;
24+
25+
protected PgArrayILikeFunction(String propertyName, String value) {
26+
this.propertyName = propertyName;
27+
this.value = value;
28+
}
29+
30+
@Override
31+
public String toSqlString(Criteria criteria, CriteriaQuery criteriaQuery) throws HibernateException {
32+
String[] columns = StringHelper.suffix(criteriaQuery.findColumns(propertyName, criteria), "");
33+
for (int i=0; i<columns.length; i++) {
34+
columns[i] = "text(" + columns[i]+ ") ilike ?";
35+
}
36+
return StringHelper.join( " and ", columns);
37+
}
38+
39+
@Override
40+
public TypedValue[] getTypedValues(Criteria criteria, CriteriaQuery criteriaQuery) throws HibernateException {
41+
return new TypedValue[]{
42+
new TypedValue(new StringType(), value)
43+
};
44+
}
45+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
package net.kaleidos.hibernate.array
2+
3+
import spock.lang.Specification
4+
import spock.lang.Unroll
5+
import test.criteria.array.Like
6+
7+
class PgILikeCriteriaTestServiceIntegrationSpec extends Specification {
8+
9+
def pgArrayTestSearchService
10+
11+
@Unroll
12+
void "check ilike for #movie in an array of strings"() {
13+
setup:
14+
new Like(favoriteMovies: ["The Matrix", "The Lord of the Rings"]).save()
15+
new Like(favoriteMovies: ["Spiderman", "Blade Runner", "Starwars"]).save()
16+
new Like(favoriteMovies: ["Starwars"]).save()
17+
new Like(favoriteMovies: ["Romeo & Juliet", "Blade Runner", "The Lord of the Rings"]).save()
18+
new Like(favoriteMovies: []).save()
19+
20+
when:
21+
def result = pgArrayTestSearchService.search('favoriteMovies', 'pgArrayIlike', movie)
22+
23+
then:
24+
result.size() == resultSize
25+
26+
where:
27+
movie | resultSize
28+
"%tarwar%" | 2
29+
"%ider%" | 1
30+
"%Suspects%" | 0
31+
"" | 0
32+
"%" | 5
33+
}
34+
35+
}

0 commit comments

Comments
 (0)