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

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

There is a newer version: 4.2.1
Show newest version
/*****************************************************************
 *   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 java.util.TreeSet;

import org.apache.cayenne.map.DataMap;
import org.apache.cayenne.map.DbEntity;
import org.apache.cayenne.map.EntityResolver;
import org.apache.cayenne.map.MapLoader;
import org.apache.cayenne.map.ObjEntity;
import org.apache.cayenne.map.Procedure;
import org.apache.cayenne.map.SQLResult;
import org.apache.cayenne.util.XMLEncoder;
import org.apache.cayenne.util.XMLSerializable;
import org.apache.commons.collections.IteratorUtils;
import org.apache.commons.collections.Transformer;

/**
 * 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 */ public class SQLTemplate extends AbstractQuery implements ParameterizedQuery, XMLSerializable { static final String COLUMN_NAME_CAPITALIZATION_PROPERTY = "cayenne.SQLTemplate.columnNameCapitalization"; 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; protected CapsStrategy columnNamesCapitalization; protected SQLResult result; SQLTemplateMetadata metaData = new SQLTemplateMetadata(); /** * 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 * @deprecated since 3.1, use SQLTemplate(DataMap rootMap, String defaultTemplate, * boolean isFetchingDataRows) instead */ @Deprecated public SQLTemplate(DataMap rootMap, String defaultTemplate) { this(rootMap, defaultTemplate, false); } /** * @since 3.1 */ public SQLTemplate(DataMap rootMap, String defaultTemplate, boolean isFetchingDataRows) { setDefaultTemplate(defaultTemplate); setRoot(rootMap); setFetchingDataRows(isFetchingDataRows); } /** * @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); } /** * @since 1.2 */ @Override public QueryMetadata getMetaData(EntityResolver resolver) { metaData.resolve(root, resolver, this); return metaData; } /** * Calls sqlAction(this) on the visitor. * * @since 1.2 */ @Override 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(") { rootType = MapLoader.JAVA_CLASS_ROOT; rootString = ((Class) root).getName(); } else if (root instanceof DataMap) { rootType = MapLoader.DATA_MAP_ROOT; rootString = ((DataMap) root).getName(); } if (rootType != null) { encoder.print("\" root=\""); encoder.print(rootType); encoder.print("\" root-name=\""); encoder.print(rootString); } encoder.println("\">"); encoder.indent(1); metaData.encodeAsXML(encoder); if (getColumnNamesCapitalization() != CapsStrategy.DEFAULT) { encoder.printProperty( COLUMN_NAME_CAPITALIZATION_PROPERTY, getColumnNamesCapitalization().name()); } // encode default SQL if (defaultTemplate != null) { encoder.print(""); } // encode adapter SQL if (templates != null && !templates.isEmpty()) { //sorting entries by adapter name TreeSet keys = new TreeSet(templates.keySet()); for (String key : keys) { String value = templates.get(key); if (key != null && value != null) { String sql = value.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 metaData.initWithProperties(properties); if (properties == null) { properties = Collections.EMPTY_MAP; } Object columnNamesCapitalization = properties .get(COLUMN_NAME_CAPITALIZATION_PROPERTY); this.columnNamesCapitalization = (columnNamesCapitalization != null) ? CapsStrategy .valueOf(columnNamesCapitalization.toString().toUpperCase()) : null; } /** * 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) { // create a query replica SQLTemplate query = new SQLTemplate(); query.setRoot(root); query.setDefaultTemplate(getDefaultTemplate()); if (templates != null) { query.templates = new HashMap(templates); } query.metaData.copyFromInfo(this.metaData); query.setParameters(parameters); query.setColumnNamesCapitalization(this.getColumnNamesCapitalization()); 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); } /** * @since 3.0 */ public QueryCacheStrategy getCacheStrategy() { return metaData.getCacheStrategy(); } /** * @since 3.0 */ public void setCacheStrategy(QueryCacheStrategy strategy) { metaData.setCacheStrategy(strategy); } /** * @since 3.0 */ public String[] getCacheGroups() { return metaData.getCacheGroups(); } /** * @since 3.0 */ public void setCacheGroups(String... cacheGroups) { this.metaData.setCacheGroups(cacheGroups); } public int getFetchLimit() { return metaData.getFetchLimit(); } public void setFetchLimit(int fetchLimit) { this.metaData.setFetchLimit(fetchLimit); } /** * @since 3.0 */ public int getFetchOffset() { return metaData.getFetchOffset(); } /** * @since 3.0 */ public void setFetchOffset(int fetchOffset) { metaData.setFetchOffset(fetchOffset); } public int getPageSize() { return metaData.getPageSize(); } public void setPageSize(int pageSize) { metaData.setPageSize(pageSize); } public void setFetchingDataRows(boolean flag) { metaData.setFetchingDataRows(flag); } public boolean isFetchingDataRows() { return metaData.isFetchingDataRows(); } /** * 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 = 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) ? templates.get(key) : null; } /** * Adds a SQL template string for a given key. Note the the keys understood by Cayenne * must be fully qualified adapter class names. This way the framework can related * current DataNode to the right template. E.g. * "org.apache.cayenne.dba.oracle.OracleAdapter" is a key that should be used to setup * an Oracle-specific template. * * @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 one or more sets of parameters. */ 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(); } } } /** * @since 1.2 */ public PrefetchTreeNode getPrefetchTree() { return metaData.getPrefetchTree(); } /** * Adds a prefetch. * * @since 1.2 */ public PrefetchTreeNode addPrefetch(String prefetchPath) { // by default use JOINT_PREFETCH_SEMANTICS return metaData.addPrefetch( prefetchPath, PrefetchTreeNode.JOINT_PREFETCH_SEMANTICS); } /** * @since 1.2 */ public void removePrefetch(String prefetch) { metaData.removePrefetch(prefetch); } /** * Adds all prefetches from a provided collection. * * @since 1.2 */ public void addPrefetches(Collection prefetches) { metaData.addPrefetches(prefetches, PrefetchTreeNode.JOINT_PREFETCH_SEMANTICS); } /** * Clears all prefetches. * * @since 1.2 */ public void clearPrefetches() { metaData.clearPrefetches(); } /** * Returns a column name capitalization policy applied to selecting queries. This is * used to simplify mapping of the queries like "SELECT * FROM ...", ensuring that a * chosen Cayenne column mapping strategy (e.g. all column names in uppercase) is * portable across database engines that can have varying default capitalization. * Default (null) value indicates that column names provided in result set are used * unchanged. * * @since 3.0 */ public CapsStrategy getColumnNamesCapitalization() { return columnNamesCapitalization != null ? columnNamesCapitalization : CapsStrategy.DEFAULT; } /** * Sets a column name capitalization policy applied to selecting queries. This is used * to simplify mapping of the queries like "SELECT * FROM ...", ensuring that a chosen * Cayenne column mapping strategy (e.g. all column names in uppercase) is portable * across database engines that can have varying default capitalization. Default * (null) value indicates that column names provided in result set are used unchanged. *

* Note that while a non-default setting is useful for queries that do not rely on a * #result directive to describe columns, it works for all SQLTemplates the same way. * * @since 3.0 */ public void setColumnNamesCapitalization(CapsStrategy columnNameCapitalization) { this.columnNamesCapitalization = columnNameCapitalization; } /** * Sets an optional explicit mapping of the result set. If result set mapping is * specified, the result of SQLTemplate may not be a normal list of Persistent objects * or DataRows, instead it will follow the {@link SQLResult} rules. * * @since 3.0 */ public void setResult(SQLResult resultSet) { this.result = resultSet; } /** * @since 3.0 */ public SQLResult getResult() { return result; } /** * Sets statement's fetch size (0 for no default size) * * @since 3.0 */ public void setStatementFetchSize(int size) { metaData.setStatementFetchSize(size); } /** * @return statement's fetch size * @since 3.0 */ public int getStatementFetchSize() { return metaData.getStatementFetchSize(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy