/* * Copyright (C) 2005-2014 Alfresco Software Limited. * * This file is part of Alfresco * * Alfresco is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Alfresco is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with Alfresco. If not, see . */ package org.alfresco.repo.policy; import java.util.ArrayList; import java.util.Collection; import java.util.List; import java.util.concurrent.locks.ReentrantReadWriteLock; import org.alfresco.service.cmr.repository.NodeRef; import org.alfresco.service.namespace.QName; import org.alfresco.util.LockHelper; /** * Class (Type/Aspect) oriented index of bound behaviours * * Note: Uses Class hierarchy to derive bindings. * * @author David Caruana * */ /*package*/ class ClassBehaviourIndex implements BehaviourIndex { // Lock private ReentrantReadWriteLock lock = new ReentrantReadWriteLock(); // Map of class bindings private BehaviourMap classMap = new BehaviourMap(); // Map of service bindings private BehaviourMap serviceMap = new BehaviourMap(); // List of registered observers private List> observers = new ArrayList>(); // Behaviour Filter private BehaviourFilter filter = null; // Try lock timeout (MNT-11371) private long tryLockTimeout; public void setTryLockTimeout(long tryLockTimeout) { this.tryLockTimeout = tryLockTimeout; } /** * Construct. */ /*package*/ ClassBehaviourIndex(BehaviourFilter filter) { // Observe class binding changes and propagate to our own observers this.classMap.addChangeObserver(new BehaviourChangeObserver() { public void addition(B binding, Behaviour behaviour) { for (BehaviourChangeObserver listener : observers) { listener.addition(binding, behaviour); } } }); // Observe service binding changes and propagate to our own observers this.serviceMap.addChangeObserver(new BehaviourChangeObserver() { public void addition(ServiceBehaviourBinding binding, Behaviour behaviour) { for (BehaviourChangeObserver listener : observers) { // Note: Don't specify class ref as service-level bindings affect all classes listener.addition(null, behaviour); } } }); // Setup state this.filter = filter; } @Override public Collection getAll() { LockHelper.tryLock(lock.readLock(), tryLockTimeout, "getting all behavior definitions in 'ClassBehaviourIndex.getAll()'"); try { List all = new ArrayList(classMap.size() + serviceMap.size()); all.addAll(classMap.getAll()); all.addAll(serviceMap.getAll()); return all; } finally { lock.readLock().unlock(); } } @Override @SuppressWarnings("unchecked") public Collection find(B binding) { LockHelper.tryLock(lock.readLock(), tryLockTimeout, "searching behavior definitions list in 'ClassBehaviourIndex.find()'"); try { List behaviours = new ArrayList(); // Determine if behaviour has been disabled boolean isEnabled = true; if (filter != null) { NodeRef nodeRef = binding.getNodeRef(); QName className = binding.getClassQName(); isEnabled = (nodeRef == null) ? filter.isEnabled(className) : filter.isEnabled(nodeRef, className); } if (isEnabled) { // Find class behaviour by scanning up the class hierarchy List> behaviour = null; while (binding != null) { behaviour = classMap.get(binding); if (behaviour != null) { behaviours.addAll(0, behaviour); // note: list base/generalised before extended/specific } binding = (B)binding.generaliseBinding(); } } // Append all service-level behaviours behaviours.addAll(serviceMap.getAll()); return behaviours; } finally { lock.readLock().unlock(); } } @Override public void addChangeObserver(BehaviourChangeObserver observer) { observers.add(observer); } @Override public BehaviourFilter getFilter() { return filter; } /** * Binds a Class Behaviour into this index * * @param behaviour the class bound behaviour */ public void putClassBehaviour(BehaviourDefinition behaviour) { LockHelper.tryLock(lock.writeLock(), tryLockTimeout, "putting behavior definition in 'ClassBehaviourIndex.putClassBehavior()'"); try { classMap.put(behaviour); } finally { lock.writeLock().unlock(); } } /** * Binds a Service Behaviour into this index * * @param behaviour the service bound behaviour */ public void putServiceBehaviour(BehaviourDefinition behaviour) { LockHelper.tryLock(lock.writeLock(), tryLockTimeout, "putting behavior definition in 'ClassBehaviourIndex.putServiceBehavior()'"); try { serviceMap.put(behaviour); } finally { lock.writeLock().unlock(); } } /** * Remove class behaviour * * @param behaviour */ public void removeClassBehaviour(BehaviourDefinition behaviour) { LockHelper.tryLock(lock.writeLock(), tryLockTimeout, "removing behavior definition in 'ClassBehaviourIndex.removeClassBehavior()'"); try { classMap.remove(behaviour); } finally { lock.writeLock().unlock(); } } }