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

org.opencastproject.util.persistencefn.Queries Maven / Gradle / Ivy

There is a newer version: 16.7
Show newest version
/**
 * Licensed to The Apereo Foundation under one or more contributor license
 * agreements. See the NOTICE file distributed with this work for additional
 * information regarding copyright ownership.
 *
 *
 * The Apereo Foundation licenses this file to you under the Educational
 * Community 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://opensource.org/licenses/ecl2.txt
 *
 * 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.opencastproject.util.persistencefn;

import static com.entwinemedia.fn.Stream.$;

import com.entwinemedia.fn.Fn;
import com.entwinemedia.fn.Monoid;
import com.entwinemedia.fn.P2;
import com.entwinemedia.fn.Unit;
import com.entwinemedia.fn.data.Opt;

import org.joda.time.base.AbstractInstant;

import java.util.Date;
import java.util.List;

import javax.persistence.EntityManager;
import javax.persistence.NoResultException;
import javax.persistence.NonUniqueResultException;
import javax.persistence.Query;
import javax.persistence.TemporalType;
import javax.persistence.TypedQuery;

/**
 * JPA query constructors.
 */
public final class Queries {
  private Queries() {
  }

  /** {@link javax.persistence.EntityManager#find(Class, Object)} as a function wrapping the result into an Option. */
  public static  Fn> find(final Class clazz, final Object primaryKey) {
    return new Fn>() {
      @Override public Opt apply(EntityManager em) {
        return Opt.nul(em.find(clazz, primaryKey));
      }
    };
  }

  /** {@link javax.persistence.EntityManager#persist(Object)} as a function. */
  public static  Fn persist(final A a) {
    return new Fn() {
      @Override public A apply(EntityManager em) {
        em.persist(a);
        return a;
      }
    };
  }

  /** {@link javax.persistence.EntityManager#merge(Object)} as a function. */
  public static  Fn merge(final A a) {
    return new Fn() {
      @Override public A apply(EntityManager em) {
        return em.merge(a);
      }
    };
  }

  /**
   * {@link javax.persistence.EntityManager#contains(Object)} as a function.
   * The function returns true if the entity is a managed instance belonging to the current persistence context.
   */
  public static Fn contains(final Object a) {
    return new Fn() {
      @Override public Boolean apply(EntityManager em) {
        return em.contains(a);
      }
    };
  }

  /**
   * Convenience for EntityManager.getEntityManagerFactory().getPersistenceUnitUtil().getIdentifier(a).
   * The function returns the ID or null if the object does not have an ID yet, i.e. is not yet persisted.
   */
  public static Object getId(EntityManager em , Object a) {
    return em.getEntityManagerFactory().getPersistenceUnitUtil().getIdentifier(a);
  }

  /**
   * {@link #getId(EntityManager, Object)} as a function.
   */
  @SuppressWarnings("unchecked")
  public static  Fn getId(final Object a) {
    return new Fn() {
      @Override public A apply(EntityManager em) {
        return (A) getId(em, a);
      }
    };
  }

  /**
   * Like {@link javax.persistence.EntityManager#remove(Object)} but the entity is allowed
   * to be detached. It will be merged into the current persistence context if necessary.
   */
  public static void remove(EntityManager em, final Object a) {
    if (em.contains(a)) {
      em.remove(a);
    } else {
      em.remove(em.merge(a));
    }
  }

  /** {@link #remove(EntityManager, Object)} as a function. */
  public static Fn remove(final Object a) {
    return new Fn() {
      @Override public Unit apply(EntityManager em) {
        remove(em, a);
        return Unit.unit;
      }
    };
  }

  /**
   * If the object does not have an ID, persist it. If it does, try to update it.
   */
  public static  A persistOrUpdate(final EntityManager em, final A a) {
    final Object id = getId(em, a);
    if (id == null) {
      // object does not have an ID
      em.persist(a);
      return a;
    } else {
      @SuppressWarnings("unchecked")
      final A dto = (A) em.find(a.getClass(), id);
      if (dto == null) {
        em.persist(a);
        return a;
      } else {
      return em.merge(a);
      }
    }
  }

