All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.objectstyle.cayenne.query.SQLTemplate Maven / Gradle / Ivy

There is a newer version: 1.2.4
Show newest version
/* ====================================================================
 * 
 * The ObjectStyle Group Software License, version 1.1
 * ObjectStyle Group - http://objectstyle.org/
 * 
 * Copyright (c) 2002-2005, Andrei (Andrus) Adamchik and individual authors
 * of the software. All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 * 
 * 3. The end-user documentation included with the redistribution, if any,
 *    must include the following acknowlegement:
 *    "This product includes software developed by independent contributors
 *    and hosted on ObjectStyle Group web site (http://objectstyle.org/)."
 *    Alternately, this acknowlegement may appear in the software itself,
 *    if and wherever such third-party acknowlegements normally appear.
 * 
 * 4. The names "ObjectStyle Group" and "Cayenne" must not be used to endorse
 *    or promote products derived from this software without prior written
 *    permission. For written permission, email
 *    "andrus at objectstyle dot org".
 * 
 * 5. Products derived from this software may not be called "ObjectStyle"
 *    or "Cayenne", nor may "ObjectStyle" or "Cayenne" appear in their
 *    names without prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED.  IN NO EVENT SHALL THE OBJECTSTYLE GROUP OR
 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * ====================================================================
 * 
 * This software consists of voluntary contributions made by many
 * individuals and hosted on ObjectStyle Group web site.  For more
 * information on the ObjectStyle Group, please see
 * .
 */

package org.objectstyle.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.objectstyle.cayenne.map.DataMap;
import org.objectstyle.cayenne.map.DbEntity;
import org.objectstyle.cayenne.map.EntityResolver;
import org.objectstyle.cayenne.map.ObjEntity;
import org.objectstyle.cayenne.map.Procedure;
import org.objectstyle.cayenne.map.QueryBuilder;
import org.objectstyle.cayenne.util.Util;
import org.objectstyle.cayenne.util.XMLEncoder;
import org.objectstyle.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(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy