Files
.externalToolBuilders
config
source
cpp
java
org
alfresco
filesys
jcr
linkvalidation
model
repo
action
admin
attributes
audit
avm
cache
clt
coci
config
configuration
content
copy
deploy
descriptor
dictionary
domain
exporter
forum
i18n
importer
jscript
lock
model
module
node
ownable
policy
registration
AssociationPolicy.java
AssociationPolicyDelegate.java
BaseBehaviour.java
Behaviour.java
BehaviourBinding.java
BehaviourChangeObserver.java
BehaviourDefinition.java
BehaviourFilter.java
BehaviourFilterImpl.java
BehaviourIndex.java
BehaviourMap.java
CachedPolicyFactory.java
ClassBehaviourBinding.java
ClassBehaviourIndex.java
ClassFeatureBehaviourBinding.java
ClassPolicy.java
ClassPolicyDelegate.java
JavaBehaviour.java
Policy.java
PolicyComponent.java
PolicyComponentImpl.java
PolicyComponentTest.java
PolicyComponentTransactionTest.java
PolicyDefinition.java
PolicyException.java
PolicyFactory.java
PolicyList.java
PolicyScope.java
PolicyType.java
PropertyPolicy.java
PropertyPolicyDelegate.java
ServiceBehaviourBinding.java
TransactionBehaviourQueue.java
TransactionInvocationHandlerFactory.java
policycomponenttest_model.xml
processor
remote
rule
search
security
service
template
tenant
transaction
version
workflow
sandbox
service
tools
util
apache
queryRegister.dtd
meta-inf
test-resources
web
.classpath
.project
build.xml
alfresco-community-repo/source/java/org/alfresco/repo/policy/PolicyFactory.java
Paul Holmes-Higgin 4e2300f095 Updated copyright
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@5186 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
2007-02-19 17:17:36 +00:00

271 lines
8.8 KiB
Java

/*
* Copyright (C) 2005-2007 Alfresco Software Limited.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program 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 General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
* As a special exception to the terms and conditions of version 2.0 of
* the GPL, you may redistribute this Program in connection with Free/Libre
* and Open Source Software ("FLOSS") applications as described in Alfresco's
* FLOSS exception. You should have recieved a copy of the text describing
* the FLOSS exception, and it is also available here:
* http://www.alfresco.com/legal/licensing"
*/
package org.alfresco.repo.policy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import org.alfresco.repo.policy.Behaviour.NotificationFrequency;
/**
* A Policy Factory is responsible for creating Policy implementations.
*
* @author David Caruana
*
* @param <B> the type of binding
* @param <P> the policy interface
*/
/*package*/ class PolicyFactory<B extends BehaviourBinding, P extends Policy>
{
// Behaviour Index to query
private BehaviourIndex<B> index;
// The policy interface class
private Class<P> policyClass;
// NOOP Invocation Handler
private static InvocationHandler NOOPHandler = new NOOPHandler();
// Transaction Invocation Handler Factory
private static TransactionInvocationHandlerFactory transactionHandlerFactory = null;
/**
* Construct.
*
* @param policyClass the policy class
* @param index the behaviour index to query
*/
/*package*/ PolicyFactory(Class<P> policyClass, BehaviourIndex<B> index)
{
this.policyClass = policyClass;
this.index = index;
}
/**
* Sets the Transaction Invocation Handler
*
* @param handlerFactory
*/
protected static void setTransactionInvocationHandlerFactory(TransactionInvocationHandlerFactory factory)
{
transactionHandlerFactory = factory;
}
/**
* Gets the Policy class created by this factory
*
* @return the policy class
*/
protected Class<P> getPolicyClass()
{
return policyClass;
}
/**
* Construct a Policy implementation for the specified binding
*
* @param binding the binding
* @return the policy implementation
*/
public P create(B binding)
{
Collection<P> policyInterfaces = createList(binding);
return toPolicy(policyInterfaces);
}
/**
* Construct a collection of Policy implementations for the specified binding
*
* @param binding the binding
* @return the collection of policy implementations
*/
@SuppressWarnings("unchecked")
public Collection<P> createList(B binding)
{
Collection<BehaviourDefinition> behaviourDefs = index.find(binding);
List<P> policyInterfaces = new ArrayList<P>(behaviourDefs.size());
for (BehaviourDefinition behaviourDef : behaviourDefs)
{
Behaviour behaviour = behaviourDef.getBehaviour();
P policyIF = behaviour.getInterface(policyClass);
if (!(behaviour.getNotificationFrequency().equals(NotificationFrequency.EVERY_EVENT)))
{
// wrap behaviour in transaction proxy which deals with delaying invocation until necessary
if (transactionHandlerFactory == null)
{
throw new PolicyException("Transaction-level policies not supported as transaction support for the Policy Component has not been initialised.");
}
InvocationHandler trxHandler = transactionHandlerFactory.createHandler(behaviour, behaviourDef.getPolicyDefinition(), policyIF);
policyIF = (P)Proxy.newProxyInstance(policyClass.getClassLoader(), new Class[]{policyClass}, trxHandler);
}
policyInterfaces.add(policyIF);
}
return policyInterfaces;
}
/**
* Construct a single aggregate policy implementation for the specified
* collection of policy implementations.
*
* @param policyList the policy implementations to aggregate
* @return the aggregate policy implementation
*/
@SuppressWarnings("unchecked")
public P toPolicy(Collection<P> policyList)
{
if (policyList.size() == 1)
{
return policyList.iterator().next();
}
else if (policyList.size() == 0)
{
return (P)Proxy.newProxyInstance(policyClass.getClassLoader(),
new Class[]{policyClass}, NOOPHandler);
}
else
{
return (P)Proxy.newProxyInstance(policyClass.getClassLoader(),
new Class[]{policyClass, PolicyList.class}, new MultiHandler<P>(policyList));
}
}
/**
* NOOP Invocation Handler.
*
* @author David Caruana
*
*/
private static class NOOPHandler implements InvocationHandler
{
/* (non-Javadoc)
* @see java.lang.reflect.InvocationHandler#invoke(java.lang.Object, java.lang.reflect.Method, java.lang.Object[])
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
{
if (method.getName().equals("toString"))
{
return toString();
}
else if (method.getName().equals("hashCode"))
{
return hashCode();
}
else if (method.getName().equals("equals"))
{
return equals(args[0]);
}
return null;
}
}
/**
* Multi-policy Invocation Handler.
*
* @author David Caruana
*
* @param <P> policy interface
*/
private static class MultiHandler<P extends Policy> implements InvocationHandler, PolicyList
{
private Collection<P> policyInterfaces;
/**
* Construct
*
* @param policyInterfaces the collection of policy implementations
*/
public MultiHandler(Collection<P> policyInterfaces)
{
this.policyInterfaces = Collections.unmodifiableCollection(policyInterfaces);
}
/* (non-Javadoc)
* @see java.lang.reflect.InvocationHandler#invoke(java.lang.Object, java.lang.reflect.Method, java.lang.Object[])
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
{
// Handle PolicyList level methods
if (method.getDeclaringClass().equals(PolicyList.class))
{
return method.invoke(this, args);
}
// Handle Object level methods
if (method.getName().equals("toString"))
{
return toString() + ": wrapped " + policyInterfaces.size() + " policies";
}
else if (method.getName().equals("hashCode"))
{
return hashCode();
}
else if (method.getName().equals("equals"))
{
return equals(args[0]);
}
// Invoke each wrapped policy in turn
try
{
Object result = null;
for (P policyInterface : policyInterfaces)
{
result = method.invoke(policyInterface, args);
}
return result;
}
catch (InvocationTargetException e)
{
throw e.getTargetException();
}
}
/* (non-Javadoc)
* @see org.alfresco.repo.policy.PolicyList#getPolicies()
*/
public Collection getPolicies()
{
return policyInterfaces;
}
}
}