com.ejushang.steward.common.genericdao.search.hibernate.HibernateSearchProcessor.java Source code

Java tutorial

Introduction

Here is the source code for com.ejushang.steward.common.genericdao.search.hibernate.HibernateSearchProcessor.java

Source

/* Copyright 2013 David Wolverton
 * 
 * Licensed 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 com.ejushang.steward.common.genericdao.search.hibernate;

import com.ejushang.steward.common.genericdao.search.*;
import org.hibernate.NonUniqueResultException;
import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.transform.ResultTransformer;
import org.hibernate.transform.Transformers;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.*;

/**
 * Implementation of BaseSearchProcessor that generates Works with standard Hibernate.
 * 
 * A singleton instance of this class is maintained for each SessionFactory. This should
 * be accessed using {@link HibernateSearchProcessor#getInstanceForSessionFactory(org.hibernate.SessionFactory)}.
 * 
 * @author dwolverton
 */
public class HibernateSearchProcessor extends BaseSearchProcessor {
    private static Logger logger = LoggerFactory.getLogger(HibernateSearchProcessor.class);

    private static Map<SessionFactory, HibernateSearchProcessor> map = new HashMap<SessionFactory, HibernateSearchProcessor>();

    public static HibernateSearchProcessor getInstanceForSessionFactory(SessionFactory sessionFactory) {
        HibernateSearchProcessor instance = map.get(sessionFactory);
        if (instance == null) {
            instance = new HibernateSearchProcessor(
                    HibernateMetadataUtil.getInstanceForSessionFactory(sessionFactory));
            map.put(sessionFactory, instance);
        }
        return instance;
    }

    private HibernateSearchProcessor(HibernateMetadataUtil mdu) {
        super(QLTYPE_HQL, mdu);
    }

    // --- Public Methods ---

    /**
     * Search for objects based on the search parameters in the specified
     * <code>ISearch</code> object.
     * 
     * @see ISearch
     */
    @SuppressWarnings("unchecked")
    public List search(Session session, ISearch search) {
        if (search == null)
            return null;

        return search(session, search.getSearchClass(), search);
    }

    /**
     * Search for objects based on the search parameters in the specified
     * <code>ISearch</code> object. Uses the specified searchClass, ignoring the
     * searchClass specified on the search itself.
     * 
     * @see ISearch
     */
    @SuppressWarnings("unchecked")
    public List search(Session session, Class<?> searchClass, ISearch search) {
        if (searchClass == null || search == null)
            return null;

        List<Object> paramList = new ArrayList<Object>();
        String hql = generateQL(searchClass, search, paramList);
        Query query = session.createQuery(hql);
        setQueryCache(query, search);
        addParams(query, paramList);
        addPaging(query, search);
        addResultMode(query, search);

        return query.list();
    }

    /**
     * ??
     * @param query
     * @param search
     */
    private void setQueryCache(Query query, ISearch search) {
        if (search.isCacheable()) {
            query.setCacheable(true);
        }
        if (search.getCacheMode() != null) {
            query.setCacheMode(search.getCacheMode());
        }

    }

    /**
    * Returns the total number of results that would be returned using the
    * given <code>ISearch</code> if there were no paging or maxResult limits.
    * 
    * @see ISearch
    */
    public int count(Session session, ISearch search) {
        if (search == null)
            return 0;
        return count(session, search.getSearchClass(), search);
    }

    /**
     * Returns the total number of results that would be returned using the
     * given <code>ISearch</code> if there were no paging or maxResult limits.
     * Uses the specified searchClass, ignoring the searchClass specified on the
     * search itself.
     * 
     * @see ISearch
     */
    public int count(Session session, Class<?> searchClass, ISearch search) {
        if (searchClass == null || search == null)
            return 0;

        List<Object> paramList = new ArrayList<Object>();
        String hql = generateRowCountQL(searchClass, search, paramList);
        if (hql == null) { // special case where the query uses column operators
            return 1;
        }
        Query query = session.createQuery(hql);
        setQueryCache(query, search);
        addParams(query, paramList);

        return ((Number) query.uniqueResult()).intValue();
    }

    /**
     * Returns a <code>SearchResult</code> object that includes the list of
     * results like <code>search()</code> and the total length like
     * <code>searchLength</code>.
     * 
     * @see ISearch
     */
    @SuppressWarnings("unchecked")
    public SearchResult searchAndCount(Session session, ISearch search) {
        if (search == null)
            return null;
        return searchAndCount(session, search.getSearchClass(), search);
    }

    /**
     * Returns a <code>SearchResult</code> object that includes the list of
     * results like <code>search()</code> and the total length like
     * <code>searchLength</code>. Uses the specified searchClass, ignoring the
     * searchClass specified on the search itself.
     * 
     * @see ISearch
     */
    @SuppressWarnings("unchecked")
    public SearchResult searchAndCount(Session session, Class<?> searchClass, ISearch search) {
        if (searchClass == null || search == null)
            return null;

        SearchResult result = new SearchResult();
        result.setResult(search(session, searchClass, search));

        if (search.getMaxResults() > 0) {
            result.setTotalCount(count(session, searchClass, search));
        } else {
            result.setTotalCount(result.getResult().size() + SearchUtil.calcFirstResult(search));
        }

        return result;
    }

    /**
     * Search for a single result using the given parameters.
     */
    public Object searchUnique(Session session, ISearch search) throws NonUniqueResultException {
        if (search == null)
            return null;
        return searchUnique(session, search.getSearchClass(), search);
    }

    /**
     * Search for a single result using the given parameters. Uses the specified
     * searchClass, ignoring the searchClass specified on the search itself.
     */
    public Object searchUnique(Session session, Class<?> entityClass, ISearch search)
            throws NonUniqueResultException {
        if (search == null)
            return null;

        List<Object> paramList = new ArrayList<Object>();
        String hql = generateQL(entityClass, search, paramList);
        Query query = session.createQuery(hql);
        addParams(query, paramList);
        addPaging(query, search);
        addResultMode(query, search);

        return query.uniqueResult();
    }

    // ---- SEARCH HELPERS ---- //

    @SuppressWarnings("unchecked")
    private void addParams(Query query, List<Object> params) {
        StringBuilder debug = null;

        int i = 1;
        for (Object o : params) {
            if (logger.isDebugEnabled()) {
                if (debug == null)
                    debug = new StringBuilder();
                else
                    debug.append("\n\t");
                debug.append("p");
                debug.append(i);
                debug.append(": ");
                debug.append(InternalUtil.paramDisplayString(o));
            }
            if (o instanceof Collection) {
                query.setParameterList("p" + Integer.toString(i++), (Collection) o);
            } else if (o instanceof Object[]) {
                query.setParameterList("p" + Integer.toString(i++), (Object[]) o);
            } else {
                query.setParameter("p" + Integer.toString(i++), o);
            }
        }
        if (debug != null && debug.length() != 0) {
            logger.debug(debug.toString());
        }
    }

    private void addPaging(Query query, ISearch search) {
        int firstResult = SearchUtil.calcFirstResult(search);
        if (firstResult > 0) {
            query.setFirstResult(firstResult);
        }
        if (search.getMaxResults() > 0) {
            query.setMaxResults(search.getMaxResults());
        }
    }

    private void addResultMode(Query query, ISearch search) {
        int resultMode = search.getResultMode();
        if (resultMode == ISearch.RESULT_AUTO) {
            int count = 0;
            Iterator<Field> fieldItr = search.getFields().iterator();
            while (fieldItr.hasNext()) {
                Field field = fieldItr.next();
                if (field.getKey() != null && !field.getKey().equals("")) {
                    resultMode = ISearch.RESULT_MAP;
                    break;
                }
                count++;
            }
            if (resultMode == ISearch.RESULT_AUTO) {
                if (count > 1)
                    resultMode = ISearch.RESULT_ARRAY;
                else
                    resultMode = ISearch.RESULT_SINGLE;
            }
        }

        switch (resultMode) {
        case ISearch.RESULT_ARRAY:
            query.setResultTransformer(ARRAY_RESULT_TRANSFORMER);
            break;
        case ISearch.RESULT_LIST:
            query.setResultTransformer(Transformers.TO_LIST);
            break;
        case ISearch.RESULT_MAP:
            List<String> keyList = new ArrayList<String>();
            Iterator<Field> fieldItr = search.getFields().iterator();
            while (fieldItr.hasNext()) {
                Field field = fieldItr.next();
                if (field.getKey() != null && !field.getKey().equals("")) {
                    keyList.add(field.getKey());
                } else {
                    keyList.add(field.getProperty());
                }
            }
            query.setResultTransformer(new MapResultTransformer(keyList.toArray(new String[0])));
            break;
        default: // ISearch.RESULT_SINGLE
            break;
        }
    }

    @SuppressWarnings("unchecked")
    private static final ResultTransformer ARRAY_RESULT_TRANSFORMER = new ResultTransformer() {
        private static final long serialVersionUID = 1L;

        public List transformList(List collection) {
            return collection;
        }

        public Object transformTuple(Object[] tuple, String[] aliases) {
            return tuple;
        }
    };

    @SuppressWarnings("unchecked")
    private static class MapResultTransformer implements ResultTransformer {
        private static final long serialVersionUID = 1L;

        private String[] keys;

        public MapResultTransformer(String[] keys) {
            this.keys = keys;
        }

        public List transformList(List collection) {
            return collection;
        }

        public Object transformTuple(Object[] tuple, String[] aliases) {
            Map<String, Object> map = new HashMap<String, Object>();
            for (int i = 0; i < keys.length; i++) {
                String key = keys[i];
                if (key != null) {
                    map.put(key, tuple[i]);
                }
            }

            return map;
        }
    }

}