Classes in this File | Line Coverage | Branch Coverage | Complexity | ||||
Criteria |
|
| 2.671232876712329;2.671 | ||||
Criteria$1 |
|
| 2.671232876712329;2.671 |
1 | /** | |
2 | * Copyright 2005-2011 The Kuali Foundation | |
3 | * | |
4 | * Licensed under the Educational Community License, Version 2.0 (the "License"); | |
5 | * you may not use this file except in compliance with the License. | |
6 | * You may obtain a copy of the License at | |
7 | * | |
8 | * http://www.opensource.org/licenses/ecl2.php | |
9 | * | |
10 | * Unless required by applicable law or agreed to in writing, software | |
11 | * distributed under the License is distributed on an "AS IS" BASIS, | |
12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
13 | * See the License for the specific language governing permissions and | |
14 | * limitations under the License. | |
15 | */ | |
16 | package org.kuali.rice.core.framework.persistence.jpa.criteria; | |
17 | ||
18 | import org.kuali.rice.core.api.util.Truth; | |
19 | import org.kuali.rice.core.api.util.type.TypeUtils; | |
20 | ||
21 | import javax.persistence.Query; | |
22 | import java.util.ArrayList; | |
23 | import java.util.Arrays; | |
24 | import java.util.Collection; | |
25 | import java.util.HashMap; | |
26 | import java.util.Iterator; | |
27 | import java.util.LinkedHashMap; | |
28 | import java.util.List; | |
29 | import java.util.Map; | |
30 | import java.util.regex.Pattern; | |
31 | ||
32 | /** | |
33 | * A criteria builder for JPQL Query objects. | |
34 | * | |
35 | * <p>All entities in queries generated by this class will be given an alias, and an initial entity (with an initial alias) | |
36 | * is mandatory. If no alias is given or if a constructor without an alias parameter is used, a new alias name will be | |
37 | * auto-generated. | |
38 | * | |
39 | * <p>The interpretation of the initial entity depends on what type of query is intended. If a SELECT query is desired, | |
40 | * the initial entity will be the first to appear in the FROM clause. If a DELETE query is desired, the initial | |
41 | * entity will be the one that appears in the DELETE FROM clause. If an UPDATE query is desired, the initial entity | |
42 | * will be the one that appears in the UPDATE clause. | |
43 | * | |
44 | * <p>Most of the methods of this class rely on String expressions representing entity properties, or functions | |
45 | * acting on constants or entity properties. If the expression contains functions or if | |
46 | * you wish to reference the property of an entity other than the first aliased entity of the query, then the | |
47 | * String expression must include the alias of the entity in the format __JPA_ALIAS[[index]]__ or the format | |
48 | * __JPA_ALIAS[['alias']]__ , where "index" is a numeric index pointing to the Nth alias of the current query in the | |
49 | * order that the aliases were added to the query (starting at zero), and where "'alias'" is the String name of the | |
50 | * entity alias enclosed in single quotes. | |
51 | * | |
52 | * <p>If the String expression as defined above does not contain custom JPA aliases, then the initial alias of the current | |
53 | * Criteria (and a '.' to separate the alias from the expression) will be prepended to the expression automatically. | |
54 | * | |
55 | * <p>If the current Criteria is intended for use as a sub-query, then aliases of the parent Criteria can be referred to | |
56 | * by name (so long as they do not conflict with the alias names in the sub-Criteria) or by specifying the bitwise NOT of | |
57 | * the alias index in the parent Criteria (-1 for 0, -2 for 1, etc.). When including a sub-query in a parent query, the | |
58 | * parent Criteria will handle the updating of the alias indexes and will auto-generate new aliases as needed if any of | |
59 | * the sub-query's defined aliases are duplicates of those in the parent query. | |
60 | * | |
61 | * <p>For Criteria objects intended to be ANDed or ORed with an existing Criteria, it is assumed that any named or indexed | |
62 | * aliases refer to exactly the same aliases as in the existing Criteria. Any auto-prepended aliases as described above | |
63 | * will always be referred to by index, so it is safe for them to be included in the Criteria to be ANDed/ORed with the | |
64 | * existing one. Both of these details also apply when adding new Criteria to the HAVING clause of an existing Criteria, or | |
65 | * when inserting the negation of another Criteria's conditions into an existing Criteria as a "NOT (...)" expression. | |
66 | * | |
67 | * <p>Also, during query construction, the Criteria API will automatically include substrings in the format | |
68 | * __JPA_PARAM[['parameter']]__ when adding input parameters, where 'parameter' is the auto-generated name of the | |
69 | * parameter. Care should be taken to ensure that __JPA_ALIAS[[...]]__ and __JPA_PARAM[[...]]__ expressions are not | |
70 | * being added to queries that wish to interpret such String patterns literally. | |
71 | * | |
72 | * <p>Note that Criteria instances are not thread-safe, so external synchronization is necessary if operating on them | |
73 | * from multiple threads. | |
74 | * | |
75 | * <p>TODO: Verify if any other features need to be added to this API. | |
76 | * | |
77 | * @author Kuali Rice Team (rice.collab@kuali.org) | |
78 | */ | |
79 | @SuppressWarnings("unchecked") | |
80 | public class Criteria { | |
81 | ||
82 | 0 | private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(Criteria.class); |
83 | 0 | private static final Pattern APOS_PAT = Pattern.compile("'"); |
84 | 0 | private static final String[] LOOKUP_WILDCARDS = {"*", "?"}; |
85 | 0 | private static final String[] ESCAPED_LOOKUP_WILDCARDS = {"\\*", "\\?"}; |
86 | 0 | private static final char[] JPQL_WILDCARDS = {'%', '_'}; |
87 | ||
88 | /** The String representing the beginning of a by-name or by-index reference to an entity alias. */ | |
89 | public static final String JPA_ALIAS_PREFIX = "__JPA_ALIAS[["; | |
90 | /** The String representing the beginning of a by-name reference to an input parameter. */ | |
91 | public static final String JPA_PARAM_PREFIX = "__JPA_PARAM[["; | |
92 | /** The String representing the termination of by-name/index references to aliases or input parameters. */ | |
93 | public static final String JPA_ALIAS_SUFFIX = "]]__"; | |
94 | ||
95 | private static final String JPA_INITIAL_ALIAS = JPA_ALIAS_PREFIX + "0" + JPA_ALIAS_SUFFIX; | |
96 | private static final String JPA_PARENT_INITIAL_ALIAS = JPA_ALIAS_PREFIX + "-1" + JPA_ALIAS_SUFFIX; | |
97 | ||
98 | private static final String JPA_PARAM_PREFIX_WITH_COLON = ":" + JPA_PARAM_PREFIX; | |
99 | 0 | private static final int ALIAS_PREFIX_LEN = JPA_ALIAS_PREFIX.length(); |
100 | 0 | private static final int PARAM_PREFIX_LEN = JPA_PARAM_PREFIX.length(); |
101 | 0 | private static final int ALIAS_SUFFIX_LEN = JPA_ALIAS_SUFFIX.length(); |
102 | ||
103 | private Integer searchLimit; | |
104 | ||
105 | private int generatedAliasCount; | |
106 | ||
107 | private int generatedBindParamCount; | |
108 | ||
109 | 0 | private boolean distinct = false; |
110 | ||
111 | 0 | private final List<String> entityAliases = new ArrayList<String>(); |
112 | ||
113 | 0 | private final List<String> indexedAliasPlaceholders = new ArrayList<String>(); |
114 | ||
115 | 0 | private final List<String> namedAliasPlaceholders = new ArrayList<String>(); |
116 | ||
117 | 0 | private final Map<String,Integer> aliasIndexes = new HashMap<String,Integer>(); |
118 | ||
119 | 0 | private final StringBuilder selectClause = new StringBuilder(); |
120 | ||
121 | 0 | private final StringBuilder fromClause = new StringBuilder(); |
122 | ||
123 | 0 | private final StringBuilder whereClause = new StringBuilder(50); |
124 | ||
125 | 0 | private final StringBuilder groupByClause = new StringBuilder(); |
126 | ||
127 | 0 | private final StringBuilder havingClause = new StringBuilder(); |
128 | ||
129 | 0 | private final StringBuilder orderByClause = new StringBuilder(); |
130 | ||
131 | 0 | private final StringBuilder setClause = new StringBuilder(); |
132 | ||
133 | private final String initialEntityName; | |
134 | ||
135 | 0 | protected Map<String, Object> params = new LinkedHashMap<String, Object>(); |
136 | ||
137 | /** | |
138 | * Constructs a new Criteria instance that includes the given initial entity. An alias will be | |
139 | * auto-generated for this entity, and the entity's alias will be included in the SELECT clause if this | |
140 | * Criteria is intended to be used as a SELECT query. | |
141 | * | |
142 | * @param entityName The class name of the initial JPA entity. | |
143 | */ | |
144 | public Criteria(String entityName) { | |
145 | 0 | this(entityName, "a", true); |
146 | 0 | } |
147 | ||
148 | /** | |
149 | * Constructs a new Criteria instance that includes the given initial entity. The given alias | |
150 | * will be used for this entity if it is non-null, and the alias will be included in the SELECT clause | |
151 | * if this Criteria is intended to be used as a SELECT query. | |
152 | * | |
153 | * @param entityName The class name of the initial JPA entity. | |
154 | * @param alias The alias to use for this entity; if it is null, a new alias will be auto-generated. | |
155 | */ | |
156 | public Criteria(String entityName, String alias) { | |
157 | 0 | this(entityName, alias, true); |
158 | 0 | } |
159 | ||
160 | /** | |
161 | * Constructs a new Criteria instance that includes the given initial entity. An alias will be | |
162 | * auto-generated for this entity, and the extra parameter indicates whether this alias should also be | |
163 | * added to the query's SELECT clause (if this Criteria is intended to be used as a SELECT query). | |
164 | * | |
165 | * @param entityName The class name of the initial JPA entity. | |
166 | * @param includeEntityInSelect Indicates whether this entity's alias should be added to the SELECT clause. | |
167 | */ | |
168 | public Criteria(String entityName, boolean includeEntityInSelect) { | |
169 | 0 | this(entityName, "a", includeEntityInSelect); |
170 | 0 | } |
171 | ||
172 | /** | |
173 | * Constructs a new Criteria instance that includes the given initial entity. The given alias | |
174 | * will be used for this entity if it is non-null, and the extra parameter indicates whether the alias | |
175 | * should be added to the SELECT clause (if this Criteria is intended to be used as a SELECT query). | |
176 | * | |
177 | * @param entityName The class name of the initial JPA entity. | |
178 | * @param alias The alias to use for this entity; if it is null, a new alias will be auto-generated. | |
179 | * @param includeEntityAliasInSelect Indicates whether this entity's alias should be added to the SELECT clause. | |
180 | */ | |
181 | 0 | public Criteria(String entityName, String alias, boolean includeEntityAliasInSelect) { |
182 | 0 | this.initialEntityName = entityName; |
183 | 0 | from(entityName, alias, includeEntityAliasInSelect); |
184 | 0 | } |
185 | ||
186 | /* | |
187 | * Adds the given alias to the Criteria's alias lists/maps. If the alias is null or it duplicates an | |
188 | * existing one, a new alias will be auto-generated. | |
189 | * | |
190 | * Returns the new name of the alias, or the passed-in name if non-null and no conflicts were found. | |
191 | */ | |
192 | private String addAlias(String alias) { | |
193 | 0 | if (alias == null) { |
194 | 0 | alias = "a" + (generatedAliasCount++); |
195 | } | |
196 | 0 | while (aliasIndexes.containsKey(alias)) { |
197 | 0 | alias = "a" + (generatedAliasCount++); |
198 | } | |
199 | 0 | entityAliases.add(alias); |
200 | 0 | aliasIndexes.put(alias, entityAliases.size() - 1); |
201 | 0 | indexedAliasPlaceholders.add(new StringBuilder(30).append(JPA_ALIAS_PREFIX).append(indexedAliasPlaceholders.size()).append(JPA_ALIAS_SUFFIX).toString()); |
202 | 0 | namedAliasPlaceholders.add(new StringBuilder(30).append(JPA_ALIAS_PREFIX).append('\'').append(alias).append('\'').append(JPA_ALIAS_SUFFIX).toString()); |
203 | 0 | return alias; |
204 | } | |
205 | ||
206 | /** | |
207 | * Adds the given expression to the SELECT clause. If no properly-referenced alias appears | |
208 | * in the String expression, it will be assumed that the expression is accessing something | |
209 | * on or through the initial entity. See the description of this class for more information | |
210 | * on how String expressions are interpreted. | |
211 | * | |
212 | * @param resultExpression The String expression to add to the SELECT clause. | |
213 | */ | |
214 | public void select(String resultExpression) { | |
215 | 0 | if (resultExpression.contains(JPA_ALIAS_PREFIX)) { |
216 | 0 | selectClause.append((selectClause.length() > 0) ? ", " : "").append(resultExpression); |
217 | } else { | |
218 | 0 | selectClause.append((selectClause.length() > 0) ? ", " : "").append(JPA_INITIAL_ALIAS).append('.').append(resultExpression); |
219 | } | |
220 | 0 | } |
221 | ||
222 | /** | |
223 | * Adds a new JPA entity to the FROM clause. The entity will be given the provided alias if it | |
224 | * is non-null and does not conflict with any existing alias names; otherwise, a new alias name | |
225 | * will be auto-generated. An extra parameter is also included for indicating whether this | |
226 | * Criteria should automatically include the entity's alias in the SELECT clause. | |
227 | * | |
228 | * @param entityName The class name of the new entity. | |
229 | * @param alias The alias to use for this entity. | |
230 | * @param includeEntityAliasInSelect Indicates whether to include the entity's alias in the SELECT clause. | |
231 | * @return The provided alias if it's non-null and does not conflict with any existing names, or an auto-generated alias name otherwise. | |
232 | */ | |
233 | public String from(String entityName, String alias, boolean includeEntityAliasInSelect) { | |
234 | 0 | alias = addAlias(alias); |
235 | 0 | if (includeEntityAliasInSelect) { |
236 | 0 | select(namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); |
237 | } | |
238 | 0 | fromClause.append((fromClause.length() > 0) ? ", " : " FROM ").append(entityName).append(" AS ").append( |
239 | namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); | |
240 | 0 | return alias; |
241 | } | |
242 | ||
243 | /** | |
244 | * Adds a new "IN (...)" expression containing the given collection expression into the FROM clause. | |
245 | * The collection will be given the provided alias if it is non-null and does not conflict with | |
246 | * any existing alias names; otherwise, a new alias name will be auto-generated. An extra parameter | |
247 | * is also included for indicating whether this Criteria should automatically include the | |
248 | * collection's alias in the SELECT clause. | |
249 | * | |
250 | * <p>If no properly-referenced alias appears in the String expression, it will be assumed that the | |
251 | * expression points to a collection that is accessible on or through the initial entity. See the | |
252 | * description of this class for more information on how String expressions are interpreted. | |
253 | * | |
254 | * @param collectionName A String expression that points to an entity collection. | |
255 | * @param alias The alias to use for this collection expression. | |
256 | * @param includeCollectionAliasInSelect Indicates whether to include the collection's alias in the SELECT clause. | |
257 | * @return The provided alias if it's non-null and does not conflict with any existing names, or an auto-generated alias name otherwise. | |
258 | */ | |
259 | public String fromIn(String collectionName, String alias, boolean includeCollectionAliasInSelect) { | |
260 | 0 | alias = addAlias(alias); |
261 | 0 | if (includeCollectionAliasInSelect) { |
262 | 0 | select(namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); |
263 | } | |
264 | // The optional "AS" keyword is excluded when creating the "IN (...)" expression because Hibernate will throw an exception if it exists. | |
265 | 0 | if (collectionName.contains(JPA_ALIAS_PREFIX)) { |
266 | 0 | fromClause.append(", IN (").append(collectionName).append(") ").append(namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); |
267 | } else { | |
268 | 0 | fromClause.append(", IN (").append(JPA_INITIAL_ALIAS).append('.').append(collectionName).append(") ").append( |
269 | namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); | |
270 | } | |
271 | 0 | return alias; |
272 | } | |
273 | ||
274 | /** | |
275 | * Adds a new JOIN to the most-recently-added entity in the FROM clause. The association | |
276 | * will be given the provided alias if it is non-null and does not conflict with | |
277 | * any existing alias names; otherwise, a new alias name will be auto-generated. An extra parameter | |
278 | * is also included for indicating whether this Criteria should automatically include the | |
279 | * association's alias in the SELECT clause. | |
280 | * | |
281 | * <p>If no properly-referenced alias appears in the String expression, it will be assumed that the | |
282 | * expression points to an association that is accessible on or through the initial entity. See the | |
283 | * description of this class for more information on how String expressions are interpreted. | |
284 | * | |
285 | * @param associationName A String expression that points to an entity association. | |
286 | * @param alias The alias to use for this association expression. | |
287 | * @param includeAssociationAliasInSelect Indicates whether to include the collection's alias in the SELECT clause. | |
288 | * @param innerJoin If true, the join will be an inner join; otherwise, it will be a left (outer) join. | |
289 | * @return The provided alias if it's non-null and does not conflict with any existing names, or an auto-generated alias name otherwise. | |
290 | */ | |
291 | public String join(String associationName, String alias, boolean includeAssociationAliasInSelect, boolean innerJoin) { | |
292 | 0 | alias = addAlias(alias); |
293 | 0 | if (includeAssociationAliasInSelect) { |
294 | 0 | select(namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); |
295 | } | |
296 | 0 | if (associationName.contains(JPA_ALIAS_PREFIX)) { |
297 | 0 | fromClause.append(innerJoin ? " INNER JOIN " : " LEFT JOIN ").append(associationName).append(" AS ").append( |
298 | namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); | |
299 | } else { | |
300 | 0 | fromClause.append(innerJoin ? " INNER JOIN " : " LEFT JOIN ").append(JPA_INITIAL_ALIAS).append('.').append( |
301 | associationName).append(" AS ").append(namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); | |
302 | } | |
303 | 0 | return alias; |
304 | } | |
305 | ||
306 | /** | |
307 | * Adds a new JOIN FETCH to the most-recently-added entity in the FROM clause. If no | |
308 | * properly-referenced alias appears in the String expression, it will be assumed that the | |
309 | * expression points to an association that is accessible on or through the initial entity. See the | |
310 | * description of this class for more information on how String expressions are interpreted. | |
311 | * | |
312 | * @param associationName A String expression that points to an entity association. | |
313 | * @param innerJoin If true, the join will be an inner join; otherwise, it will be a left (outer) join. | |
314 | */ | |
315 | public void joinFetch(String associationName, boolean innerJoin) { | |
316 | 0 | if (associationName.contains(JPA_ALIAS_PREFIX)) { |
317 | 0 | fromClause.append(innerJoin ? " INNER JOIN FETCH " : " LEFT JOIN FETCH ").append(associationName); |
318 | } else { | |
319 | 0 | fromClause.append(innerJoin ? " INNER JOIN FETCH " : " LEFT JOIN FETCH ").append(JPA_INITIAL_ALIAS).append('.').append(associationName); |
320 | } | |
321 | 0 | } |
322 | ||
323 | /** | |
324 | * Adds a new attribute assignment expression to the SET clause (for use with UPDATE queries). It is | |
325 | * assumed that the given attribute belongs to the initial entity, so a custom alias should not be used. | |
326 | * | |
327 | * @param attributeName An expression pointing to an attribute that is accessible on or through the initial entity. | |
328 | * @param value The value to assign to the attribute. | |
329 | */ | |
330 | public void set(String attributeName, Object value) { | |
331 | 0 | setClause.append((setClause.length() > 0) ? ", " : " SET ").append(JPA_INITIAL_ALIAS).append('.').append(attributeName).append( |
332 | " = ").append(addAttr(fixPeriods(attributeName), value)); | |
333 | 0 | } |
334 | ||
335 | /** | |
336 | * Adds one new attribute assignment expression to the SET clause for each entry in the given Map (for use with | |
337 | * UPDATE queries). It is assumed that the attributes belong to the initial entity, so custom aliases should not be used. | |
338 | * | |
339 | * @param attributes A Map containing expressions pointing to initial-entity-accessible attributes and their associated values. | |
340 | */ | |
341 | public void set(Map<String,Object> attributes) { | |
342 | 0 | for (Map.Entry<String,Object> attribute : attributes.entrySet()) { |
343 | 0 | setClause.append((setClause.length() > 0) ? ", " : " SET ").append(JPA_INITIAL_ALIAS).append('.').append(attribute.getKey()).append( |
344 | " = ").append(addAttr(fixPeriods(attribute.getKey()), attribute.getValue())); | |
345 | } | |
346 | 0 | } |
347 | ||
348 | /* | |
349 | * Converts an Object into a JPQL-valid constant, or creates a new input parameter for it, depending on the Object type. | |
350 | * Booleans are converted to "true" or "false", numbers are converted to their String representations, Strings are | |
351 | * surrounded in single quotes and their existing single quotes are doubled, Class objects are converted to their | |
352 | * getName() values and are left unquoted, null values are ignored, and all other Objects are assigned to an input | |
353 | * parameter. The fixed constant or parameter name will be appended to the end of the given StringBuilder. | |
354 | */ | |
355 | private void fixValue(StringBuilder queryClause, Object value) { | |
356 | 0 | if (value == null) { return; } |
357 | 0 | Class<?> propertyType = value.getClass(); |
358 | 0 | if(TypeUtils.isIntegralClass(propertyType) || TypeUtils.isDecimalClass(propertyType)) { |
359 | 0 | queryClause.append(value.toString()); |
360 | 0 | } else if (TypeUtils.isStringClass(propertyType)) { |
361 | 0 | queryClause.append('\'').append(fixSingleQuotes(value.toString())).append('\''); |
362 | 0 | } else if (TypeUtils.isBooleanClass(propertyType)) { |
363 | 0 | queryClause.append(value.toString()); |
364 | 0 | } else if (value instanceof Class) { |
365 | 0 | queryClause.append(((Class)value).getName()); |
366 | } else { | |
367 | 0 | queryClause.append(addAttr("bind_param" + (++generatedBindParamCount), value)); |
368 | } | |
369 | 0 | } |
370 | ||
371 | /* | |
372 | * Fixes the single quotes of the given String by doubling them. Note that this method | |
373 | * will *not* wrap the final value in single quotes. | |
374 | */ | |
375 | private String fixSingleQuotes(String value) { | |
376 | 0 | return APOS_PAT.matcher(value).replaceAll("''"); |
377 | } | |
378 | ||
379 | /* | |
380 | * Fixes the periods of the given String by converting them to underscores. | |
381 | */ | |
382 | private String fixPeriods(String value) { | |
383 | 0 | return value.replace('.', '_'); |
384 | } | |
385 | ||
386 | /* | |
387 | * Fixes the search pattern by converting all non-escaped lookup wildcards ("*" and "?") into their | |
388 | * respective JPQL wildcards ("%" and "_"). Any lookup wildcards escaped by a backslash are converted | |
389 | * into their non-backslashed equivalents. | |
390 | */ | |
391 | private String fixSearchPattern(String value) { | |
392 | 0 | StringBuilder fixedPattern = new StringBuilder(value); |
393 | 0 | int valueLen = value.length(); |
394 | String lookupWildcard; | |
395 | String escapedLookupWildcard; | |
396 | char jpqlWildcard; | |
397 | // Convert all non-escaped wildcards. | |
398 | 0 | for (int i = 0; i < LOOKUP_WILDCARDS.length; i++) { |
399 | 0 | lookupWildcard = LOOKUP_WILDCARDS[i]; |
400 | 0 | escapedLookupWildcard = ESCAPED_LOOKUP_WILDCARDS[i]; |
401 | 0 | jpqlWildcard = JPQL_WILDCARDS[i]; |
402 | 0 | int wildcardIndex = fixedPattern.indexOf(lookupWildcard); |
403 | 0 | int escapedWildcardIndex = fixedPattern.indexOf(escapedLookupWildcard); |
404 | 0 | while (wildcardIndex != -1) { |
405 | 0 | if (wildcardIndex == 0 || escapedWildcardIndex != wildcardIndex - 1) { |
406 | 0 | fixedPattern.setCharAt(wildcardIndex, jpqlWildcard); |
407 | 0 | wildcardIndex = fixedPattern.indexOf(lookupWildcard, wildcardIndex); |
408 | } else { | |
409 | 0 | fixedPattern.replace(escapedWildcardIndex, wildcardIndex + 1, lookupWildcard); |
410 | 0 | wildcardIndex = fixedPattern.indexOf(lookupWildcard, wildcardIndex); |
411 | 0 | escapedWildcardIndex = fixedPattern.indexOf(escapedLookupWildcard, wildcardIndex); |
412 | } | |
413 | } | |
414 | } | |
415 | 0 | return fixedPattern.toString(); |
416 | } | |
417 | ||
418 | /* | |
419 | * Automatically appends a new " AND " to the WHERE clause if adding more than one condition to it, | |
420 | * and also appends the initial entity's alias and a '.' if the expression does not contain any | |
421 | * properly-referenced aliases (as defined in the description of this class). | |
422 | */ | |
423 | private void preparePrefixIfNecessary(String attribute) { | |
424 | 0 | whereClause.append((whereClause.length() > 0) ? " AND " : ""); |
425 | 0 | if (!attribute.contains(JPA_ALIAS_PREFIX)) { |
426 | 0 | whereClause.append(JPA_INITIAL_ALIAS).append('.'); |
427 | } | |
428 | 0 | } |
429 | ||
430 | /* | |
431 | * Generates a new input parameter name and associates it with the given value, or (if the value | |
432 | * is a Class object) just returns the value's getName() String w/out adding it to a parameter. The | |
433 | * parameter's name will resemble the "attribute" String with the periods converted to | |
434 | * underscores, unless such a parameter already exists or "attribute" contains a properly-referenced | |
435 | * alias (in which case a new unique name will be auto-generated). | |
436 | * Returns the finalized input parameter name as a "__JPA_PARAM[['...']]__" String (with a ":" prefix) | |
437 | * if the value is not a Class; otherwise, returns the getName() String of the Class object. | |
438 | */ | |
439 | private String prepareAttribute(String attribute, Object value) { | |
440 | 0 | return ( value instanceof Class ? ((Class)value).getName() : |
441 | ( attribute.contains(JPA_ALIAS_PREFIX) ? addAttr("bind_param" + (++generatedBindParamCount), value) : addAttr(fixPeriods(attribute), value) ) ); | |
442 | } | |
443 | ||
444 | /* | |
445 | * Generates new input parameter names using "attribute" as a base, and associates each new | |
446 | * parameter with the corresponding provided value (unless the given value is a Class object, | |
447 | * in which case the Class's getName() String will be used w/out adding it to a parameter). | |
448 | * The parameters' names will resemble the "attribute" String (with the periods converted to | |
449 | * underscores) plus the suffix "_bN" (where N is the index of the value associated with the | |
450 | * parameter), unless such parameters already exist or "attribute" contains a | |
451 | * properly-referenced alias (in which case new unique names will be auto-generated). | |
452 | * Returns an array of parameter names as "__JPA_PARAM[['...']]__" Strings (with ":" prefixes), | |
453 | * where each name corresponds to the associated value at the same index, and where the returned | |
454 | * array has a length equal to the number of provided values (but note that if a given value is a | |
455 | * Class object, its getName() String will be returned at that index instead and no input | |
456 | * parameter will be added for it). | |
457 | */ | |
458 | private String[] prepareAttributes(String attribute, Object... values) { | |
459 | 0 | int count = values.length; |
460 | 0 | String[] fixedAttrs = new String[count]; |
461 | 0 | if (attribute.contains(JPA_ALIAS_PREFIX)) { |
462 | 0 | int tempParamCount = ++generatedBindParamCount; |
463 | 0 | for (int i = 0; i < count; i++) { |
464 | 0 | fixedAttrs[i] = (values[i] instanceof Class) ? ((Class)values[i]).getName() : |
465 | addAttr(new StringBuilder(20).append("bind_param").append(tempParamCount).append("_b").append(i + 1).toString(), values[i]); | |
466 | } | |
467 | 0 | } else { |
468 | 0 | attribute = fixPeriods(attribute); |
469 | 0 | for (int i = 0; i < count; i++) { |
470 | 0 | fixedAttrs[i] = (values[i] instanceof Class) ? ((Class)values[i]).getName() : |
471 | addAttr(new StringBuilder(attribute.length() + 5).append(attribute).append("_b").append(i + 1).toString(), values[i]); | |
472 | } | |
473 | } | |
474 | 0 | return fixedAttrs; |
475 | } | |
476 | ||
477 | /* | |
478 | * Automatically appends a new " AND " to the WHERE clause if adding more than one condition to it, | |
479 | * and also appends the initial entity's alias and a '.' if the expression does not contain any | |
480 | * properly-referenced aliases (as defined in the description of this class). In addition, a new | |
481 | * input parameter name is generated, and the parameter will be associated with the given value (unless | |
482 | * the object is a Class object, in which case its getName() value will be used instead and no input | |
483 | * parameter will be created for it). The parameter's name will resemble the "attribute" String with | |
484 | * the periods converted to underscores, unless such a parameter already exists or "attribute" contains | |
485 | * a properly-referenced alias (in which case a new unique name will be auto-generated). | |
486 | * Returns the finalized input parameter name as a "__JPA_PARAM[['...']]__" String (with a ":" prefix) | |
487 | * if it is not a Class object; otherwise, its getName() String will be returned instead. | |
488 | */ | |
489 | private String preparePrefixAndAttributeIfNecessary(String attribute, Object value) { | |
490 | 0 | whereClause.append((whereClause.length() > 0) ? " AND " : ""); |
491 | 0 | if (value instanceof Class) { |
492 | 0 | return ((Class)value).getName(); |
493 | 0 | } else if (attribute.contains(JPA_ALIAS_PREFIX)) { |
494 | 0 | return addAttr("bind_param" + (++generatedBindParamCount), value); |
495 | } else { | |
496 | 0 | whereClause.append(JPA_INITIAL_ALIAS).append('.'); |
497 | 0 | return addAttr(fixPeriods(attribute), value); |
498 | } | |
499 | } | |
500 | ||
501 | /* | |
502 | * Adds a new input parameter and its value to the parameter Map, auto-generating a new key if the provided one already exists. | |
503 | * Returns the finalized key as a "__JPA_PARAM[['...']]__" String with a prepended ":". | |
504 | */ | |
505 | private String addAttr(String string, Object value) { | |
506 | 0 | while (params.containsKey(string)) { |
507 | 0 | string = "bind_param" + (++generatedBindParamCount); |
508 | } | |
509 | 0 | params.put(string, value); |
510 | 0 | return new StringBuilder(45).append(JPA_PARAM_PREFIX_WITH_COLON).append('\'').append(string).append('\'').append(JPA_ALIAS_SUFFIX).toString(); |
511 | } | |
512 | ||
513 | /** | |
514 | * Adds a new BETWEEN condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
515 | * | |
516 | * @param attribute An expression representing the subject of the BETWEEN condition. | |
517 | * @param value1 The lower value of the BETWEEN condition. | |
518 | * @param value2 The upper value of the BETWEEN condition. | |
519 | */ | |
520 | public void between(String attribute, Object value1, Object value2) { | |
521 | 0 | preparePrefixIfNecessary(attribute); |
522 | 0 | String[] fixedAttrs = prepareAttributes(attribute, value1, value2); |
523 | 0 | whereClause.append(attribute).append(" BETWEEN ").append(fixedAttrs[0]).append(" AND ").append(fixedAttrs[1]); |
524 | 0 | } |
525 | ||
526 | /** | |
527 | * Adds a new NOT BETWEEN condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
528 | * | |
529 | * @param attribute An expression representing the subject of the NOT BETWEEN condition. | |
530 | * @param value1 The lower value of the NOT BETWEEN condition. | |
531 | * @param value2 The upper value of the NOT BETWEEN condition. | |
532 | */ | |
533 | public void notBetween(String attribute, Object value1, Object value2) { | |
534 | 0 | preparePrefixIfNecessary(attribute); |
535 | 0 | String[] fixedAttrs = prepareAttributes(attribute, value1, value2); |
536 | 0 | whereClause.append(attribute).append(" NOT BETWEEN ").append(fixedAttrs[0]).append(" AND ").append(fixedAttrs[1]); |
537 | 0 | } |
538 | ||
539 | /** | |
540 | * Adds an equality condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
541 | * | |
542 | * @param attribute An expression representing the left-hand side of this condition. | |
543 | * @param value The value to check for equality with. | |
544 | */ | |
545 | public void eq(String attribute, Object value) { | |
546 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, value); |
547 | 0 | whereClause.append(attribute).append(" = ").append(fixedAttr); |
548 | 0 | } |
549 | ||
550 | /** | |
551 | * Adds a greater-than condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
552 | * | |
553 | * @param attribute An expression representing the left-hand side of this condition. | |
554 | * @param value The value on the right-hand side of the greater-than condition. | |
555 | */ | |
556 | public void gt(String attribute, Object value) { | |
557 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, value); |
558 | 0 | whereClause.append(attribute).append(" > ").append(fixedAttr); |
559 | 0 | } |
560 | ||
561 | /** | |
562 | * Adds a greater-than-or-equal condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
563 | * | |
564 | * @param attribute An expression representing the left-hand side of this condition. | |
565 | * @param value The value on the right-hand side of the greater-than-or-equal condition. | |
566 | */ | |
567 | public void gte(String attribute, Object value) { | |
568 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, value); |
569 | 0 | whereClause.append(attribute).append(" >= ").append(fixedAttr); |
570 | 0 | } |
571 | ||
572 | /** | |
573 | * Adds a new LIKE condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
574 | * If the provided value is not a String, its toString() representation will be used as the pattern. | |
575 | * | |
576 | * @param attribute An expression representing the left-hand side of this condition. | |
577 | * @param value The pattern to compare with for "likeness". | |
578 | */ | |
579 | public void like(String attribute, Object value) { | |
580 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, fixSearchPattern(value.toString())); |
581 | 0 | whereClause.append(attribute).append(" LIKE ").append(fixedAttr); |
582 | 0 | } |
583 | ||
584 | /** | |
585 | * Adds a new NOT LIKE condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
586 | * If the provided value is not a String, its toString() representation will be used as the pattern. | |
587 | * | |
588 | * @param attribute An expression representing the left-hand side of this condition. | |
589 | * @param value The pattern to compare with for "non-likeness". | |
590 | */ | |
591 | public void notLike(String attribute, Object value) { | |
592 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, fixSearchPattern(value.toString())); |
593 | 0 | whereClause.append(attribute).append(" NOT LIKE ").append(fixedAttr); |
594 | 0 | } |
595 | ||
596 | /** | |
597 | * Adds a new LIKE condition (containing an ESCAPE clause) to the WHERE clause, ANDing it with any | |
598 | * existing conditions if necessary. | |
599 | * If the provided value is not a String, its toString() representation will be used as the pattern. | |
600 | * | |
601 | * @param attribute An expression representing the left-hand side of this condition. | |
602 | * @param value The pattern to compare with for "likeness". | |
603 | * @param escapeChar The designated wildcard escape character for the given value. | |
604 | */ | |
605 | public void likeEscape(String attribute, Object value, char escapeChar) { | |
606 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, fixSearchPattern(value.toString())); |
607 | 0 | whereClause.append(attribute).append(" LIKE ").append(fixedAttr).append(" ESCAPE ").append( |
608 | prepareAttribute(JPA_ALIAS_PREFIX, Character.valueOf(escapeChar))); | |
609 | 0 | } |
610 | ||
611 | /** | |
612 | * Adds a new NOT LIKE condition (containing an ESCAPE clause) to the WHERE clause, ANDing it with any | |
613 | * existing conditions if necessary. | |
614 | * If the provided value is not a String, its toString() representation will be used as the pattern. | |
615 | * | |
616 | * @param attribute An expression representing the left-hand side of this condition. | |
617 | * @param value The pattern to compare with for "non-likeness". | |
618 | * @param escapeChar The designated wildcard escape character for the given value. | |
619 | */ | |
620 | public void notLikeEscape(String attribute, Object value, char escapeChar) { | |
621 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, fixSearchPattern(value.toString())); |
622 | 0 | whereClause.append(attribute).append(" NOT LIKE ").append(fixedAttr).append(" ESCAPE ").append( |
623 | prepareAttribute(JPA_ALIAS_PREFIX, Character.valueOf(escapeChar))); | |
624 | 0 | } |
625 | ||
626 | /** | |
627 | * Adds a less-than condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
628 | * | |
629 | * @param attribute An expression representing the left-hand side of this condition. | |
630 | * @param value The value on the right-hand side of the less-than condition. | |
631 | */ | |
632 | public void lt(String attribute, Object value) { | |
633 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, value); |
634 | 0 | whereClause.append(attribute).append(" < ").append(fixedAttr); |
635 | 0 | } |
636 | ||
637 | /** | |
638 | * Adds a less-than-or-equal condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
639 | * | |
640 | * @param attribute An expression representing the left-hand side of this condition. | |
641 | * @param value The value on the right-hand side of the less-than-or-equal condition. | |
642 | */ | |
643 | public void lte(String attribute, Object value) { | |
644 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, value); |
645 | 0 | whereClause.append(attribute).append(" <= ").append(fixedAttr); |
646 | 0 | } |
647 | ||
648 | /** | |
649 | * Adds a non-equality condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
650 | * | |
651 | * @param attribute An expression representing the left-hand side of this condition. | |
652 | * @param value The value on the right-hand side of the non-equality condition. | |
653 | */ | |
654 | public void ne(String attribute, Object value) { | |
655 | 0 | String fixedAttr = preparePrefixAndAttributeIfNecessary(attribute, value); |
656 | 0 | whereClause.append(attribute).append(" <> ").append(fixedAttr); |
657 | 0 | } |
658 | ||
659 | /** | |
660 | * Adds an IS NULL condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
661 | * | |
662 | * @param attribute An expression representing what needs the null check. | |
663 | */ | |
664 | public void isNull(String attribute) { | |
665 | 0 | preparePrefixIfNecessary(attribute); |
666 | 0 | whereClause.append(attribute).append(" IS NULL"); |
667 | 0 | } |
668 | ||
669 | /** | |
670 | * Adds an IS NOT NULL condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
671 | * | |
672 | * @param attribute An expression representing what needs the not-null check. | |
673 | */ | |
674 | public void notNull(String attribute) { | |
675 | 0 | preparePrefixIfNecessary(attribute); |
676 | 0 | whereClause.append(attribute).append(" IS NOT NULL"); |
677 | 0 | } |
678 | ||
679 | /** | |
680 | * Adds a MEMBER OF condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
681 | * Uses an input parameter for the single-value part of the condition. | |
682 | * | |
683 | * @param value An Object that may or may not be in the specified collection. | |
684 | * @param collection An expression representing a collection. | |
685 | */ | |
686 | public void memberOf(Object value, String collection) { | |
687 | 0 | String fixedAttr = prepareAttribute(collection, value); |
688 | 0 | preparePrefixIfNecessary(JPA_ALIAS_PREFIX); |
689 | 0 | whereClause.append(fixedAttr).append(" MEMBER OF ").append( |
690 | collection.contains(JPA_ALIAS_PREFIX) ? "" : JPA_INITIAL_ALIAS + ".").append(collection); | |
691 | 0 | } |
692 | ||
693 | /** | |
694 | * Adds a MEMBER OF condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
695 | * Uses an expression for the single-value part of the condition. | |
696 | * | |
697 | * @param attribute An expression pointing to an attribute that may or may not be in the given collection. | |
698 | * @param collection An expression representing a collection. | |
699 | */ | |
700 | public void memberOf(String attribute, String collection) { | |
701 | 0 | preparePrefixIfNecessary(attribute); |
702 | 0 | whereClause.append(attribute).append(" MEMBER OF ").append( |
703 | collection.contains(JPA_ALIAS_PREFIX) ? "" : JPA_INITIAL_ALIAS + ".").append(collection); | |
704 | 0 | } |
705 | ||
706 | /** | |
707 | * Adds a NOT MEMBER OF condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
708 | * Uses an input parameter for the single-value part of the condition. | |
709 | * | |
710 | * @param value An Object that may or may not be in the specified collection. | |
711 | * @param collection An expression representing a collection. | |
712 | */ | |
713 | public void notMemberOf(Object value, String collection) { | |
714 | 0 | String fixedAttr = prepareAttribute(collection, value); |
715 | 0 | preparePrefixIfNecessary(JPA_ALIAS_PREFIX); |
716 | 0 | whereClause.append(fixedAttr).append(" NOT MEMBER OF ").append( |
717 | collection.contains(JPA_ALIAS_PREFIX) ? "" : JPA_INITIAL_ALIAS + ".").append(collection); | |
718 | 0 | } |
719 | ||
720 | /** | |
721 | * Adds a NOT MEMBER OF condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
722 | * Uses an expression for the single-value part of the condition. | |
723 | * | |
724 | * @param attribute An expression pointing to an attribute that may or may not be in the given collection. | |
725 | * @param collection An expression representing a collection. | |
726 | */ | |
727 | public void notMemberOf(String attribute, String collection) { | |
728 | 0 | preparePrefixIfNecessary(attribute); |
729 | 0 | whereClause.append(attribute).append(" NOT MEMBER OF ").append( |
730 | collection.contains(JPA_ALIAS_PREFIX) ? "" : JPA_INITIAL_ALIAS + ".").append(collection); | |
731 | 0 | } |
732 | ||
733 | /** | |
734 | * Adds an IS EMPTY condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
735 | * | |
736 | * @param collection An expression representing a collection. | |
737 | */ | |
738 | public void isEmpty(String collection) { | |
739 | 0 | preparePrefixIfNecessary(collection); |
740 | 0 | whereClause.append(collection).append(" IS EMPTY"); |
741 | 0 | } |
742 | ||
743 | /** | |
744 | * Adds an IS NOT EMPTY condition to the WHERE clause, ANDing it with any existing conditions if necessary. | |
745 | * | |
746 | * @param collection An expression representing a collection. | |
747 | */ | |
748 | public void notEmpty(String collection) { | |
749 | 0 | preparePrefixIfNecessary(collection); |
750 | 0 | whereClause.append(collection).append(" IS NOT EMPTY"); |
751 | 0 | } |
752 | ||
753 | /** | |
754 | * Converts the given Criteria into a sub-SELECT query, surrounds it in parentheses, and adds a new | |
755 | * "IN (...)" condition containing the resulting expression into the WHERE clause, while ANDing | |
756 | * the new condition with any existing conditions if necessary. | |
757 | * | |
758 | * @param attribute An expression representing the left-hand side of the "IN (...)" condition. | |
759 | * @param in The Criteria representing the sub-query of the "IN (...)" condition. | |
760 | */ | |
761 | public void in(String attribute, Criteria in) { | |
762 | 0 | if (in == null) { throw new IllegalArgumentException("'IN' Criteria cannot be null"); } |
763 | 0 | preparePrefixIfNecessary(attribute); |
764 | 0 | whereClause.append(attribute).append(" IN "); |
765 | 0 | subQuery(in); |
766 | 0 | } |
767 | ||
768 | /** | |
769 | * Converts the given Criteria into a sub-SELECT query, surrounds it in parentheses, and adds a new | |
770 | * "NOT IN (...)" condition containing the resulting expression into the WHERE clause, while ANDing | |
771 | * the new condition with any existing conditions if necessary. | |
772 | * | |
773 | * @param attribute An expression representing the left-hand side of the "NOT IN (...)" condition. | |
774 | * @param in The Criteria representing the sub-query of the "NOT IN (...)" condition. | |
775 | */ | |
776 | public void notIn(String attribute, Criteria notIn) { | |
777 | 0 | if (notIn == null) { throw new IllegalArgumentException("'NOT IN' Criteria cannot be null"); } |
778 | 0 | preparePrefixIfNecessary(attribute); |
779 | 0 | whereClause.append(attribute).append(" NOT IN "); |
780 | 0 | subQuery(notIn); |
781 | 0 | } |
782 | ||
783 | /** | |
784 | * Takes the individual elements of the provided collection, converts them to literals or assigns them to input | |
785 | * parameters as needed, and adds a new "IN (...)" condition to the WHERE clause containing those elements, | |
786 | * ANDing it with any existing conditions if necessary. String, number, boolean, and Class collection elements | |
787 | * will be converted into literals, and all other non-null collection elements will be assigned to input parameters. | |
788 | * | |
789 | * @param attribute An expression representing the left-hand side of the "IN (...)" condition. | |
790 | * @param values The collection of values to check against. | |
791 | */ | |
792 | public void in(String attribute, Collection<?> values) { | |
793 | 0 | preparePrefixIfNecessary(attribute); |
794 | 0 | whereClause.append(attribute).append(" IN ("); |
795 | 0 | appendBodyOfIn(values); |
796 | 0 | whereClause.append(')'); |
797 | 0 | } |
798 | ||
799 | /** | |
800 | * Takes the individual elements of the provided collection, converts them to literals or assigns them to input | |
801 | * parameters as needed, and adds a new "IN (...)" condition to the WHERE clause containing those elements, | |
802 | * ANDing it with any existing conditions if necessary. String, number, boolean, and Class collection elements | |
803 | * will be converted into literals, and all other non-null collection elements will be assigned to input parameters. | |
804 | * | |
805 | * @param attribute An expression representing the left-hand side of the "NOT IN (...)" condition. | |
806 | * @param values The collection of values to check against. | |
807 | */ | |
808 | public void notIn(String attribute, Collection<?> values) { | |
809 | 0 | preparePrefixIfNecessary(attribute); |
810 | 0 | whereClause.append(attribute).append(" NOT IN ("); |
811 | 0 | appendBodyOfIn(values); |
812 | 0 | whereClause.append(')'); |
813 | 0 | } |
814 | ||
815 | /** | |
816 | * Takes the provided values, converts them to literals or assigns them to input parameters as needed, and adds | |
817 | * a new "IN (...)" condition to the WHERE clause containing those elements, ANDing it with any existing | |
818 | * conditions if necessary. String, number, boolean, and Class values will be converted into literals, and all | |
819 | * other non-null values will be assigned to input parameters. | |
820 | * | |
821 | * @param attribute An expression representing the left-hand side of the "IN (...)" condition. | |
822 | * @param values The values to check against. | |
823 | */ | |
824 | public void in(String attribute, Object... values) { | |
825 | 0 | preparePrefixIfNecessary(attribute); |
826 | 0 | whereClause.append(attribute).append(" IN ("); |
827 | 0 | appendBodyOfIn(Arrays.asList(values)); |
828 | 0 | whereClause.append(')'); |
829 | 0 | } |
830 | ||
831 | /** | |
832 | * Takes the provided values, converts them to literals or assigns them to input parameters as needed, and adds | |
833 | * a new "NOT IN (...)" condition to the WHERE clause containing those elements, ANDing it with any existing | |
834 | * conditions if necessary. String, number, boolean, and Class values will be converted into literals, and all | |
835 | * other non-null values will be assigned to input parameters. | |
836 | * | |
837 | * @param attribute An expression representing the left-hand side of the "NOT IN (...)" condition. | |
838 | * @param values The values to check against. | |
839 | */ | |
840 | public void notIn(String attribute, Object... values) { | |
841 | 0 | preparePrefixIfNecessary(attribute); |
842 | 0 | whereClause.append(attribute).append(" NOT IN ("); |
843 | 0 | appendBodyOfIn(Arrays.asList(values)); |
844 | 0 | whereClause.append(')'); |
845 | 0 | } |
846 | ||
847 | /* | |
848 | * Creates the body of an "IN (...)" condition using the provided collection of values. | |
849 | * String, number, and boolean values are converted into literals, and all other | |
850 | * non-null values are assigned to input parameters. | |
851 | */ | |
852 | private void appendBodyOfIn(Collection<?> values) { | |
853 | 0 | Iterator<?> valuesIter = values.iterator(); |
854 | 0 | if (valuesIter.hasNext()) { |
855 | 0 | fixValue(whereClause, valuesIter.next()); |
856 | 0 | while (valuesIter.hasNext()) { |
857 | 0 | whereClause.append(", "); |
858 | 0 | fixValue(whereClause, valuesIter.next()); |
859 | } | |
860 | } | |
861 | 0 | } |
862 | ||
863 | /** | |
864 | * Takes the conditions from the given Criteria's WHERE clause, surrounds them in parentheses, and ANDs | |
865 | * the resulting expression with this instance's WHERE clause (or makes the expression become the new | |
866 | * WHERE clause of this instance if its current WHERE clause is empty). Any input parameters from the | |
867 | * given Criteria will be copied over to this Criteria, and will be renamed as needed if conflicts arise. | |
868 | * | |
869 | * @param and The Criteria instance whose conditions should be ANDed with those in this Criteria instance. | |
870 | */ | |
871 | public void and(Criteria and) { | |
872 | 0 | if (and == null) { throw new IllegalArgumentException("'AND' Criteria cannot be null"); } |
873 | 0 | if (and.whereClause.length() > 0) { |
874 | 0 | int oldLen = whereClause.length(); |
875 | 0 | preparePrefixIfNecessary(JPA_ALIAS_PREFIX); |
876 | 0 | whereClause.append('(').append(and.whereClause).append(')'); |
877 | 0 | copyParams(and, oldLen, true); |
878 | } | |
879 | 0 | } |
880 | ||
881 | /** | |
882 | * Takes the conditions from the given Criteria's WHERE clause, surrounds them in parentheses, and adds | |
883 | * a new "NOT (...)" condition containing the resulting expression to this instance's WHERE clause (or | |
884 | * adds a new "AND NOT(...)" condition if the WHERE clause on the existing Criteria is not empty). Any | |
885 | * input parameters from the given Criteria will be copied over to this Criteria, and will be renamed | |
886 | * as needed if conflicts arise. | |
887 | * | |
888 | * @param not The Criteria instance whose group of conditions should be negated. | |
889 | */ | |
890 | public void not(Criteria not) { | |
891 | 0 | if (not == null) { throw new IllegalArgumentException("'NOT' Criteria cannot be null"); } |
892 | 0 | if (not.whereClause.length() > 0) { |
893 | 0 | int oldLen = whereClause.length(); |
894 | 0 | preparePrefixIfNecessary(JPA_ALIAS_PREFIX); |
895 | 0 | whereClause.append("NOT (").append(not.whereClause).append(')'); |
896 | 0 | copyParams(not, oldLen, true); |
897 | } | |
898 | 0 | } |
899 | ||
900 | /** | |
901 | * Takes the conditions from the given Criteria's WHERE clause, surrounds them in parentheses, and ORs | |
902 | * the resulting expression with this instance's WHERE clause (or makes the expression become the new | |
903 | * WHERE clause of this instance if its current WHERE clause is empty). Any input parameters from the | |
904 | * given Criteria will be copied over to this Criteria, and will be renamed as needed if conflicts arise. | |
905 | * | |
906 | * @param or The Criteria instance whose conditions should be ORed with those in this Criteria instance. | |
907 | */ | |
908 | public void or(Criteria or) { | |
909 | 0 | if (or == null) { throw new IllegalArgumentException("'OR' Criteria cannot be null"); } |
910 | 0 | if (or.whereClause.length() > 0) { |
911 | 0 | int oldLen = whereClause.length(); |
912 | 0 | whereClause.append((oldLen > 0) ? " OR (" : "(").append(or.whereClause).append(')'); |
913 | 0 | copyParams(or, oldLen, true); |
914 | } | |
915 | 0 | } |
916 | ||
917 | /** | |
918 | * Takes the conditions from the given Criteria's WHERE clause, surrounds them in parentheses, and adds | |
919 | * a new "OR NOT (...)" condition containing the resulting expression to this instance's WHERE clause (or | |
920 | * just adds a new "NOT (...)" condition if its current WHERE clause is empty). Any input parameters from the | |
921 | * given Criteria will be copied over to this Criteria, and will be renamed as needed if conflicts arise. | |
922 | * | |
923 | * @param orNot The Criteria instance whose group of conditions should be negated and then ORed with those in this Criteria instance. | |
924 | */ | |
925 | public void orNot(Criteria orNot) { | |
926 | 0 | if (orNot == null) { throw new IllegalArgumentException("'OR NOT' Criteria cannot be null"); } |
927 | 0 | if (orNot.whereClause.length() > 0) { |
928 | 0 | int oldLen = whereClause.length(); |
929 | 0 | whereClause.append((oldLen > 0) ? " OR NOT (" : "NOT (").append(orNot.whereClause).append(')'); |
930 | 0 | copyParams(orNot, oldLen, true); |
931 | } | |
932 | 0 | } |
933 | ||
934 | /** | |
935 | * Converts the given Criteria into a sub-SELECT query, surrounds it in parentheses, and adds a new | |
936 | * EXISTS condition containing the resulting expression into the WHERE clause, while ANDing the new | |
937 | * condition with any existing conditions if necessary. | |
938 | * | |
939 | * @param exists The Criteria instance representing the sub-query of the EXISTS condition. | |
940 | */ | |
941 | public void exists(Criteria exists) { | |
942 | 0 | if (exists == null) { throw new IllegalArgumentException("'EXISTS' Criteria cannot be null"); } |
943 | 0 | preparePrefixIfNecessary(JPA_ALIAS_PREFIX); |
944 | 0 | whereClause.append("EXISTS "); |
945 | 0 | subQuery(exists); |
946 | 0 | } |
947 | ||
948 | /** | |
949 | * Converts the given Criteria into a sub-SELECT query, surrounds it in parentheses, and adds a new | |
950 | * NOT EXISTS condition containing the resulting expression into the WHERE clause, while ANDing the | |
951 | * new condition with any existing conditions if necessary. | |
952 | * | |
953 | * @param exists The Criteria instance representing the sub-query of the NOT EXISTS condition. | |
954 | */ | |
955 | public void notExists(Criteria notExists) { | |
956 | 0 | if (notExists == null) { throw new IllegalArgumentException("'NOT EXISTS' Criteria cannot be null"); } |
957 | 0 | preparePrefixIfNecessary(JPA_ALIAS_PREFIX); |
958 | 0 | whereClause.append("NOT EXISTS "); |
959 | 0 | subQuery(notExists); |
960 | 0 | } |
961 | ||
962 | /** | |
963 | * Inserts raw JPQL into the WHERE clause. An extra whitespace character will automatically be prepended | |
964 | * (if the WHERE clause is non-empty) and appended to the JPQL string. | |
965 | * | |
966 | * @param jpql The raw JPQL to insert. | |
967 | */ | |
968 | public void rawJpql(String jpql) { | |
969 | 0 | whereClause.append((whereClause.length() > 0) ? " " : "").append(jpql).append(' '); |
970 | 0 | } |
971 | ||
972 | /** | |
973 | * Adds a new expression to the GROUP BY clause. If no properly-referenced alias appears in | |
974 | * the String expression, it will be assumed that the expression points to something | |
975 | * that is accessible on or through the initial entity. See the description of this class | |
976 | * for more information on how String expressions are interpreted. | |
977 | * | |
978 | * @param attribute The expression representing what the query results should be grouped by. | |
979 | */ | |
980 | public void groupBy(String attribute) { | |
981 | 0 | groupByClause.append((groupByClause.length() > 0) ? ", " : " GROUP BY "); |
982 | 0 | if (!attribute.contains(JPA_ALIAS_PREFIX)) { |
983 | 0 | groupByClause.append(JPA_INITIAL_ALIAS).append('.'); |
984 | } | |
985 | 0 | groupByClause.append(attribute); |
986 | 0 | } |
987 | ||
988 | /** | |
989 | * Takes the conditions from the given Criteria's WHERE clause and then ANDs them with this | |
990 | * instance's HAVING clause (or makes them become the new HAVING clause of this instance if | |
991 | * its current HAVING clause is empty). Any input parameters from the given Criteria will be | |
992 | * copied over to this Criteria, and will be renamed as needed if conflicts arise. | |
993 | * | |
994 | * @param having The Criteria instance whose WHERE clause should represent this instance's HAVING clause (or a piece of it). | |
995 | */ | |
996 | public void having(Criteria having) { | |
997 | 0 | if (having == null) { throw new IllegalArgumentException("'HAVING' Criteria cannot be null"); } |
998 | 0 | if (having.whereClause.length() > 0) { |
999 | 0 | int oldLen = havingClause.length(); |
1000 | 0 | havingClause.append((oldLen > 0) ? " AND " : " HAVING ").append(having.whereClause); |
1001 | 0 | copyParams(having, oldLen, false); |
1002 | } | |
1003 | 0 | } |
1004 | ||
1005 | /** | |
1006 | * Adds a new expression to the ORDER BY clause. If no properly-referenced alias appears in | |
1007 | * the String expression, it will be assumed that the expression points to something | |
1008 | * that is accessible on or through the initial entity. See the description of this class | |
1009 | * for more information on how String expressions are interpreted. | |
1010 | * | |
1011 | * @param attribute The expression representing what the query results should be ordered by. | |
1012 | * @param sortAscending If true, the order will be ascending; otherwise, it will be descending. | |
1013 | */ | |
1014 | public void orderBy(String attribute, boolean sortAscending) { | |
1015 | 0 | orderByClause.append((orderByClause.length() > 0) ? ", " : " ORDER BY "); |
1016 | 0 | if (!attribute.contains(JPA_ALIAS_PREFIX)) { |
1017 | 0 | orderByClause.append(JPA_INITIAL_ALIAS).append('.'); |
1018 | } | |
1019 | 0 | orderByClause.append(attribute).append(sortAscending ? " ASC" : " DESC"); |
1020 | 0 | } |
1021 | ||
1022 | /* | |
1023 | * Copies the input parameters from the given child Criteria to the current Criteria. Any parameter placeholders from the sub-query will | |
1024 | * be renamed accordingly if they conflict with those defined by the parent Criteria. | |
1025 | * | |
1026 | * It is assumed that the sub-query has already been appended to the clause given by modifyWhereClause | |
1027 | * (WHERE if true, HAVING if false), starting at the index given by startIndex. | |
1028 | */ | |
1029 | private void copyParams(Criteria subCriteria, int startIndex, boolean modifyWhereClause) { | |
1030 | 0 | final StringBuilder tempClause = modifyWhereClause ? whereClause : havingClause; |
1031 | 0 | Map<String,String> modifiedNames = new HashMap<String,String>(); |
1032 | 0 | int paramClose = 0; |
1033 | 0 | int paramIndex = tempClause.indexOf(JPA_PARAM_PREFIX_WITH_COLON, startIndex); |
1034 | 0 | String newPlaceholder = null; |
1035 | ||
1036 | 0 | for (Map.Entry<String,Object> param : subCriteria.params.entrySet()) { |
1037 | 0 | if (params.containsKey(param.getKey())) { |
1038 | 0 | modifiedNames.put(param.getKey(), addAttr(param.getKey(), param.getValue())); |
1039 | } else { | |
1040 | 0 | params.put(param.getKey(), param.getValue()); |
1041 | } | |
1042 | } | |
1043 | ||
1044 | 0 | while (paramIndex >= 0) { |
1045 | 0 | paramClose = tempClause.indexOf(JPA_ALIAS_SUFFIX, paramIndex); |
1046 | 0 | newPlaceholder = modifiedNames.get(tempClause.substring(paramIndex + PARAM_PREFIX_LEN + 2, paramClose - 1)); |
1047 | 0 | if (newPlaceholder != null) { |
1048 | 0 | tempClause.replace(paramIndex, paramClose + ALIAS_SUFFIX_LEN, newPlaceholder); |
1049 | } | |
1050 | 0 | paramIndex = tempClause.indexOf(JPA_PARAM_PREFIX_WITH_COLON, paramIndex + 1); |
1051 | } | |
1052 | 0 | } |
1053 | ||
1054 | /* | |
1055 | * Converts the given Criteria into a sub-query and appends it to the WHERE clause. Any named aliases of the sub-query will be renamed | |
1056 | * accordingly if they conflict with those defined by the parent Criteria, and all indexed aliases will be updated accordingly. In | |
1057 | * addition, the input parameters will be copied over and renamed as needed. | |
1058 | */ | |
1059 | private void subQuery(Criteria subCriteria) { | |
1060 | 0 | Map<String,String> modifiedAliases = new HashMap<String,String>(); |
1061 | 0 | int count = subCriteria.entityAliases.size(); |
1062 | 0 | int paramIndex = 0; |
1063 | 0 | int paramClose = 0; |
1064 | 0 | String newPlaceholder = null; |
1065 | ||
1066 | 0 | for (int i = entityAliases.size() - 1; i >= 0; i--) { |
1067 | 0 | modifiedAliases.put(JPA_ALIAS_PREFIX + (~i) + JPA_ALIAS_SUFFIX, indexedAliasPlaceholders.get(i)); |
1068 | } | |
1069 | 0 | for (int i = 0; i < count; i++) { |
1070 | 0 | String oldAlias = subCriteria.entityAliases.get(i); |
1071 | 0 | if (!addAlias(oldAlias).equals(oldAlias)) { |
1072 | 0 | modifiedAliases.put(subCriteria.namedAliasPlaceholders.get(i), namedAliasPlaceholders.get(namedAliasPlaceholders.size() - 1)); |
1073 | } | |
1074 | 0 | modifiedAliases.put(subCriteria.indexedAliasPlaceholders.get(i), indexedAliasPlaceholders.get(indexedAliasPlaceholders.size() - 1)); |
1075 | } | |
1076 | ||
1077 | 0 | count = whereClause.length(); |
1078 | 0 | whereClause.append(subCriteria.distinct ? "(SELECT DISTINCT " : "(SELECT ").append(subCriteria.selectClause).append(subCriteria.fromClause).append( |
1079 | (subCriteria.whereClause.length() > 0) ? " WHERE " : "").append(subCriteria.whereClause).append( | |
1080 | subCriteria.groupByClause).append(subCriteria.havingClause).append(')'); | |
1081 | ||
1082 | 0 | paramIndex = whereClause.indexOf(JPA_ALIAS_PREFIX, count); |
1083 | 0 | while (paramIndex >= 0) { |
1084 | 0 | paramClose = whereClause.indexOf(JPA_ALIAS_SUFFIX, paramIndex) + ALIAS_SUFFIX_LEN; |
1085 | 0 | newPlaceholder = modifiedAliases.get(whereClause.substring(paramIndex, paramClose)); |
1086 | 0 | if (newPlaceholder != null) { |
1087 | 0 | whereClause.replace(paramIndex, paramClose, newPlaceholder); |
1088 | } | |
1089 | 0 | paramIndex = whereClause.indexOf(JPA_ALIAS_PREFIX, paramIndex + 1); |
1090 | } | |
1091 | 0 | copyParams(subCriteria, count, true); |
1092 | 0 | } |
1093 | ||
1094 | /** | |
1095 | * Converts the current Criteria instance into a JPQL query. | |
1096 | * | |
1097 | * @param type An indicator of what type of query should be generated (SELECT, UPDATE, or DELETE). | |
1098 | * @return A String representing the JPQL query generated by this Criteria instance. | |
1099 | */ | |
1100 | public String toQuery(QueryByCriteria.QueryByCriteriaType type) { | |
1101 | 0 | return toQuery(type, new String[0]); |
1102 | } | |
1103 | ||
1104 | /** | |
1105 | * Converts the current Criteria instance into a JPQL query. If a SELECT query is desired, then | |
1106 | * alternative String expressions to insert into the SELECT clause can be specified, though doing | |
1107 | * so is not required and must be done with care (see below). | |
1108 | * | |
1109 | * <p>WARNING! If queryAttr is non-null and has a length greater than zero, and if a SELECT query is | |
1110 | * desired, then this method will completely overwrite the SELECT clause so that it only contains | |
1111 | * the expressions given by the queryAttr array. | |
1112 | * | |
1113 | * @param type An indicator of what type of query should be generated (SELECT, UPDATE, or DELETE). | |
1114 | * @param queryAttr (Optional) An array of String expressions to use as the values to be returned by the SELECT clause (if creating a SELECT query). | |
1115 | * @return A String representing the JPQL query generated by this Criteria instance. | |
1116 | */ | |
1117 | public String toQuery(QueryByCriteria.QueryByCriteriaType type, String[] queryAttr) { | |
1118 | StringBuilder newQuery; | |
1119 | 0 | int querySize = selectClause.length() + fromClause.length() + whereClause.length() + groupByClause.length() + havingClause.length() + orderByClause.length() + 7; |
1120 | // Construct the Criteria appropriately. | |
1121 | 0 | switch (type) { |
1122 | case SELECT : | |
1123 | 0 | if (queryAttr != null && queryAttr.length > 0) { |
1124 | 0 | if (selectClause.length() > 0) { selectClause.delete(0, selectClause.length()); } |
1125 | 0 | for (int i = 0; i < queryAttr.length; i++) { |
1126 | 0 | select(queryAttr[i]); |
1127 | } | |
1128 | } | |
1129 | 0 | newQuery = new StringBuilder(querySize + (distinct ? 16 : 7)).append(distinct ? "SELECT DISTINCT " : "SELECT ").append( |
1130 | selectClause).append(fromClause).append(whereClause.length() > 0 ? " WHERE " : "").append(whereClause).append( | |
1131 | groupByClause).append(havingClause).append(orderByClause); | |
1132 | 0 | break; |
1133 | case UPDATE : | |
1134 | 0 | newQuery = new StringBuilder(querySize + setClause.length() + 7).append("UPDATE ").append(initialEntityName).append(" AS ").append( |
1135 | getAlias()).append(setClause).append(whereClause.length() > 0 ? " WHERE " : "").append(whereClause); | |
1136 | 0 | break; |
1137 | case DELETE : | |
1138 | 0 | newQuery = new StringBuilder(querySize + 6).append("DELETE").append(fromClause).append( |
1139 | whereClause.length() > 0 ? " WHERE " : "").append(whereClause); | |
1140 | 0 | break; |
1141 | default : | |
1142 | 0 | return null; |
1143 | } | |
1144 | 0 | return fix(newQuery); |
1145 | } | |
1146 | ||
1147 | /** | |
1148 | * Converts the current Criteria instance into a JPQL SELECT query, but replaces the existing SELECT clause with "SELECT COUNT(*)" instead. | |
1149 | * | |
1150 | * @return A JPQL query String given by this Criteria, but with "SELECT COUNT(*)" as the SELECT clause instead. | |
1151 | */ | |
1152 | public String toCountQuery() { | |
1153 | 0 | StringBuilder newQuery = new StringBuilder( |
1154 | fromClause.length() + whereClause.length() + groupByClause.length() + havingClause.length() + orderByClause.length() + 15 + 7).append( | |
1155 | "SELECT COUNT(*)").append(fromClause).append(whereClause.length() > 0 ? " WHERE " : "").append(whereClause).append( | |
1156 | groupByClause).append(havingClause).append(orderByClause); | |
1157 | 0 | return fix(newQuery); |
1158 | } | |
1159 | ||
1160 | /* | |
1161 | * Converts all the named and indexed alias/parameter placeholders into the actual aliases/parameters. | |
1162 | * You can view the resulting JPQL String in the logs by configuring a log4j DEBUG output level for | |
1163 | * this class. | |
1164 | */ | |
1165 | private String fix(StringBuilder newQuery) { | |
1166 | 0 | Map<String,String> modifiedAliases = new HashMap<String,String>(); |
1167 | 0 | for (int i = entityAliases.size() - 1; i >= 0; i--) { |
1168 | 0 | modifiedAliases.put(indexedAliasPlaceholders.get(i), entityAliases.get(i)); |
1169 | 0 | modifiedAliases.put(namedAliasPlaceholders.get(i), entityAliases.get(i)); |
1170 | } | |
1171 | ||
1172 | // resolve the aliases. | |
1173 | 0 | String newParam = null; |
1174 | 0 | int paramClose = 0; |
1175 | 0 | int paramIndex = newQuery.indexOf(JPA_ALIAS_PREFIX); |
1176 | 0 | while (paramIndex >= 0) { |
1177 | 0 | paramClose = newQuery.indexOf(JPA_ALIAS_SUFFIX, paramIndex) + ALIAS_SUFFIX_LEN; |
1178 | 0 | newParam = modifiedAliases.get(newQuery.substring(paramIndex, paramClose)); |
1179 | 0 | if (newParam != null) { |
1180 | 0 | newQuery.replace(paramIndex, paramClose, newParam); |
1181 | } else { | |
1182 | 0 | LOG.error("Detected an unresolvable JPA alias when constructing query: " + newQuery.substring(paramIndex, paramClose)); |
1183 | 0 | throw new IllegalStateException("Detected an unresolvable alias: " + newQuery.substring(paramIndex, paramClose)); |
1184 | } | |
1185 | 0 | paramIndex = newQuery.indexOf(JPA_ALIAS_PREFIX, paramIndex + 1); |
1186 | } | |
1187 | ||
1188 | // Resolve the parameters. | |
1189 | 0 | paramIndex = newQuery.indexOf(JPA_PARAM_PREFIX); |
1190 | 0 | while (paramIndex >= 0) { |
1191 | 0 | paramClose = newQuery.indexOf(JPA_ALIAS_SUFFIX, paramIndex); |
1192 | 0 | newParam = newQuery.substring(paramIndex + PARAM_PREFIX_LEN + 1, paramClose - 1); |
1193 | 0 | if (params.containsKey(newParam)) { |
1194 | 0 | newQuery.replace(paramIndex, paramClose + ALIAS_SUFFIX_LEN, newParam); |
1195 | } else { | |
1196 | 0 | LOG.error("Detected an unresolvable input parameter when constructing query: " + newQuery.substring(paramIndex, paramClose + ALIAS_SUFFIX_LEN)); |
1197 | 0 | throw new IllegalStateException("Detected an unresolvable input parameter: " + newQuery.substring(paramIndex, paramClose + ALIAS_SUFFIX_LEN)); |
1198 | } | |
1199 | 0 | paramIndex = newQuery.indexOf(JPA_PARAM_PREFIX, paramIndex + 1); |
1200 | } | |
1201 | ||
1202 | 0 | String finishedQuery = newQuery.toString(); |
1203 | 0 | if (LOG.isDebugEnabled()) { |
1204 | 0 | LOG.debug("********** SEARCH JPQL QUERY **********"); |
1205 | 0 | LOG.debug(finishedQuery); |
1206 | 0 | LOG.debug("***************************************"); |
1207 | } | |
1208 | 0 | return finishedQuery; |
1209 | } | |
1210 | ||
1211 | // Keep this package access so the QueryByCriteria can call it from this package. | |
1212 | /** | |
1213 | * Populates the given Query instance with the parameters stored in this Criteria. It is assumed that the given query | |
1214 | * is using this Criteria's toQuery() or toCountQuery() value as its JPQL String. | |
1215 | * | |
1216 | * @param query A JPA Query instance containing this Criteria's toQuery() or toCountQuery() value for its JPQL String. | |
1217 | */ | |
1218 | public void prepareParameters(Query query) { | |
1219 | 0 | for (Map.Entry<String, Object> param : params.entrySet()) { |
1220 | 0 | Object value = param.getValue(); |
1221 | 0 | if (value != null) { |
1222 | 0 | if (value instanceof String) { |
1223 | 0 | if (query.getParameter(param.getKey()).getParameterType().equals(java.lang.Boolean.class)) { |
1224 | 0 | value = Truth.strToBooleanIgnoreCase((String) value, Boolean.FALSE); |
1225 | } else { | |
1226 | //value = ((String)value).replaceAll("\\*", "%"); | |
1227 | } | |
1228 | } | |
1229 | 0 | query.setParameter(param.getKey(), value); |
1230 | } | |
1231 | 0 | } |
1232 | 0 | } |
1233 | ||
1234 | /** | |
1235 | * Retrieves the current limit on the result set size for this Criteria, if any (only relevant if creating a SELECT query). | |
1236 | * | |
1237 | * @return The current limit on the number of results to be returned by this Criteria's query, or null (default) if no such limit has been set. | |
1238 | */ | |
1239 | public Integer getSearchLimit() { | |
1240 | 0 | return this.searchLimit; |
1241 | } | |
1242 | ||
1243 | /** | |
1244 | * Sets the current limit on the result set size for this Criteria (only relevant if creating a SELECT query). | |
1245 | * | |
1246 | * @param searchLimit The new limit on the number of results to be returned by this Criteria's query. | |
1247 | */ | |
1248 | public void setSearchLimit(Integer searchLimit) { | |
1249 | 0 | this.searchLimit = searchLimit; |
1250 | 0 | } |
1251 | ||
1252 | /** | |
1253 | * Sets whether or not the query should include the DISTINCT keyword in the SELECT clause, if creating a SELECT query. | |
1254 | * If this property is not set explicitly, it is assumed that DISTINCT should *not* be included in the SELECT clause. | |
1255 | * | |
1256 | * @param distinct An indicator for whether to do a SELECT DISTINCT or just a SELECT. | |
1257 | */ | |
1258 | public void distinct(boolean distinct){ | |
1259 | 0 | this.distinct = distinct; |
1260 | 0 | } |
1261 | ||
1262 | /** | |
1263 | * Retrieves the initial alias of this Criteria instance. | |
1264 | * | |
1265 | * @return The initial alias. | |
1266 | */ | |
1267 | public String getAlias() { | |
1268 | 0 | return this.entityAliases.get(0); |
1269 | } | |
1270 | ||
1271 | /** | |
1272 | * Retrieves the alias associated with the given index. | |
1273 | * | |
1274 | * @param index The index pointing to a given alias. | |
1275 | * @return The alias at the given index. | |
1276 | */ | |
1277 | public String getAlias(int index) { | |
1278 | 0 | return this.entityAliases.get(index); |
1279 | } | |
1280 | ||
1281 | /** | |
1282 | * Retrieves the index that points to the given alias. | |
1283 | * | |
1284 | * @param alias The indexed alias. | |
1285 | * @return The index of the alias, or -1 if no such index was found. | |
1286 | */ | |
1287 | public int getAliasIndex(String alias) { | |
1288 | 0 | Integer tempIndex = aliasIndexes.get(alias); |
1289 | 0 | return (tempIndex != null) ? tempIndex.intValue() : -1; |
1290 | } | |
1291 | ||
1292 | /** | |
1293 | * Retrieves a copy of all the aliases defined for this Criteria instance. | |
1294 | * The index of a given alias corresponds to the index at which the alias | |
1295 | * is located at in the returned List. | |
1296 | * | |
1297 | * @return A List containing all the defined aliases of this Criteria instance. | |
1298 | */ | |
1299 | public List<String> getAliases() { | |
1300 | 0 | return new ArrayList(this.entityAliases); |
1301 | } | |
1302 | ||
1303 | /** | |
1304 | * @return the name of the class of the initial Entity this Criteria will search for | |
1305 | */ | |
1306 | public String getEntityName() { | |
1307 | 0 | return this.initialEntityName; |
1308 | } | |
1309 | } |