blob: c3879e879cf7ea828225268d8db7443aaef47c37 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF 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
*
* 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 org.apache.syncope.core.persistence.dao.impl;
import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import javax.persistence.NoResultException;
import javax.persistence.TypedQuery;
import org.apache.syncope.common.services.InvalidSearchConditionException;
import org.apache.syncope.common.types.AttributableType;
import org.apache.syncope.common.types.PolicyType;
import org.apache.syncope.core.persistence.beans.AbstractAttributable;
import org.apache.syncope.core.persistence.beans.AbstractVirAttr;
import org.apache.syncope.core.persistence.beans.Entitlement;
import org.apache.syncope.core.persistence.beans.ExternalResource;
import org.apache.syncope.core.persistence.beans.Policy;
import org.apache.syncope.core.persistence.beans.membership.Membership;
import org.apache.syncope.core.persistence.beans.role.RAttrValue;
import org.apache.syncope.core.persistence.beans.role.SyncopeRole;
import org.apache.syncope.core.persistence.beans.user.SyncopeUser;
import org.apache.syncope.core.persistence.dao.EntitlementDAO;
import org.apache.syncope.core.persistence.dao.RoleDAO;
import org.apache.syncope.core.persistence.dao.UserDAO;
import org.apache.syncope.core.util.AttributableUtil;
import org.apache.syncope.core.util.EntitlementUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Repository;
@Repository
public class RoleDAOImpl extends AbstractAttributableDAOImpl implements RoleDAO {
@Autowired
private UserDAO userDAO;
@Autowired
private EntitlementDAO entitlementDAO;
@SuppressWarnings("unchecked")
@Override
protected <T extends AbstractAttributable> T findInternal(final Long id) {
return (T) find(id);
}
@Override
public SyncopeRole find(final Long id) {
TypedQuery<SyncopeRole> query = entityManager.createQuery("SELECT e FROM SyncopeRole e WHERE e.id = :id",
SyncopeRole.class);
query.setParameter("id", id);
SyncopeRole result = null;
try {
result = query.getSingleResult();
} catch (NoResultException e) {
LOG.debug("No role found with id {}", id, e);
}
return result;
}
@Override
public List<SyncopeRole> find(final String name) {
TypedQuery<SyncopeRole> query =
entityManager.createQuery("SELECT e FROM SyncopeRole e WHERE e.name = :name", SyncopeRole.class);
query.setParameter("name", name);
return query.getResultList();
}
@Override
public SyncopeRole find(final String name, final Long parentId) {
TypedQuery<SyncopeRole> query;
if (parentId == null) {
query = entityManager.createQuery("SELECT r FROM SyncopeRole r WHERE "
+ "r.name=:name AND r.parent IS NULL", SyncopeRole.class);
} else {
query = entityManager.createQuery("SELECT r FROM SyncopeRole r WHERE "
+ "r.name=:name AND r.parent.id=:parentId", SyncopeRole.class);
query.setParameter("parentId", parentId);
}
query.setParameter("name", name);
List<SyncopeRole> result = query.getResultList();
return result.isEmpty()
? null
: result.get(0);
}
private void findSameOwnerDescendants(final List<SyncopeRole> result, final SyncopeRole role) {
List<SyncopeRole> children = findChildren(role);
if (children != null) {
for (SyncopeRole child : children) {
if ((child.getUserOwner() == null && child.getRoleOwner() == null && child.isInheritOwner())
|| (child.getUserOwner() != null && child.getUserOwner().equals(role.getUserOwner()))
|| (child.getRoleOwner() != null && child.getRoleOwner().equals(role.getRoleOwner()))) {
findDescendants(result, child);
}
}
}
result.add(role);
}
@Override
public List<SyncopeRole> findOwned(final SyncopeUser owner) {
StringBuilder queryString = new StringBuilder("SELECT e FROM ").append(SyncopeRole.class.getSimpleName()).
append(" e WHERE e.userOwner=:owner ");
for (Long roleId : owner.getRoleIds()) {
queryString.append("OR e.roleOwner.id=").append(roleId).append(' ');
}
TypedQuery<SyncopeRole> query = entityManager.createQuery(queryString.toString(), SyncopeRole.class);
query.setParameter("owner", owner);
List<SyncopeRole> result = new ArrayList<SyncopeRole>();
for (SyncopeRole role : query.getResultList()) {
findSameOwnerDescendants(result, role);
}
return result;
}
@Override
public List<SyncopeRole> findByEntitlement(final Entitlement entitlement) {
TypedQuery<SyncopeRole> query =
entityManager.createQuery("SELECT e FROM " + SyncopeRole.class.getSimpleName() + " e "
+ "WHERE :entitlement MEMBER OF e.entitlements", SyncopeRole.class);
query.setParameter("entitlement", entitlement);
return query.getResultList();
}
private Map.Entry<String, String> getPolicyFields(final PolicyType type) {
String policyField;
String inheritPolicyField;
if (type == PolicyType.GLOBAL_ACCOUNT || type == PolicyType.ACCOUNT) {
policyField = "accountPolicy";
inheritPolicyField = "inheritAccountPolicy";
} else {
policyField = "passwordPolicy";
inheritPolicyField = "inheritPasswordPolicy";
}
return new AbstractMap.SimpleEntry<String, String>(policyField, inheritPolicyField);
}
private List<SyncopeRole> findSamePolicyChildren(final SyncopeRole role, final PolicyType type) {
List<SyncopeRole> result = new ArrayList<SyncopeRole>();
for (SyncopeRole child : findChildren(role)) {
boolean inherit = type == PolicyType.GLOBAL_ACCOUNT || type == PolicyType.ACCOUNT
? child.isInheritAccountPolicy()
: child.isInheritPasswordPolicy();
if (inherit) {
result.add(child);
result.addAll(findSamePolicyChildren(child, type));
}
}
return result;
}
@Override
public List<SyncopeRole> findByPolicy(final Policy policy) {
if (policy.getType() == PolicyType.GLOBAL_SYNC || policy.getType() == PolicyType.SYNC) {
return Collections.<SyncopeRole>emptyList();
}
Map.Entry<String, String> policyFields = getPolicyFields(policy.getType());
StringBuilder queryString = new StringBuilder("SELECT e FROM ").
append(SyncopeRole.class.getSimpleName()).append(" e WHERE e.").
append(policyFields.getKey()).append(" = :policy AND (e.").
append(policyFields.getValue()).append(" IS NULL OR e.").
append(policyFields.getValue()).append(" = 0)");
TypedQuery<SyncopeRole> query = entityManager.createQuery(queryString.toString(), SyncopeRole.class);
query.setParameter("policy", policy);
List<SyncopeRole> result = new ArrayList<SyncopeRole>();
for (SyncopeRole role : query.getResultList()) {
result.add(role);
result.addAll(findSamePolicyChildren(role, policy.getType()));
}
return result;
}
@Override
public List<SyncopeRole> findWithoutPolicy(final PolicyType type) {
if (type == PolicyType.GLOBAL_SYNC || type == PolicyType.SYNC) {
return Collections.<SyncopeRole>emptyList();
}
Map.Entry<String, String> policyFields = getPolicyFields(type);
StringBuilder queryString = new StringBuilder("SELECT e FROM ").
append(SyncopeRole.class.getSimpleName()).append(" e WHERE e.").
append(policyFields.getKey()).append(" IS NULL AND (e.").
append(policyFields.getValue()).append(" IS NULL OR e.").
append(policyFields.getValue()).append(" = 0)");
TypedQuery<SyncopeRole> query = entityManager.createQuery(queryString.toString(), SyncopeRole.class);
return query.getResultList();
}
private void findAncestors(final List<SyncopeRole> result, final SyncopeRole role) {
if (role.getParent() != null && !result.contains(role.getParent())) {
result.add(role.getParent());
findAncestors(result, role.getParent());
}
}
@Override
public List<SyncopeRole> findAncestors(final SyncopeRole role) {
List<SyncopeRole> result = new ArrayList<SyncopeRole>();
findAncestors(result, role);
return result;
}
@Override
public List<SyncopeRole> findChildren(final SyncopeRole role) {
TypedQuery<SyncopeRole> query =
entityManager.createQuery("SELECT r FROM SyncopeRole r WHERE r.parent=:role", SyncopeRole.class);
query.setParameter("role", role);
return query.getResultList();
}
private void findDescendants(final List<SyncopeRole> result, final SyncopeRole role) {
List<SyncopeRole> children = findChildren(role);
if (children != null) {
for (SyncopeRole child : children) {
findDescendants(result, child);
}
}
result.add(role);
}
@Override
public List<SyncopeRole> findDescendants(final SyncopeRole role) {
List<SyncopeRole> result = new ArrayList<SyncopeRole>();
findDescendants(result, role);
return result;
}
@Override
public List<SyncopeRole> findByDerAttrValue(final String schemaName, final String value)
throws InvalidSearchConditionException {
return findByDerAttrValue(schemaName, value, AttributableUtil.getInstance(AttributableType.ROLE));
}
@Override
public List<SyncopeRole> findByAttrValue(final String schemaName, final RAttrValue attrValue) {
return findByAttrValue(schemaName, attrValue, AttributableUtil.getInstance(AttributableType.ROLE));
}
@Override
public SyncopeRole findByAttrUniqueValue(final String schemaName, final RAttrValue attrUniqueValue) {
return (SyncopeRole) findByAttrUniqueValue(schemaName, attrUniqueValue,
AttributableUtil.getInstance(AttributableType.ROLE));
}
@Override
public List<SyncopeRole> findByResource(final ExternalResource resource) {
return findByResource(resource, SyncopeRole.class);
}
@Override
public List<SyncopeRole> findAll() {
TypedQuery<SyncopeRole> query = entityManager.createQuery("SELECT e FROM SyncopeRole e", SyncopeRole.class);
return query.getResultList();
}
@Override
public List<Membership> findMemberships(final SyncopeRole role) {
TypedQuery<Membership> query =
entityManager.createQuery("SELECT e FROM " + Membership.class.getSimpleName() + " e"
+ " WHERE e.syncopeRole=:role", Membership.class);
query.setParameter("role", role);
return query.getResultList();
}
@Override
public SyncopeRole save(final SyncopeRole role) {
// reset account policy in case of inheritance
if (role.isInheritAccountPolicy()) {
role.setAccountPolicy(null);
}
// reset password policy in case of inheritance
if (role.isInheritPasswordPolicy()) {
role.setPasswordPolicy(null);
}
final SyncopeRole merged = entityManager.merge(role);
for (AbstractVirAttr virtual : merged.getVirtualAttributes()) {
virtual.setValues(role.getVirtualAttribute(virtual.getVirtualSchema().getName()).getValues());
}
entitlementDAO.saveEntitlementRole(merged);
return merged;
}
@Override
public void delete(final SyncopeRole role) {
for (SyncopeRole roleToBeDeleted : findDescendants(role)) {
for (Membership membership : findMemberships(roleToBeDeleted)) {
membership.getSyncopeUser().removeMembership(membership);
userDAO.save(membership.getSyncopeUser());
entityManager.remove(membership);
}
roleToBeDeleted.getEntitlements().clear();
roleToBeDeleted.setParent(null);
roleToBeDeleted.setUserOwner(null);
roleToBeDeleted.setRoleOwner(null);
entityManager.remove(roleToBeDeleted);
entitlementDAO.delete(EntitlementUtil.getEntitlementNameFromRoleId(roleToBeDeleted.getId()));
}
}
@Override
public void delete(final Long id) {
SyncopeRole role = findInternal(id);
if (role == null) {
return;
}
delete(role);
}
}