/* * #%L * BroadleafCommerce Framework * %% * Copyright (C) 2009 - 2013 Broadleaf Commerce * %% * 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. * #L% */ package org.broadleafcommerce.core.catalog.dao; import org.broadleafcommerce.common.extension.ExtensionResultHolder; import org.broadleafcommerce.common.extension.ExtensionResultStatusType; import org.broadleafcommerce.common.persistence.EntityConfiguration; import org.broadleafcommerce.common.persistence.Status; import org.broadleafcommerce.common.sandbox.SandBoxHelper; import org.broadleafcommerce.common.time.SystemTime; import org.broadleafcommerce.common.util.dao.TypedQueryBuilder; import org.broadleafcommerce.core.catalog.domain.Category; import org.broadleafcommerce.core.catalog.domain.CategoryImpl; import org.broadleafcommerce.core.catalog.domain.Product; import org.hibernate.ejb.QueryHints; import org.springframework.stereotype.Repository; import java.util.Date; import java.util.List; import javax.annotation.Nonnull; import javax.annotation.Resource; import javax.persistence.EntityManager; import javax.persistence.NoResultException; import javax.persistence.PersistenceContext; import javax.persistence.Query; import javax.persistence.TypedQuery; import javax.persistence.criteria.CriteriaBuilder; import javax.persistence.criteria.CriteriaQuery; import javax.persistence.criteria.Root; /** * * @author Jeff Fischer */ @Repository("blCategoryDao") public class CategoryDaoImpl implements CategoryDao { protected Long currentDateResolution = 10000L; protected Date cachedDate = SystemTime.asDate(); protected Date getCurrentDateAfterFactoringInDateResolution() { Date returnDate = SystemTime.getCurrentDateWithinTimeResolution(cachedDate, currentDateResolution); if (returnDate != cachedDate) { if (SystemTime.shouldCacheDate()) { cachedDate = returnDate; } } return returnDate; } @PersistenceContext(unitName="blPU") protected EntityManager em; @Resource(name="blEntityConfiguration") protected EntityConfiguration entityConfiguration; @Resource(name = "blSandBoxHelper") protected SandBoxHelper sandBoxHelper; @Resource(name = "blCategoryDaoExtensionManager") protected CategoryDaoExtensionManager extensionManager; @Override public Category save(Category category) { return em.merge(category); } @Override public Category readCategoryById(Long categoryId) { return em.find(CategoryImpl.class, categoryId); } @Override public Category readCategoryByExternalId(@Nonnull String externalId) { TypedQuery<Category> query = new TypedQueryBuilder<Category>(Category.class, "cat") .addRestriction("cat.externalId", "=", externalId) .toQuery(em); try { return query.getSingleResult(); } catch (NoResultException e) { return null; } } @Override @Deprecated public Category readCategoryByName(String categoryName) { Query query = em.createNamedQuery("BC_READ_CATEGORY_BY_NAME"); query.setParameter("categoryName", categoryName); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); return (Category) query.getSingleResult(); } @Override public List<Category> readAllParentCategories() { CriteriaBuilder builder = em.getCriteriaBuilder(); CriteriaQuery<Category> criteria = builder.createQuery(Category.class); Root<CategoryImpl> category = criteria.from(CategoryImpl.class); criteria.select(category); criteria.where(builder.isNull(category.get("defaultParentCategory"))); TypedQuery<Category> query = em.createQuery(criteria); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); try { return query.getResultList(); } catch (NoResultException e) { return null; } } @Override public List<Category> readCategoriesByName(String categoryName) { TypedQuery<Category> query = em.createNamedQuery("BC_READ_CATEGORY_BY_NAME", Category.class); query.setParameter("categoryName", categoryName); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); return query.getResultList(); } @Override public List<Category> readCategoriesByName(String categoryName, int limit, int offset) { TypedQuery<Category> query = em.createNamedQuery("BC_READ_CATEGORY_BY_NAME", Category.class); query.setParameter("categoryName", categoryName); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); query.setFirstResult(offset); query.setMaxResults(limit); return query.getResultList(); } @Override public List<Category> readAllCategories() { TypedQuery<Category> query = em.createNamedQuery("BC_READ_ALL_CATEGORIES", Category.class); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); return query.getResultList(); } @Override public List<Category> readAllCategories(int limit, int offset) { TypedQuery<Category> query = em.createNamedQuery("BC_READ_ALL_CATEGORIES", Category.class); query.setFirstResult(offset); query.setMaxResults(limit); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); return query.getResultList(); } @Override public List<Product> readAllProducts() { TypedQuery<Product> query = em.createNamedQuery("BC_READ_ALL_PRODUCTS", Product.class); //don't cache - could take up too much memory return query.getResultList(); } @Override public List<Product> readAllProducts(int limit, int offset) { TypedQuery<Product> query = em.createNamedQuery("BC_READ_ALL_PRODUCTS", Product.class); //don't cache - could take up too much memory query.setFirstResult(offset); query.setMaxResults(limit); return query.getResultList(); } @Override public List<Category> readAllSubCategories(Category category) { TypedQuery<Category> query = em.createNamedQuery("BC_READ_ALL_SUBCATEGORIES", Category.class); query.setParameter("parentCategoryId", sandBoxHelper.mergeCloneIds(CategoryImpl.class, category.getId())); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); return query.getResultList(); } @Override public List<Category> readAllSubCategories(Category category, int limit, int offset) { TypedQuery<Category> query = em.createNamedQuery("BC_READ_ALL_SUBCATEGORIES", Category.class); query.setParameter("parentCategoryId", sandBoxHelper.mergeCloneIds(CategoryImpl.class, category.getId())); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); query.setFirstResult(offset); query.setMaxResults(limit); return query.getResultList(); } @Override public List<Category> readActiveSubCategoriesByCategory(Category category) { TypedQuery<Category> query = em.createNamedQuery("BC_READ_ACTIVE_SUBCATEGORIES_BY_CATEGORY", Category.class); query.setParameter("parentCategoryId", sandBoxHelper.mergeCloneIds(CategoryImpl.class, category.getId())); query.setParameter("currentDate", getCurrentDateAfterFactoringInDateResolution()); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); return query.getResultList(); } @Override public List<Category> readActiveSubCategoriesByCategory(Category category, int limit, int offset) { TypedQuery<Category> query = em.createNamedQuery("BC_READ_ACTIVE_SUBCATEGORIES_BY_CATEGORY", Category.class); query.setParameter("parentCategoryId", sandBoxHelper.mergeCloneIds(CategoryImpl.class, category.getId())); query.setParameter("currentDate", getCurrentDateAfterFactoringInDateResolution()); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); query.setFirstResult(offset); query.setMaxResults(limit); return query.getResultList(); } @Override public Long getCurrentDateResolution() { return currentDateResolution; } @Override public void setCurrentDateResolution(Long currentDateResolution) { this.currentDateResolution = currentDateResolution; } @Override public void delete(Category category) { ((Status) category).setArchived('Y'); em.merge(category); } @Override public Category create() { return (Category) entityConfiguration.createEntityInstance(Category.class.getName()); } @Override public Category findCategoryByURI(String uri) { if (extensionManager != null) { ExtensionResultHolder holder = new ExtensionResultHolder(); ExtensionResultStatusType result = extensionManager.getProxy().findCategoryByURI(uri, holder); if (ExtensionResultStatusType.HANDLED.equals(result)) { return (Category) holder.getResult(); } } Query query; query = em.createNamedQuery("BC_READ_CATEGORY_OUTGOING_URL"); query.setParameter("currentDate", getCurrentDateAfterFactoringInDateResolution()); query.setParameter("url", uri); query.setHint(QueryHints.HINT_CACHEABLE, true); query.setHint(QueryHints.HINT_CACHE_REGION, "query.Catalog"); @SuppressWarnings("unchecked") List<Category> results = query.getResultList(); if (results != null && !results.isEmpty()) { return results.get(0); } else { return null; } } }