/***************************************************************** * 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.wocompat; import org.apache.cayenne.exp.Expression; import org.apache.cayenne.exp.ExpressionException; import org.apache.cayenne.exp.ExpressionFactory; import org.apache.cayenne.exp.ExpressionParameter; import org.apache.cayenne.exp.parser.ASTObjPath; import org.apache.cayenne.map.Entity; import org.apache.cayenne.map.ObjAttribute; import org.apache.cayenne.map.ObjEntity; import org.apache.cayenne.map.ObjRelationship; import org.apache.cayenne.query.SelectQuery; import org.apache.cayenne.query.SortOrder; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; /** * A descriptor of SelectQuery loaded from EOModel. It is an informal * "decorator" of Cayenne SelectQuery to provide access to the extra information * of WebObjects EOFetchSpecification. * * @since 1.1 */ public class EOQuery<T> extends SelectQuery<T> { protected Map<String, ?> plistMap; protected Map bindings; public EOQuery(ObjEntity root, Map<String, ?> plistMap) { super(root); this.plistMap = plistMap; initFromPlist(plistMap); } protected void initFromPlist(Map<String, ?> plistMap) { setDistinct("YES".equalsIgnoreCase((String) plistMap.get("usesDistinct"))); Object fetchLimit = plistMap.get("fetchLimit"); if (fetchLimit != null) { try { if (fetchLimit instanceof Number) { setFetchLimit(((Number) fetchLimit).intValue()); } else { setFetchLimit(Integer.parseInt(fetchLimit.toString())); } } catch (NumberFormatException nfex) { // ignoring... } } // sort orderings List<Map<String, String>> orderings = (List<Map<String, String>>) plistMap.get("sortOrderings"); if (orderings != null && !orderings.isEmpty()) { for (Map<String, String> ordering : orderings) { boolean asc = !"compareDescending:".equals(ordering.get("selectorName")); String key = ordering.get("key"); if (key != null) { addOrdering(key, asc ? SortOrder.ASCENDING : SortOrder.DESCENDING); } } } // qualifiers Map<String, ?> qualifierMap = (Map<String, ?>) plistMap.get("qualifier"); if (qualifierMap != null && !qualifierMap.isEmpty()) { this.setQualifier(makeQualifier(qualifierMap)); } // prefetches List prefetches = (List) plistMap.get("prefetchingRelationshipKeyPaths"); if (prefetches != null && !prefetches.isEmpty()) { Iterator it = prefetches.iterator(); while (it.hasNext()) { addPrefetch((String) it.next()); } } // data rows - note that we do not support fetching individual columns // in the // modeler... if (plistMap.containsKey("rawRowKeyPaths")) { setFetchingDataRows(true); } } public String getEOName() { if (root instanceof EOObjEntity) { return ((EOObjEntity) root).localQueryName(getName()); } else { return getName(); } } public Collection getBindingNames() { if (bindings == null) { initBindings(); } return bindings.keySet(); } public String bindingClass(String name) { if (bindings == null) { initBindings(); } return (String) bindings.get(name); } private synchronized void initBindings() { if (bindings != null) { return; } bindings = new HashMap(); if (!(root instanceof Entity)) { return; } Map qualifier = (Map) plistMap.get("qualifier"); initBindings(bindings, (Entity) root, qualifier); } private void initBindings(Map bindings, Entity entity, Map qualifier) { if (qualifier == null) { return; } if ("EOKeyValueQualifier".equals(qualifier.get("class"))) { String key = (String) qualifier.get("key"); if (key == null) { return; } Object value = qualifier.get("value"); if (!(value instanceof Map)) { return; } Map valueMap = (Map) value; if (!"EOQualifierVariable".equals(valueMap.get("class")) || !valueMap.containsKey("_key")) { return; } String name = (String) valueMap.get("_key"); String className = null; // we don't know whether its obj path or db path, so the expression // can blow // ... in fact we can't support DB Path as the key is different from // external // name, // so we will use Object type for all DB path... try { Object lastObject = new ASTObjPath(key).evaluate(entity); if (lastObject instanceof ObjAttribute) { className = ((ObjAttribute) lastObject).getType(); } else if (lastObject instanceof ObjRelationship) { ObjEntity target = ((ObjRelationship) lastObject).getTargetEntity(); if (target != null) { className = target.getClassName(); } } } catch (ExpressionException ex) { className = "java.lang.Object"; } if (className == null) { className = "java.lang.Object"; } bindings.put(name, className); return; } List children = (List) qualifier.get("qualifiers"); if (children != null) { Iterator it = children.iterator(); while (it.hasNext()) { initBindings(bindings, entity, (Map) it.next()); } } } /** * Creates the Expression equivalent of the EOFetchSpecification represented * by the Map. * * @param qualifierMap * - FetchSpecification to translate * @return Expression equivalent to FetchSpecification */ public synchronized Expression makeQualifier(Map<String, ?> qualifierMap) { if (qualifierMap == null) { return null; } return EOFetchSpecificationParser.makeQualifier((EOObjEntity) getRoot(), qualifierMap); } /** * EOFetchSpecificationParser parses EOFetchSpecifications from a * WebObjects-style EOModel. It recursively builds Cayenne Expression * objects and assembles them into the final aggregate Expression. */ static class EOFetchSpecificationParser { // Xcode/EOModeler expressions have a colon at the end of the selector // name // (just like standard Objective-C syntax). WOLips does not. Add both // sets to the hash map to handle both types of models. // Selector strings (Java-base). static final String IS_EQUAL_TO = "isEqualTo"; static final String IS_NOT_EQUAL_TO = "isNotEqualTo"; static final String IS_LIKE = "isLike"; static final String CASE_INSENSITIVE_LIKE = "isCaseInsensitiveLike"; static final String IS_LESS_THAN = "isLessThan"; static final String IS_LESS_THAN_OR_EQUAL_TO = "isLessThanOrEqualTo"; static final String IS_GREATER_THAN = "isGreaterThan"; static final String IS_GREATER_THAN_OR_EQUAL_TO = "isGreaterThanOrEqualTo"; private static final String OBJ_C = ":"; // Objective-C syntax addition. private static Map<String, Integer> selectorToExpressionBridge; /** * selectorToExpressionBridge is just a mapping of EOModeler's selector * types to Cayenne Expression types. * * @return HashMap of Expression types, keyed by the corresponding * selector name */ static synchronized Map<String, Integer> selectorToExpressionBridge() { // Initialize selectorToExpressionBridge if needed. if (null == selectorToExpressionBridge) { selectorToExpressionBridge = new HashMap<>(); selectorToExpressionBridge.put(IS_EQUAL_TO, Expression.EQUAL_TO); selectorToExpressionBridge.put(IS_EQUAL_TO + OBJ_C, Expression.EQUAL_TO); selectorToExpressionBridge.put(IS_NOT_EQUAL_TO, Expression.NOT_EQUAL_TO); selectorToExpressionBridge.put(IS_NOT_EQUAL_TO + OBJ_C, Expression.NOT_EQUAL_TO); selectorToExpressionBridge.put(IS_LIKE, Expression.LIKE); selectorToExpressionBridge.put(IS_LIKE + OBJ_C, Expression.LIKE); selectorToExpressionBridge.put(CASE_INSENSITIVE_LIKE, Expression.LIKE_IGNORE_CASE); selectorToExpressionBridge.put(CASE_INSENSITIVE_LIKE + OBJ_C, Expression.LIKE_IGNORE_CASE); selectorToExpressionBridge.put(IS_LESS_THAN, Expression.LESS_THAN); selectorToExpressionBridge.put(IS_LESS_THAN + OBJ_C, Expression.LESS_THAN); selectorToExpressionBridge.put(IS_LESS_THAN_OR_EQUAL_TO, Expression.LESS_THAN_EQUAL_TO); selectorToExpressionBridge.put(IS_LESS_THAN_OR_EQUAL_TO + OBJ_C, Expression.LESS_THAN_EQUAL_TO); selectorToExpressionBridge.put(IS_GREATER_THAN, Expression.GREATER_THAN); selectorToExpressionBridge.put(IS_GREATER_THAN + OBJ_C, Expression.GREATER_THAN); selectorToExpressionBridge.put(IS_GREATER_THAN_OR_EQUAL_TO, Expression.GREATER_THAN_EQUAL_TO); selectorToExpressionBridge.put(IS_GREATER_THAN_OR_EQUAL_TO + OBJ_C, Expression.GREATER_THAN_EQUAL_TO); } return selectorToExpressionBridge; } /** * isAggregate determines whether a qualifier is "aggregate" -- has * children -- or "simple". * * @param qualifier * - a Map containing the qualifier settings * @return boolean indicating whether the qualifier is "aggregate" * qualifier */ static boolean isAggregate(Map qualifier) { boolean result = true; String theClass = (String) qualifier.get("class"); if (theClass == null) { return false; // should maybe throw an exception? } if (theClass.equalsIgnoreCase("EOKeyValueQualifier") || theClass.equalsIgnoreCase("EOKeyComparisonQualifier")) { result = false; } return result; } /** * expressionTypeForQualifier looks at a qualifier containing the * EOModeler FetchSpecification and returns the equivalent Cayenne * Expression type for its selector. * * @param qualifierMap * - a Map containing the qualifier settings to examine. * @return int Expression type */ static int expressionTypeForQualifier(Map qualifierMap) { // get selector String selector = (String) qualifierMap.get("selectorName"); return expressionTypeForSelector(selector); } /** * expressionTypeForSelector looks at a selector from an EOModeler * FetchSpecification and returns the equivalent Cayenne Expression * type. * * @param selector * - a String containing the selector name. * @return int Expression type */ static int expressionTypeForSelector(String selector) { Integer expType = selectorToExpressionBridge().get(selector); return (expType != null ? expType.intValue() : -1); } /** * aggregateExpressionClassForQualifier looks at a qualifer and returns * the aggregate type: one of Expression.AND, Expression.OR, or * Expression.NOT * * @param qualifierMap * - containing the qualifier to examine * @return int aggregate Expression type */ static int aggregateExpressionClassForQualifier(Map qualifierMap) { String qualifierClass = (String) qualifierMap.get("class"); if (qualifierClass != null) { if (qualifierClass.equalsIgnoreCase("EOAndQualifier")) { return Expression.AND; } else if (qualifierClass.equalsIgnoreCase("EOOrQualifier")) { return Expression.OR; } else if (qualifierClass.equalsIgnoreCase("EONotQualifier")) { return Expression.NOT; } } return -1; // error } /** * makeQualifier recursively builds an Expression for each condition in * the qualifierMap and assembles from them the complex Expression to * represent the entire EOFetchSpecification. * * @param qualifierMap * - Map representation of EOFetchSpecification * @return Expression translation of the EOFetchSpecification */ static Expression makeQualifier(EOObjEntity entity, Map qualifierMap) { if (isAggregate(qualifierMap)) { // the fetch specification has more than one qualifier int aggregateClass = aggregateExpressionClassForQualifier(qualifierMap); // AND, // OR, // NOT if (aggregateClass == Expression.NOT) { // NOT qualifiers only have one child, keyed with // "qualifier" Map child = (Map) qualifierMap.get("qualifier"); // build the child expression Expression childExp = makeQualifier(entity, child); return childExp.notExp(); // add the "not" clause and return // the // result } else { // AND, OR qualifiers can have multiple children, keyed with // "qualifiers" // get the list of children List children = (List) qualifierMap.get("qualifiers"); if (children != null) { ArrayList<Expression> childExpressions = new ArrayList<>(); // build an Expression for each child Iterator<Map> it = children.iterator(); while (it.hasNext()) { Expression childExp = makeQualifier(entity, it.next()); childExpressions.add(childExp); } // join the child expressions and return the result return ExpressionFactory.joinExp(aggregateClass, childExpressions); } } } // end if isAggregate(qualifierMap)... // the query has a single qualifier // get expression selector type String qualifierClass = (String) qualifierMap.get("class"); // the key or key path we're comparing String key = null; // the key, keyPath, value, or parameterized value against which // we're // comparing the key Object comparisonValue = null; if ("EOKeyComparisonQualifier".equals(qualifierClass)) { // Comparing two keys or key paths key = (String) qualifierMap.get("leftValue"); comparisonValue = qualifierMap.get("rightValue"); // FIXME: I think EOKeyComparisonQualifier sytle Expressions are // not // supported... return null; } else if ("EOKeyValueQualifier".equals(qualifierClass)) { // Comparing key with a value or parameterized value key = (String) qualifierMap.get("key"); Object value = qualifierMap.get("value"); if (value instanceof Map) { Map<String, String> valueMap = (Map<String, String>) value; String objClass = valueMap.get("class"); // can be a // qualifier class // or java type if ("EOQualifierVariable".equals(objClass) && valueMap.containsKey("_key")) { // make a parameterized expression String paramName = valueMap.get("_key"); comparisonValue = new ExpressionParameter(paramName); } else { Object queryVal = valueMap.get("value"); if ("NSNumber".equals(objClass)) { // comparison to NSNumber -- cast comparisonValue = queryVal; } else if ("EONull".equals(objClass)) { // comparison to null comparisonValue = null; } else { // Could there be other types? boolean, date, // etc.??? // no cast comparisonValue = queryVal; } } } else if (value instanceof String) { // value expression comparisonValue = value; } // end if (value instanceof Map) else... } // check whether the key is an object path; if at least one // component is not, // switch to db path.. Expression keyExp = ExpressionFactory.exp(key); try { entity.lastPathComponent(keyExp, Collections.emptyMap()); } catch (ExpressionException e) { try { keyExp = entity.translateToDbPath(keyExp); } catch (Exception dbpathEx) { return null; } } try { Expression exp = ExpressionFactory.expressionOfType(expressionTypeForQualifier(qualifierMap)); exp.setOperand(0, keyExp); exp.setOperand(1, comparisonValue); return exp; } catch (ExpressionException e) { return null; } } } }