/** * Licensed to Apereo under one or more contributor license agreements. See the NOTICE file * distributed with this work for additional information regarding copyright ownership. Apereo * licenses this file to you 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 the * following location: * * <p>http://www.apache.org/licenses/LICENSE-2.0 * * <p>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.apereo.portal.portlet.dao.jpa; import com.google.common.base.Function; import java.util.HashSet; import java.util.List; import java.util.Set; import javax.persistence.EntityManager; import javax.persistence.TypedQuery; import javax.persistence.criteria.CriteriaBuilder; import javax.persistence.criteria.CriteriaQuery; import javax.persistence.criteria.JoinType; import javax.persistence.criteria.ParameterExpression; import javax.persistence.criteria.Root; import org.apache.commons.lang.StringUtils; import org.apache.commons.lang.Validate; import org.apereo.portal.jpa.BasePortalJpaDao; import org.apereo.portal.jpa.OpenEntityManager; import org.apereo.portal.portlet.dao.IPortletDefinitionDao; import org.apereo.portal.portlet.dao.IPortletEntityDao; import org.apereo.portal.portlet.om.IPortletDefinition; import org.apereo.portal.portlet.om.IPortletDefinitionId; import org.apereo.portal.portlet.om.IPortletEntity; import org.apereo.portal.portlet.om.IPortletEntityId; import org.apereo.portal.spring.tx.DialectAwareTransactional; import org.hibernate.dialect.PostgreSQL81Dialect; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.dao.DataRetrievalFailureException; import org.springframework.stereotype.Repository; @Repository @Qualifier("persistence") public class JpaPortletEntityDao extends BasePortalJpaDao implements IPortletEntityDao { private CriteriaQuery<PortletEntityImpl> findEntitiesForDefinitionQuery; private CriteriaQuery<PortletEntityImpl> findEntitiesForUserIdQuery; private ParameterExpression<Integer> userIdParameter; private ParameterExpression<PortletDefinitionImpl> portletDefinitionParameter; private IPortletDefinitionDao portletDefinitionDao; @Autowired public void setPortletDefinitionDao(IPortletDefinitionDao portletDefinitionDao) { this.portletDefinitionDao = portletDefinitionDao; } @Override public void afterPropertiesSet() throws Exception { this.userIdParameter = this.createParameterExpression(Integer.class, "userId"); this.portletDefinitionParameter = this.createParameterExpression(PortletDefinitionImpl.class, "portletDefinition"); this.findEntitiesForDefinitionQuery = this.createCriteriaQuery( new Function<CriteriaBuilder, CriteriaQuery<PortletEntityImpl>>() { @Override public CriteriaQuery<PortletEntityImpl> apply(CriteriaBuilder cb) { final CriteriaQuery<PortletEntityImpl> criteriaQuery = cb.createQuery(PortletEntityImpl.class); final Root<PortletEntityImpl> entityRoot = criteriaQuery.from(PortletEntityImpl.class); criteriaQuery.select(entityRoot); addFetches(entityRoot); criteriaQuery.where( cb.equal( entityRoot.get( PortletEntityImpl_.portletDefinition), portletDefinitionParameter)); return criteriaQuery; } }); this.findEntitiesForUserIdQuery = this.createCriteriaQuery( new Function<CriteriaBuilder, CriteriaQuery<PortletEntityImpl>>() { @Override public CriteriaQuery<PortletEntityImpl> apply(CriteriaBuilder cb) { final CriteriaQuery<PortletEntityImpl> criteriaQuery = cb.createQuery(PortletEntityImpl.class); final Root<PortletEntityImpl> entityRoot = criteriaQuery.from(PortletEntityImpl.class); criteriaQuery.select(entityRoot); addFetches(entityRoot); criteriaQuery.where( cb.equal( entityRoot.get(PortletEntityImpl_.userId), userIdParameter)); return criteriaQuery; } }); } /** Add all the fetches needed for completely loading the object graph */ protected void addFetches(final Root<PortletEntityImpl> definitionRoot) { definitionRoot .fetch(PortletEntityImpl_.portletPreferences, JoinType.LEFT) .fetch(PortletPreferencesImpl_.portletPreferences, JoinType.LEFT) .fetch(PortletPreferenceImpl_.values, JoinType.LEFT); definitionRoot.fetch(PortletEntityImpl_.windowStates, JoinType.LEFT); } @Override @PortalTransactional public IPortletEntity createPortletEntity( IPortletDefinitionId portletDefinitionId, String layoutNodeId, int userId) { Validate.notNull(portletDefinitionId, "portletDefinitionId can not be null"); Validate.notEmpty(layoutNodeId, "layoutNodeId can not be null"); final IPortletDefinition portletDefinition = this.portletDefinitionDao.getPortletDefinition(portletDefinitionId); if (portletDefinition == null) { throw new DataRetrievalFailureException( "No IPortletDefinition exists for IPortletDefinitionId='" + portletDefinitionId + "'"); } IPortletEntity portletEntity = new PortletEntityImpl(portletDefinition, layoutNodeId, userId); this.getEntityManager().persist(portletEntity); return portletEntity; } @Override @PortalTransactional public void deletePortletEntity(IPortletEntity portletEntity) { Validate.notNull(portletEntity, "portletEntity can not be null"); final IPortletEntity persistentPortletEntity; final EntityManager entityManager = this.getEntityManager(); if (entityManager.contains(portletEntity)) { persistentPortletEntity = portletEntity; } else { persistentPortletEntity = entityManager.merge(portletEntity); } entityManager.remove(persistentPortletEntity); } @Override @DialectAwareTransactional(value = PostgreSQL81Dialect.class, exclude = false) @PortalTransactionalReadOnly @OpenEntityManager(unitName = PERSISTENCE_UNIT_NAME) public IPortletEntity getPortletEntity(IPortletEntityId portletEntityId) { Validate.notNull(portletEntityId, "portletEntityId can not be null"); final long internalPortletEntityId = getNativePortletEntityId(portletEntityId); final PortletEntityImpl portletEntity = this.getEntityManager().find(PortletEntityImpl.class, internalPortletEntityId); return portletEntity; } @Override @PortalTransactionalRequiresNew public boolean portletEntityExists(IPortletEntityId portletEntityId) { Validate.notNull(portletEntityId, "portletEntityId can not be null"); final EntityManager entityManager = this.getEntityManager(); entityManager.clear(); final long internalPortletEntityId = getNativePortletEntityId(portletEntityId); final PortletEntityImpl portletEntity = entityManager.find(PortletEntityImpl.class, internalPortletEntityId); return portletEntity != null; } @Override @DialectAwareTransactional(value = PostgreSQL81Dialect.class, exclude = false) @PortalTransactionalReadOnly @OpenEntityManager(unitName = PERSISTENCE_UNIT_NAME) public IPortletEntity getPortletEntity(String layoutNodeId, int userId) { Validate.notNull(layoutNodeId, "portletEntity can not be null"); /* Since portal entities mostly are retrieved in batches (for each "channel" element in user's layout), it is * faster to retrieve all portlet entities, so that persistence framework can place them in 2nd level cache, and * iterate over them manually instead of retrieving single portlet entity one by one. */ Set<IPortletEntity> entities = getPortletEntitiesForUser(userId); for (IPortletEntity entity : entities) { if (StringUtils.equals(entity.getLayoutNodeId(), layoutNodeId)) { return entity; } } return null; } @Override @DialectAwareTransactional(value = PostgreSQL81Dialect.class, exclude = false) @PortalTransactionalReadOnly @OpenEntityManager(unitName = PERSISTENCE_UNIT_NAME) public Set<IPortletEntity> getPortletEntities(IPortletDefinitionId portletDefinitionId) { Validate.notNull(portletDefinitionId, "portletEntity can not be null"); final IPortletDefinition portletDefinition = this.portletDefinitionDao.getPortletDefinition(portletDefinitionId); final TypedQuery<PortletEntityImpl> query = this.createCachedQuery(this.findEntitiesForDefinitionQuery); query.setParameter( this.portletDefinitionParameter, (PortletDefinitionImpl) portletDefinition); final List<PortletEntityImpl> portletEntities = query.getResultList(); return new HashSet<IPortletEntity>(portletEntities); } @Override @DialectAwareTransactional(value = PostgreSQL81Dialect.class, exclude = false) @PortalTransactionalReadOnly @OpenEntityManager(unitName = PERSISTENCE_UNIT_NAME) public Set<IPortletEntity> getPortletEntitiesForUser(int userId) { final TypedQuery<PortletEntityImpl> query = this.createCachedQuery(this.findEntitiesForUserIdQuery); query.setParameter(this.userIdParameter, userId); final List<PortletEntityImpl> portletEntities = query.getResultList(); return new HashSet<IPortletEntity>(portletEntities); } @Override @PortalTransactional public void updatePortletEntity(IPortletEntity portletEntity) { Validate.notNull(portletEntity, "portletEntity can not be null"); this.getEntityManager().persist(portletEntity); } protected long getNativePortletEntityId(IPortletEntityId portletEntityId) { final long internalPortletEntityId = Long.parseLong(portletEntityId.getStringId()); return internalPortletEntityId; } }