OPENJPA-806: Refactor different domain paths into a single list in QueryDefinitionImpl.

git-svn-id: https://svn.apache.org/repos/asf/openjpa/trunk@724564 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Pinaki Poddar 2008-12-09 00:17:23 +00:00
parent 9946b6c19b
commit 411382d4d9
14 changed files with 182 additions and 152 deletions

View File

@ -118,7 +118,9 @@ public class TestCriteria extends SingleEMFTestCase {
.where(photo.key().like("egret")); .where(photo.key().like("egret"));
String jpql = "select i.name, VALUE(p) from Item i join i.photos p where KEY(p) like 'egret'"; String jpql = "select i.name, VALUE(p)"
+ " from Item i join i.photos p"
+ " where KEY(p) like 'egret'";
compare(jpql, qdef); compare(jpql, qdef);
} }
@ -444,6 +446,33 @@ public class TestCriteria extends SingleEMFTestCase {
compare(jpql, e); compare(jpql, e);
} }
public void testCorrelatedSubquerySpecialCase1() {
DomainObject o = qb.createQueryDefinition(Order.class);
DomainObject a = qb.createSubqueryDefinition(o.get("customer").get("accounts"));
o.select(o)
.where(o.literal(10000).lessThan(a.select(a.get("balance")).all()));
String jpql = "select o from Order o"
+ " where 10000 < ALL "
+ " (select a.balance from o.customer c join o.customer.accounts a)";
compare(jpql, o);
}
public void testCorrelatedSubquerySpecialCase2() {
DomainObject o = qb.createQueryDefinition(Order.class);
DomainObject c = o.join("customer");
DomainObject a = qb.createSubqueryDefinition(c.get("accounts"));
o.select(o)
.where(o.literal(10000).lessThan(a.select(a.get("balance")).all()));
String jpql = "select o from Order o JOIN o.customer c"
+ " where 10000 < ALL "
+ " (select a.balance from c.accounts a)";
compare(jpql, o);
}
public void testRecursiveDefinitionIsNotAllowed() { public void testRecursiveDefinitionIsNotAllowed() {
DomainObject q = qb.createQueryDefinition(Customer.class); DomainObject q = qb.createQueryDefinition(Customer.class);
q.where(q.exists().and(q.get("name").equal("wrong"))); q.where(q.exists().and(q.get("name").equal("wrong")));

View File

@ -18,7 +18,6 @@
*/ */
package org.apache.openjpa.persistence.query; package org.apache.openjpa.persistence.query;
import java.util.ArrayList;
import java.util.Calendar; import java.util.Calendar;
import java.util.Date; import java.util.Date;
import java.util.List; import java.util.List;
@ -43,36 +42,10 @@ import javax.persistence.Subquery;
*/ */
public abstract class AbstractDomainObject extends AbstractPath public abstract class AbstractDomainObject extends AbstractPath
implements DomainObject { implements DomainObject {
private final QueryDefinitionImpl _owner;
private List<JoinPath> _joins;
private List<FetchPath> _fetchJoins;
protected AbstractDomainObject(QueryDefinitionImpl owner, protected AbstractDomainObject(QueryDefinitionImpl owner,
AbstractPath parent, PathOperator op, Object part2) { AbstractPath parent, PathOperator op, Object part2) {
super(parent, op, part2); super(owner, parent, op, part2);
_owner = owner;
}
/**
* Gets the QueryDefinition that created this path.
* @return
*/
public QueryDefinitionImpl getOwner() {
return _owner;
}
/**
* Gets the fetch joins associated with this path. Can be null.
*/
public List<FetchPath> getFetchJoins() {
return _fetchJoins;
}
/**
* Gets the joins associated with this path. Can be null.
*/
public List<JoinPath> getJoins() {
return _joins;
} }
/** /**
@ -100,49 +73,32 @@ public abstract class AbstractDomainObject extends AbstractPath
* Derives a path from this path by joining the given field. * Derives a path from this path by joining the given field.
* Also the joined path becomes a domain of the owning query. * Also the joined path becomes a domain of the owning query.
*/ */
public DomainObject join(String attribute) { public DomainObject join(String attr) {
return join(attribute, PathOperator.INNER); return _owner.addDomain(new JoinPath(this, PathOperator.INNER, attr));
} }
/** /**
* Derives a path from this path by outer joining the given field. * Derives a path from this path by outer joining the given field.
* Also the joined path becomes a domain of the owning query. * Also the joined path becomes a domain of the owning query.
*/ */
public DomainObject leftJoin(String attribute) { public DomainObject leftJoin(String attr) {
return join(attribute, PathOperator.OUTER); return _owner.addDomain(new JoinPath(this, PathOperator.OUTER, attr));
}
protected DomainObject join(String attr, PathOperator joinType) {
JoinPath join = new JoinPath(this, joinType, attr);
if (_joins == null) {
_joins = new ArrayList<JoinPath>();
}
_joins.add(join);
return join;
} }
/** /**
* Derives a path from this path by fetch joining the given field. * Derives a path from this path by fetch joining the given field.
*/ */
public FetchJoinObject joinFetch(String attribute) { public FetchJoinObject joinFetch(String attr) {
return fetchJoin(attribute, PathOperator.FETCH_INNER); return _owner.addDomain(new FetchPath(this, PathOperator.FETCH_INNER,
attr));
} }
/** /**
* Derives a path from this path by fetch joining the given field. * Derives a path from this path by fetch joining the given field.
*/ */
public FetchJoinObject leftJoinFetch(String attribute) { public FetchJoinObject leftJoinFetch(String attr) {
return fetchJoin(attribute, PathOperator.FETCH_OUTER); return _owner.addDomain(new FetchPath(this, PathOperator.FETCH_OUTER,
} attr));
private FetchJoinObject fetchJoin(String attr, PathOperator joinType) {
NavigationPath path = new NavigationPath(_owner, this, attr);
FetchPath join = new FetchPath(path, joinType);
if (_fetchJoins == null) {
_fetchJoins = new ArrayList<FetchPath>();
}
_fetchJoins.add(join);
return join;
} }
/** /**
@ -376,19 +332,4 @@ public abstract class AbstractDomainObject extends AbstractPath
public QueryDefinition where(Predicate predicate) { public QueryDefinition where(Predicate predicate) {
return _owner.where(predicate); return _owner.where(predicate);
} }
// -----------------------------------------------------------------------
// contract for conversion to JPQL.
// -----------------------------------------------------------------------
/**
* Sets alias for this domain and all its joins.
*/
public void setAlias(AliasContext ctx) {
ctx.getAlias(this);
if (_joins != null)
for (JoinPath join : _joins)
join.setAlias(ctx);
}
} }

View File

@ -18,6 +18,8 @@
*/ */
package org.apache.openjpa.persistence.query; package org.apache.openjpa.persistence.query;
import java.util.LinkedList;
import javax.persistence.Aggregate; import javax.persistence.Aggregate;
import javax.persistence.Expression; import javax.persistence.Expression;
import javax.persistence.PathExpression; import javax.persistence.PathExpression;
@ -45,8 +47,10 @@ abstract class AbstractPath extends ExpressionImpl implements
protected final AbstractPath _parent; protected final AbstractPath _parent;
protected final Object _part2; protected final Object _part2;
protected final PathOperator _operator; protected final PathOperator _operator;
protected final QueryDefinitionImpl _owner;
protected AbstractPath(AbstractPath parent, PathOperator op, Object part2) { protected AbstractPath(QueryDefinitionImpl owner, AbstractPath parent, PathOperator op, Object part2) {
_owner = owner;
_parent = parent; _parent = parent;
_part2 = part2; _part2 = part2;
_operator = op; _operator = op;
@ -55,6 +59,10 @@ abstract class AbstractPath extends ExpressionImpl implements
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
// Path related functions. // Path related functions.
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
final QueryDefinitionImpl getOwner() {
return _owner;
}
/** /**
* Gets the parent from which this receiver has been derived. Can be null * Gets the parent from which this receiver has been derived. Can be null
* for a root path. * for a root path.
@ -112,4 +120,17 @@ abstract class AbstractPath extends ExpressionImpl implements
public Expression type() { public Expression type() {
return new TypeExpression(this); return new TypeExpression(this);
} }
LinkedList<AbstractPath> split() {
return _split(this, new LinkedList<AbstractPath>());
}
private LinkedList<AbstractPath> _split(AbstractPath path,
LinkedList<AbstractPath> list) {
if (path == null)
return list;
_split(path.getParent(), list);
list.add(path);
return list;
}
} }

View File

@ -39,10 +39,6 @@ abstract class AbstractVisitable implements Visitable {
throw new UnsupportedOperationException(this.getClass().getName()); throw new UnsupportedOperationException(this.getClass().getName());
} }
public void setAlias(AliasContext ctx) {
throw new UnsupportedOperationException(this.getClass().getName());
}
public String asJoinable(AliasContext ctx) { public String asJoinable(AliasContext ctx) {
throw new UnsupportedOperationException(this.getClass().getName()); throw new UnsupportedOperationException(this.getClass().getName());
} }

View File

@ -49,7 +49,13 @@ class AliasContext {
String alias = _aliases.get(path); String alias = _aliases.get(path);
if (alias != null) if (alias != null)
return alias; return alias;
alias = path.getAliasHint(this).substring(0,1).toLowerCase(); return setAlias(path);
}
public String setAlias(ExpressionImpl path) {
if (_aliases.containsKey(path))
return _aliases.get(path);
String alias = path.getAliasHint(this).substring(0,1).toLowerCase();
int i = 2; int i = 2;
while (_aliases.containsValue(alias)) { while (_aliases.containsValue(alias)) {
alias = alias.substring(0,1) + i; alias = alias.substring(0,1) + i;

View File

@ -462,14 +462,7 @@ abstract class ExpressionImpl extends AbstractVisitable
// //
// Visitable/Selectable implementation // Visitable/Selectable implementation
// //
public void setAlias(AliasContext ctx) {
ctx.getAlias(this);
}
public String getAliasHint(AliasContext ctx) { public String getAliasHint(AliasContext ctx) {
return "o"; return "o";
} }
public abstract String asExpression(AliasContext ctx);
public abstract String asProjection(AliasContext ctx);
} }

View File

@ -18,6 +18,8 @@
*/ */
package org.apache.openjpa.persistence.query; package org.apache.openjpa.persistence.query;
import static org.apache.openjpa.persistence.query.PathOperator.NAVIGATION;
import javax.persistence.FetchJoinObject; import javax.persistence.FetchJoinObject;
/** /**
@ -26,18 +28,21 @@ import javax.persistence.FetchJoinObject;
* @author Pinaki Poddar * @author Pinaki Poddar
* *
*/ */
public class FetchPath extends AbstractVisitable public class FetchPath extends AbstractDomainObject
implements FetchJoinObject, Visitable { implements FetchJoinObject, Visitable {
private NavigationPath _path; FetchPath(AbstractDomainObject parent, PathOperator joinType, String attr) {
private PathOperator _joinType; super(parent.getOwner(), parent, joinType, attr);
FetchPath(NavigationPath path, PathOperator joinType) {
_path = path;
_joinType = joinType;
} }
public String asExpression(AliasContext ctx) { @Override
return _joinType + " " + _path.asExpression(ctx); public String asJoinable(AliasContext ctx) {
StringBuffer tmp = new StringBuffer(getOperator().toString());
tmp.append(getParent().asProjection(ctx))
.append(NAVIGATION)
.append(getLastSegment())
.append(" ");
return tmp.toString();
} }
} }

View File

@ -49,17 +49,12 @@ public class JoinPath extends AbstractDomainObject implements DomainObject {
@Override @Override
public String asJoinable(AliasContext ctx) { public String asJoinable(AliasContext ctx) {
StringBuffer tmp = new StringBuffer(getOperator().toString()); return new StringBuffer(getOperator().toString())
tmp.append(getParent().asProjection(ctx)) .append(getParent().asProjection(ctx))
.append(NAVIGATION) .append(NAVIGATION)
.append(getLastSegment()) .append(getLastSegment())
.append(" ") .append(" ")
.append(ctx.getAlias(this)); .append(ctx.getAlias(this)).toString();
if (getJoins() != null)
for (JoinPath join : getJoins())
tmp.append(join.asJoinable(ctx));
return tmp.toString();
} }
@Override @Override
@ -75,4 +70,9 @@ public class JoinPath extends AbstractDomainObject implements DomainObject {
return ctx.getAlias(this); return ctx.getAlias(this);
return getParent().asProjection(ctx) + NAVIGATION + getLastSegment(); return getParent().asProjection(ctx) + NAVIGATION + getLastSegment();
} }
public String toString() {
return getOperator() + getParent().toString() + "*" + getLastSegment();
}
} }

View File

@ -37,6 +37,11 @@ class NavigationPath extends AbstractDomainObject implements PathExpression {
return (String)super.getLastSegment(); return (String)super.getLastSegment();
} }
@Override
public String getAliasHint(AliasContext ctx) {
return getLastSegment();
}
@Override @Override
public String asProjection(AliasContext ctx) { public String asProjection(AliasContext ctx) {
AbstractPath parent = getParent(); AbstractPath parent = getParent();
@ -59,4 +64,9 @@ class NavigationPath extends AbstractDomainObject implements PathExpression {
public String asJoinable(AliasContext ctx) { public String asJoinable(AliasContext ctx) {
return asProjection(ctx) + " " + ctx.getAlias(this); return asProjection(ctx) + " " + ctx.getAlias(this);
} }
public String toString() {
return getParent().toString()+"."+getLastSegment();
}
} }

View File

@ -55,4 +55,9 @@ public class OperatorPath extends AbstractDomainObject {
public String asJoinable(AliasContext ctx) { public String asJoinable(AliasContext ctx) {
throw new UnsupportedOperationException(); throw new UnsupportedOperationException();
} }
public String toString() {
return getOperator() + "(" + getParent().toString() + ")";
}
} }

View File

@ -18,17 +18,12 @@
*/ */
package org.apache.openjpa.persistence.query; package org.apache.openjpa.persistence.query;
import java.io.Serializable;
import javax.persistence.DomainObject; import javax.persistence.DomainObject;
import javax.persistence.PathExpression; import javax.persistence.PathExpression;
import javax.persistence.QueryBuilder; import javax.persistence.QueryBuilder;
import javax.persistence.QueryDefinition; import javax.persistence.QueryDefinition;
import org.apache.openjpa.meta.ClassMetaData;
import org.apache.openjpa.meta.MetaDataRepository; import org.apache.openjpa.meta.MetaDataRepository;
import org.apache.openjpa.persistence.EntityManagerFactoryImpl;
import org.apache.openjpa.persistence.OpenJPAEntityManagerFactory;
import org.apache.openjpa.persistence.OpenJPAEntityManagerFactorySPI; import org.apache.openjpa.persistence.OpenJPAEntityManagerFactorySPI;
/** /**

View File

@ -23,7 +23,10 @@ import java.sql.Timestamp;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.Calendar; import java.util.Calendar;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date; import java.util.Date;
import java.util.LinkedList;
import java.util.List; import java.util.List;
import javax.persistence.CaseExpression; import javax.persistence.CaseExpression;
@ -79,16 +82,34 @@ public class QueryDefinitionImpl extends ExpressionImpl
public DomainObject addSubqueryRoot(PathExpression path) { public DomainObject addSubqueryRoot(PathExpression path) {
AbstractPath impl = (AbstractPath)path; AbstractPath impl = (AbstractPath)path;
AbstractDomainObject newRoot = new NavigationPath(this, LinkedList<AbstractPath> paths = impl.split();
impl.getParent(), impl.getLastSegment().toString()); QueryDefinitionImpl owner = impl.getOwner();
addDomain(newRoot); int i = 0;
while (i < paths.size() && owner.hasDomain(paths.get(i))) {
i++;
}
AbstractPath next = paths.get(i);
DomainObject newRoot = new NavigationPath(this,
next.getParent(), next.getLastSegment().toString());
addDomain((AbstractDomainObject)newRoot);
i++;
for (; i < paths.size(); i++) {
next = paths.get(i);
newRoot = newRoot.join(next.getLastSegment().toString());
}
return newRoot; return newRoot;
} }
protected void addDomain(AbstractDomainObject path) { boolean hasDomain(PathExpression path) {
return _domains != null && _domains.contains(path);
}
protected <T extends AbstractDomainObject> T addDomain(T path) {
if (_domains == null) if (_domains == null)
_domains = new ArrayList<AbstractDomainObject>(); _domains = new ArrayList<AbstractDomainObject>();
_domains.add(path); _domains.add(path);
return path;
} }
public Subquery all() { public Subquery all() {
@ -138,6 +159,8 @@ public class QueryDefinitionImpl extends ExpressionImpl
public QueryDefinition groupBy(PathExpression... pathExprs) { public QueryDefinition groupBy(PathExpression... pathExprs) {
if (_groupBys == null) { if (_groupBys == null) {
_groupBys = new ArrayList<PathExpression>(); _groupBys = new ArrayList<PathExpression>();
} else {
_groupBys.clear();
} }
for (PathExpression e : pathExprs) for (PathExpression e : pathExprs)
_groupBys.add(e); _groupBys.add(e);
@ -147,6 +170,8 @@ public class QueryDefinitionImpl extends ExpressionImpl
public QueryDefinition groupBy(List<PathExpression> pathExprList) { public QueryDefinition groupBy(List<PathExpression> pathExprList) {
if (_groupBys == null) { if (_groupBys == null) {
_groupBys = new ArrayList<PathExpression>(); _groupBys = new ArrayList<PathExpression>();
} else {
_groupBys.clear();
} }
for (PathExpression e : pathExprList) for (PathExpression e : pathExprList)
_groupBys.add(e); _groupBys.add(e);
@ -336,38 +361,17 @@ public class QueryDefinitionImpl extends ExpressionImpl
return _projections; return _projections;
} }
/**
*
*/
@Override @Override
public String asExpression(AliasContext ctx) { public String asExpression(AliasContext ctx) {
ctx.push(this); ctx.push(this);
StringBuffer buffer = new StringBuffer(); StringBuffer buffer = new StringBuffer();
registerDomains(ctx); registerDomains(ctx);
String select = _distinct ? "SELECT DISTINCT " : "SELECT ";
fillBuffer(_distinct ? "SELECT DISTINCT " : "SELECT ", buffer, ctx, fillBuffer(select, buffer, ctx, getProjections(), Visit.PROJECTION);
getProjections(), Visit.PROJECTION); fillBuffer(" FROM ", buffer, ctx, _domains, Visit.JOINABLE);
buffer.append(" FROM "); fillBuffer(" WHERE ", buffer, ctx, _where);
for (int i=0; _domains != null && i < _domains.size(); i++) {
buffer.append(_domains.get(i).asJoinable(ctx));
fillBuffer(" ", buffer, ctx, _domains.get(i).getJoins(),
Visit.JOINABLE);
fillBuffer(" ", buffer, ctx, _domains.get(i).getFetchJoins(),
Visit.EXPRESSION);
if (i != _domains.size()-1)
buffer.append(",");
}
if (_where != null) {
buffer.append(" WHERE ")
.append(((Visitable)_where).asExpression(ctx));
}
fillBuffer(" GROUP BY ", buffer, ctx, _groupBys, Visit.EXPRESSION); fillBuffer(" GROUP BY ", buffer, ctx, _groupBys, Visit.EXPRESSION);
fillBuffer(" HAVING ", buffer, ctx, _having);
if (_having != null) {
buffer.append(" HAVING ")
.append(((Visitable)_having).asExpression(ctx));
}
fillBuffer(" ORDER BY ", buffer, ctx, _orderBys, Visit.EXPRESSION); fillBuffer(" ORDER BY ", buffer, ctx, _orderBys, Visit.EXPRESSION);
return buffer.toString(); return buffer.toString();
@ -391,28 +395,50 @@ public class QueryDefinitionImpl extends ExpressionImpl
case EXPRESSION : buffer.append(v.asExpression(ctx)) case EXPRESSION : buffer.append(v.asExpression(ctx))
.append(i != list.size()-1 ? ", " : " "); .append(i != list.size()-1 ? ", " : " ");
break; break;
case JOINABLE : buffer.append(v.asJoinable(ctx)); case JOINABLE : buffer.append(v.asJoinable(ctx))
.append(i > 0 && v instanceof RootPath ?
"," : " ");
break; break;
} }
} }
} }
public void fillBuffer(String header, StringBuffer buffer, AliasContext ctx,
Predicate p) {
if (p == null)
return;
Visitable v = (Visitable)p;
buffer.append(header);
buffer.append(v.asExpression(ctx));
}
/** /**
* Registers each domain with an alias. * Registers each domain with an alias. Also set alias for order by items
* @param ctx * that are projected.
*/ */
private void registerDomains(AliasContext ctx) { private void registerDomains(AliasContext ctx) {
if (_domains != null) { if (_domains != null) {
Collections.sort(_domains, new DomainSorter());
for (AbstractDomainObject domain : _domains) { for (AbstractDomainObject domain : _domains) {
domain.setAlias(ctx); ctx.setAlias(domain);
} }
} }
if (_orderBys != null) { if (_orderBys != null) {
for (OrderableItem o : _orderBys) { for (OrderableItem o : _orderBys) {
ExpressionImpl e = o.getExpression(); ExpressionImpl e = o.getExpression();
if (_projections != null && _projections.contains(e)) if (_projections != null && _projections.contains(e))
e.setAlias(ctx); ctx.setAlias(e);
} }
} }
} }
static class DomainSorter implements Comparator<AbstractDomainObject> {
static List<Class> _order = Arrays.asList(new Class[] {
RootPath.class, NavigationPath.class, OperatorPath.class,
JoinPath.class, FetchPath.class, } );
public int compare(AbstractDomainObject a, AbstractDomainObject b) {
return _order.indexOf(a.getClass()) - _order.indexOf(b.getClass());
}
}
} }

View File

@ -38,4 +38,8 @@ public class RootPath extends AbstractDomainObject implements DomainObject {
public String asProjection(AliasContext ctx) { public String asProjection(AliasContext ctx) {
return ctx.getAlias(this); return ctx.getAlias(this);
} }
public String toString() {
return getLastSegment().getSimpleName();
}
} }

View File

@ -29,24 +29,23 @@ import java.io.Serializable;
*/ */
public interface Visitable extends Serializable { public interface Visitable extends Serializable {
/** /**
* Get a JPQL fragment as used in a WHERE clause. * Get a JPQL fragment as used in WHERE clause.
*/ */
String asExpression(AliasContext ctx); String asExpression(AliasContext ctx);
/** /**
* Gets the string representation in a SELECT projection. * Gets the string representation in SELECT projection.
*/ */
String asProjection(AliasContext ctx); String asProjection(AliasContext ctx);
/** /**
* Sets alias. * Gets the string representation in FROM clause.
*/ */
void setAlias(AliasContext ctx);
String getAliasHint(AliasContext ctx);
String asJoinable(AliasContext ctx); String asJoinable(AliasContext ctx);
/**
* Gets the hint to be used while creating alias.
*/
String getAliasHint(AliasContext ctx);
} }