user829237
user829237

Reputation: 1759

JPA Criteria Disjunction

Im trying to learn and understand JPA criteria. So far i am very capable in SQL, and fairly capable in Hibernate Criteria and HQL.

I am trying to do a fairly simple select with an OR statement.

In plain SQL my select looks like this:

SELECT * FROM CHANGED_LOG
WHERE key1 = 52540 AND objectCode = 'Order'
OR key1 = 48398 AND objectCode = 'Package'

This gives me every row where key1 = 52540 and objectCode equals Order AND also every row where key = 48398 and objectCode equals Package. This is exactly what i want.

So trying to do this with JPA criteria (which seems exceptionally complicated...), my best guess so far has been this:

    CriteriaBuilder builder = entityManager.getCriteriaBuilder();

    CriteriaQuery<ChangedLogBean> query = builder.createQuery(ChangedLogBean.class);
    Root<ChangedLogBean> from = query.from(ChangedLogBean.class);
    CriteriaQuery<ChangedLogBean> select = query.select(from);

    Predicate orderChangedLogBeans = builder.conjunction();
    builder.and(orderChangedLogBeans, builder.equal(from.get("key1"), orderId));
    builder.and(orderChangedLogBeans, builder.equal(from.get("objectCode"), ChangedLogBean.ObjectType.OrderBean));

    Predicate packageChangedLogBeans = builder.conjunction();
    builder.and(packageChangedLogBeans, builder.equal(from.get("key1"), packageId));
    builder.and(packageChangedLogBeans, builder.equal(from.get("objectCode"), ChangedLogBean.ObjectType.PackageBean));

    Predicate orderOrPackage = builder.disjunction();
    orderOrPackage.getExpressions().add(orderChangedLogBeans);
    orderOrPackage.getExpressions().add(packageChangedLogBeans);

    query.where(orderOrPackage);

    return entityManager.createQuery(select).getResultList();

Wow. A lot of lines for a simple query... But still, this returnes to me EVERY single row from the DB.

What am i doing wrong here?

Thanks for all your helpful answers :)

Upvotes: 4

Views: 8657

Answers (1)

perissf
perissf

Reputation: 16273

First thing to do in such cases is configuring the JPA provider in order to log the generated queries. See this q/a for an example.

To answer your question, I am unsure about the use of getExpressions(), so I'd suggest the following rude approach:

Predicate p1 = builder.equal(...);
Predicate p2 = builder.equal(...);
Predicate p3 = builder.equal(...);
Predicate p4 = builder.equal(...);

Predicate p5 = builder.and(p1, p2);
Predicate p6 = builder.and(p3, p4);

query.where(builder.or(p5, p6));

Your error is that such a line:

builder.and(orderChangedLogBeans, builder.equal(from.get("key1"), orderId));

will NOT modify in place orderChangedLogBeans. You have to get the Predicate returned by CriteriaBuilder#and() instead:

orderChangedLogBeans = builder.and(orderChangedLogBeans, builder.equal(from.get("key1"), orderId));

Upvotes: 6

Related Questions