org.apache.cayenne.query.SQLTemplate Maven / Gradle / Ivy
Show all versions of cayenne-client-nodeps
/*****************************************************************
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.cayenne.query;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import org.apache.commons.collections.IteratorUtils;
import org.apache.commons.collections.Transformer;
import org.apache.cayenne.map.DataMap;
import org.apache.cayenne.map.DbEntity;
import org.apache.cayenne.map.EntityResolver;
import org.apache.cayenne.map.ObjEntity;
import org.apache.cayenne.map.Procedure;
import org.apache.cayenne.map.QueryBuilder;
import org.apache.cayenne.util.Util;
import org.apache.cayenne.util.XMLEncoder;
import org.apache.cayenne.util.XMLSerializable;
/**
* A query that executes unchanged (except for template preprocessing) "raw" SQL specified
* by the user.
* Template Script
*
* SQLTemplate stores a dynamic template for the SQL query that supports parameters and
* customization using Velocity scripting language. The most straightforward use of
* scripting abilities is to build parameterized queries. For example:
*
*
*
* SELECT ID, NAME FROM SOME_TABLE WHERE NAME LIKE $a
*
*
*
* For advanced scripting options see "Scripting SQLTemplate" chapter in the User
* Guide.
*
* Per-Database Template Customization
*
* SQLTemplate has a {@link #getDefaultTemplate() default template script}, but also it
* allows to configure multiple templates and switch them dynamically. This way a single
* query can have multiple "dialects" specific to a given database.
*
* Parameter Sets
*
* SQLTemplate supports multiple sets of parameters, so a single query can be executed
* multiple times with different parameters. "Scrolling" through parameter list is done by
* calling {@link #parametersIterator()}. This iterator goes over parameter sets,
* returning a Map on each call to "next()"
*
*
* @since 1.1
* @author Andrus Adamchik
*/
public class SQLTemplate extends AbstractQuery implements GenericSelectQuery,
ParameterizedQuery, XMLSerializable {
private static final Transformer nullMapTransformer = new Transformer() {
public Object transform(Object input) {
return (input != null) ? input : Collections.EMPTY_MAP;
}
};
protected String defaultTemplate;
protected Map templates;
protected Map[] parameters;
BaseQueryMetadata selectInfo = new BaseQueryMetadata();
/**
* @deprecated Since 1.2 this property is redundant.
*/
protected boolean selecting;
/**
* Creates an empty SQLTemplate. Note this constructor does not specify the "root" of
* the query, so a user must call "setRoot" later to make sure SQLTemplate can be
* executed.
*
* @since 1.2
*/
public SQLTemplate() {
}
/**
* @since 1.2
*/
public SQLTemplate(DataMap rootMap, String defaultTemplate) {
setDefaultTemplate(defaultTemplate);
setRoot(rootMap);
}
/**
* @since 1.2
*/
public SQLTemplate(ObjEntity rootEntity, String defaultTemplate) {
setDefaultTemplate(defaultTemplate);
setRoot(rootEntity);
}
/**
* @since 1.2
*/
public SQLTemplate(Class rootClass, String defaultTemplate) {
setDefaultTemplate(defaultTemplate);
setRoot(rootClass);
}
/**
* @since 1.2
*/
public SQLTemplate(DbEntity rootEntity, String defaultTemplate) {
setDefaultTemplate(defaultTemplate);
setRoot(rootEntity);
}
/**
* @since 1.2
*/
public SQLTemplate(String objEntityName, String defaultTemplate) {
setRoot(objEntityName);
setDefaultTemplate(defaultTemplate);
}
/**
* Creates an empty SQLTemplate. Note this constructor does not specify the "root" of
* the query, so a user must call "setRoot" later to make sure SQLTemplate can be
* executed.
*
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public SQLTemplate(boolean selecting) {
this();
setSelecting(selecting);
}
/**
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public SQLTemplate(DataMap rootMap, String defaultTemplate, boolean selecting) {
this(rootMap, defaultTemplate);
setSelecting(selecting);
}
/**
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public SQLTemplate(ObjEntity rootEntity, String defaultTemplate, boolean selecting) {
this(rootEntity, defaultTemplate);
setSelecting(selecting);
}
/**
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public SQLTemplate(Class rootClass, String defaultTemplate, boolean selecting) {
this(rootClass, defaultTemplate);
setSelecting(selecting);
}
/**
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public SQLTemplate(DbEntity rootEntity, String defaultTemplate, boolean selecting) {
this(rootEntity, defaultTemplate);
setSelecting(selecting);
}
/**
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public SQLTemplate(String objEntityName, String defaultTemplate, boolean selecting) {
this(objEntityName, defaultTemplate);
setSelecting(selecting);
}
/**
* @since 1.2
*/
public QueryMetadata getMetaData(EntityResolver resolver) {
selectInfo.resolve(root, resolver, getName());
return selectInfo;
}
/**
* Calls sqlAction(this) on the visitor.
*
* @since 1.2
*/
public SQLAction createSQLAction(SQLActionVisitor visitor) {
return visitor.sqlAction(this);
}
/**
* Prints itself as XML to the provided PrintWriter.
*
* @since 1.1
*/
public void encodeAsXML(XMLEncoder encoder) {
encoder.print("");
encoder.indent(1);
selectInfo.encodeAsXML(encoder);
// encode default SQL
if (defaultTemplate != null) {
encoder.print(" ");
}
// encode adapter SQL
if (templates != null && !templates.isEmpty()) {
Iterator it = templates.entrySet().iterator();
while (it.hasNext()) {
Map.Entry entry = (Map.Entry) it.next();
Object key = entry.getKey();
Object value = entry.getValue();
if (key != null && value != null) {
String sql = value.toString().trim();
if (sql.length() > 0) {
encoder.print(" ");
}
}
}
}
// TODO: support parameter encoding
encoder.indent(-1);
encoder.println(" ");
}
/**
* Initializes query parameters using a set of properties.
*
* @since 1.1
*/
public void initWithProperties(Map properties) {
// must init defaults even if properties are empty
selectInfo.initWithProperties(properties);
}
/**
* Returns an iterator over parameter sets. Each element returned from the iterator is
* a java.util.Map.
*/
public Iterator parametersIterator() {
return (parameters == null || parameters.length == 0) ? IteratorUtils
.emptyIterator() : IteratorUtils.transformedIterator(IteratorUtils
.arrayIterator(parameters), nullMapTransformer);
}
/**
* Returns the number of parameter sets.
*/
public int parametersSize() {
return (parameters != null) ? parameters.length : 0;
}
/**
* Returns a new query built using this query as a prototype and a new set of
* parameters.
*/
public SQLTemplate queryWithParameters(Map parameters) {
return queryWithParameters(new Map[] {
parameters
});
}
/**
* Returns a new query built using this query as a prototype and a new set of
* parameters.
*/
public SQLTemplate queryWithParameters(Map[] parameters) {
// create a query replica
SQLTemplate query = new SQLTemplate();
query.setRoot(root);
query.setDefaultTemplate(getDefaultTemplate());
if (templates != null) {
query.templates = new HashMap(templates);
}
query.selectInfo.copyFromInfo(this.selectInfo);
query.setParameters(parameters);
// The following algorithm is for building the new query name based
// on the original query name and a hashcode of the map of parameters.
// This way the query clone can take advantage of caching. Fixes
// problem reported in CAY-360.
if (!Util.isEmptyString(name)) {
StringBuffer buffer = new StringBuffer(name);
if (parameters != null) {
for (int i = 0; i < parameters.length; i++) {
if (!parameters[i].isEmpty()) {
buffer.append(parameters[i].hashCode());
}
}
}
query.setName(buffer.toString());
}
return query;
}
/**
* Creates and returns a new SQLTemplate built using this query as a prototype and
* substituting template parameters with the values from the map.
*
* @since 1.1
*/
public Query createQuery(Map parameters) {
return queryWithParameters(parameters);
}
public String getCachePolicy() {
return selectInfo.getCachePolicy();
}
public void setCachePolicy(String policy) {
this.selectInfo.setCachePolicy(policy);
}
public int getFetchLimit() {
return selectInfo.getFetchLimit();
}
public void setFetchLimit(int fetchLimit) {
this.selectInfo.setFetchLimit(fetchLimit);
}
public int getPageSize() {
return selectInfo.getPageSize();
}
public void setPageSize(int pageSize) {
selectInfo.setPageSize(pageSize);
}
public void setFetchingDataRows(boolean flag) {
selectInfo.setFetchingDataRows(flag);
}
public boolean isFetchingDataRows() {
return selectInfo.isFetchingDataRows();
}
public boolean isRefreshingObjects() {
return selectInfo.isRefreshingObjects();
}
public void setRefreshingObjects(boolean flag) {
selectInfo.setRefreshingObjects(flag);
}
public boolean isResolvingInherited() {
return selectInfo.isResolvingInherited();
}
public void setResolvingInherited(boolean b) {
selectInfo.setResolvingInherited(b);
}
/**
* Returns default SQL template for this query.
*/
public String getDefaultTemplate() {
return defaultTemplate;
}
/**
* Sets default SQL template for this query.
*/
public void setDefaultTemplate(String string) {
defaultTemplate = string;
}
/**
* Returns a template for key, or a default template if a template for key is not
* found.
*/
public synchronized String getTemplate(String key) {
if (templates == null) {
return defaultTemplate;
}
String template = (String) templates.get(key);
return (template != null) ? template : defaultTemplate;
}
/**
* Returns template for key, or null if there is no template configured for this key.
* Unlike {@link #getTemplate(String)}this method does not return a default template
* as a failover strategy, rather it returns null.
*/
public synchronized String getCustomTemplate(String key) {
return (templates != null) ? (String) templates.get(key) : null;
}
/**
* Adds a SQL template string for a given key.
*
* @see #setDefaultTemplate(String)
*/
public synchronized void setTemplate(String key, String template) {
if (templates == null) {
templates = new HashMap();
}
templates.put(key, template);
}
public synchronized void removeTemplate(String key) {
if (templates != null) {
templates.remove(key);
}
}
/**
* Returns a collection of configured template keys.
*/
public synchronized Collection getTemplateKeys() {
return (templates != null) ? Collections.unmodifiableCollection(templates
.keySet()) : Collections.EMPTY_LIST;
}
/**
* Utility method to get the first set of parameters, since most queries will only
* have one.
*/
public Map getParameters() {
Map map = (parameters != null && parameters.length > 0) ? parameters[0] : null;
return (map != null) ? map : Collections.EMPTY_MAP;
}
/**
* Utility method to initialize query with only a single set of parameters. Useful,
* since most queries will only have one set. Internally calls
* {@link #setParameters(Map[])}.
*/
public void setParameters(Map map) {
setParameters(map != null ? new Map[] {
map
} : null);
}
public void setParameters(Map[] parameters) {
if (parameters == null) {
this.parameters = null;
}
else {
// clone parameters to ensure that we don't have immutable maps that are not
// serializable with Hessian...
this.parameters = new Map[parameters.length];
for (int i = 0; i < parameters.length; i++) {
this.parameters[i] = parameters[i] != null
? new HashMap(parameters[i])
: new HashMap();
}
}
}
/**
* Returns true if SQLTemplate is expected to return a ResultSet.
*
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public boolean isSelecting() {
return selecting;
}
/**
* Sets whether SQLTemplate is expected to return a ResultSet.
*
* @deprecated Since 1.2 'selecting' property is redundant.
*/
public void setSelecting(boolean b) {
selecting = b;
}
/**
* @since 1.2
*/
public PrefetchTreeNode getPrefetchTree() {
return selectInfo.getPrefetchTree();
}
/**
* Adds a prefetch.
*
* @since 1.2
*/
public PrefetchTreeNode addPrefetch(String prefetchPath) {
// by default use JOINT_PREFETCH_SEMANTICS
return selectInfo.addPrefetch(
prefetchPath,
PrefetchTreeNode.JOINT_PREFETCH_SEMANTICS);
}
/**
* @since 1.2
*/
public void removePrefetch(String prefetch) {
selectInfo.removePrefetch(prefetch);
}
/**
* Adds all prefetches from a provided collection.
*
* @since 1.2
*/
public void addPrefetches(Collection prefetches) {
selectInfo.addPrefetches(prefetches, PrefetchTreeNode.JOINT_PREFETCH_SEMANTICS);
}
/**
* Clears all prefetches.
*
* @since 1.2
*/
public void clearPrefetches() {
selectInfo.clearPrefetches();
}
}