  /**
   * {@link #persistOrUpdate(EntityManager, Object)} as a function.
   */
  public static  Fn persistOrUpdate(final A a) {
    return new Fn() {
      @Override public A apply(EntityManager em) {
        return persistOrUpdate(em, a);
      }
    };
  }

  /**
   * Set a list of named parameters on a query.
   * 

* Values of type {@link java.util.Date} and {@link org.joda.time.base.AbstractInstant} * are recognized and set as a timestamp ({@link javax.persistence.TemporalType#TIMESTAMP}. */ public static A setParams(A q, P2... params) { for (P2 p : params) { final Object value = p.get2(); if (value instanceof Date) { q.setParameter(p.get1(), (Date) value, TemporalType.TIMESTAMP); } if (value instanceof AbstractInstant) { q.setParameter(p.get1(), ((AbstractInstant) value).toDate(), TemporalType.TIMESTAMP); } else { q.setParameter(p.get1(), p.get2()); } } return q; } /** * Set a list of positional parameters on a query. *

* Values of type {@link java.util.Date} and {@link org.joda.time.base.AbstractInstant} * are recognized and set as a timestamp ({@link javax.persistence.TemporalType#TIMESTAMP}. */ public static A setParams(A q, Object... params) { for (int i = 0; i < params.length; i++) { final Object value = params[i]; if (value instanceof Date) { q.setParameter(i + 1, (Date) value, TemporalType.TIMESTAMP); } if (value instanceof AbstractInstant) { q.setParameter(i + 1, ((AbstractInstant) value).toDate(), TemporalType.TIMESTAMP); } else { q.setParameter(i + 1, value); } } return q; } /** * Compose multiple queries qs and sum up their results. * * @param m monoid to sum up results * @param qs list of queries * @return the sum of the results */ @SafeVarargs public static Fn sum(final Monoid m, final Fn... qs) { return new Fn() { @Override public A apply(final EntityManager em) { return $(qs).map(new Fn, A>() { @Override public A apply(Fn q) { return q.apply(em); } }).sum(m); } }; } // ------------------------------------------------------------------------------------------------------------------- /** Named queries with support for named parameters. */ public static final TypedQueriesBase> named = new TypedQueriesBase>() { @Override public Query query(EntityManager em, String q, P2... params) { return setParams(em.createNamedQuery(q), params); } @Override public TypedQuery query(EntityManager em, String q, Class type, P2... params) { return setParams(em.createNamedQuery(q, type), params); } }; /** JPQL queries with support for named parameters. */ public static final TypedQueriesBase> jpql = new TypedQueriesBase>() { @Override public Query query(EntityManager em, String q, P2... params) { return setParams(em.createQuery(q), params); } @Override public TypedQuery query(EntityManager em, String q, Class type, P2... params) { return setParams(em.createQuery(q, type), params); } }; /** Native SQL queries. Only support positional parameters. */ public static final QueriesBase sql = new QueriesBase() { @Override public Query query(EntityManager em, String q, Object... params) { return setParams(em.createNativeQuery(q), params); } }; // ------------------------------------------------------------------------------------------------------------------- public abstract static class TypedQueriesBase

extends QueriesBase

{ protected TypedQueriesBase() { } /** * Create a typed query from q with a list of parameters. * Values of type {@link java.util.Date} are recognized * and set as a timestamp ({@link javax.persistence.TemporalType#TIMESTAMP}. */ public abstract TypedQuery query(EntityManager em, String queryName, Class type, P2... params); /** Find multiple entities. */ public List findAll(EntityManager em, final String q, final Class type, final P2... params) { return query(em, q, type, params).getResultList(); } /** {@link #findAll(javax.persistence.EntityManager, String, Class, com.entwinemedia.fn.P2[])} as a function. */ public Fn> findAll(final String q, final Class type, final P2... params) { return new Fn>() { @Override public List apply(EntityManager em) { return findAll(em, q, type, params); } }; } } // ------------------------------------------------------------------------------------------------------------------- public abstract static class QueriesBase

{ protected QueriesBase() { } /** * Create a query from q with a list of parameters. * Values of type {@link java.util.Date} are recognized * and set as a timestamp ({@link javax.persistence.TemporalType#TIMESTAMP}. */ public abstract Query query(EntityManager em, String q, P... params); /** Run an update (UPDATE or DELETE) query and ensure that at least one row got affected. */ public boolean update(EntityManager em, String q, P... params) { return query(em, q, params).executeUpdate() > 0; } /** {@link #update(javax.persistence.EntityManager, String, Object[])} as a function. */ public Fn update(final String q, final P... params) { return new Fn() { @Override public Boolean apply(EntityManager em) { return update(em, q, params); } }; } /** * Run a SELECT query that should return a single result. * * @return some value if the query yields exactly one result, none otherwise */ public Opt findSingle(final EntityManager em, final String q, final P... params) { try { return Opt.some((A) query(em, q, params).getSingleResult()); } catch (NoResultException e) { return Opt.none(); } catch (NonUniqueResultException e) { return Opt.none(); } } /** {@link #findSingle(javax.persistence.EntityManager, String, Object[])} as a function. */ public Fn> findSingle(final String q, final P... params) { return new Fn>() { @Override public Opt apply(EntityManager em) { return findSingle(em, q, params); } }; } /** Run a SELECT query and return only the first result item. */ public Opt findFirst(final EntityManager em, final String q, final P... params) { try { return Opt.some((A) query(em, q, params).setMaxResults(1).getSingleResult()); } catch (NoResultException e) { return Opt.none(); } catch (NonUniqueResultException e) { return Opt.none(); } } /** {@link #findSingle(javax.persistence.EntityManager, String, Object[])} as a function. */ public Fn> findFirst(final String q, final P... params) { return new Fn>() { @Override public Opt apply(EntityManager em) { return findFirst(em, q, params); } }; } /** Run a COUNT(x) query. */ public long count(final EntityManager em, final String q, final P... params) { return (Long) query(em, q, params).getSingleResult(); } /** {@link #count(javax.persistence.EntityManager, String, Object[])} as a function. */ public Fn count(final String q, final P... params) { return new Fn() { @Override public Long apply(EntityManager em) { return count(em, q, params); } }; } /** Find multiple entities. */ public List findAll(EntityManager em, final String q, final P... params) { return (List) query(em, q, params).getResultList(); } /** {@link #findAll(javax.persistence.EntityManager, String, Object[])} as a function. */ public Fn> findAll(final String q, final P... params) { return new Fn>() { @Override public List apply(EntityManager em) { return findAll(em, q, params); } }; } /** Find multiple objects with optional pagination. */ public List findAll(final EntityManager em, final String q, final Opt offset, final Opt limit, final P... params) { final Query query = query(em, q, params); for (Integer x : offset) query.setFirstResult(x); for (Integer x : limit) query.setMaxResults(x); return (List) query.getResultList(); } public Fn> findAll(final String q, final Opt offset, final Opt limit, final P... params) { return new Fn>() { @Override public List apply(EntityManager em) { return findAll(em, q, offset, limit, params); } }; } /** Find multiple objects with pagination. */ public List findAll(final EntityManager em, final String q, final int offset, final int limit, final P... params) { final Query query = query(em, q, params); query.setFirstResult(offset); query.setMaxResults(limit); return (List) query.getResultList(); } public Fn> findAll(final String q, final int offset, final int limit, final P... params) { return new Fn>() { @Override public List apply(EntityManager em) { return findAll(em, q, offset, limit, params); } }; } } }