Added MethodDefinitionSourceAdvisor for performance and autoproxying.
This commit is contained in:
parent
740517c55b
commit
8d973af603
|
@ -1,3 +1,8 @@
|
|||
Changes in version 0.7 (2004-xx-xx)
|
||||
-----------------------------------
|
||||
|
||||
* Added MethodDefinitionSourceAdvisor for performance and autoproxying
|
||||
|
||||
Changes in version 0.6.1 (2004-09-25)
|
||||
-------------------------------------
|
||||
|
||||
|
|
|
@ -0,0 +1,120 @@
|
|||
/* Copyright 2004 Acegi Technology Pty Limited
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
package net.sf.acegisecurity.intercept.method;
|
||||
|
||||
import org.aopalliance.intercept.MethodInvocation;
|
||||
|
||||
import org.springframework.aop.framework.AopConfigException;
|
||||
import org.springframework.aop.support.StaticMethodMatcherPointcutAdvisor;
|
||||
|
||||
import java.lang.reflect.AccessibleObject;
|
||||
import java.lang.reflect.Method;
|
||||
|
||||
|
||||
/**
|
||||
* Advisor driven by a {@link MethodDefinitionSource}, used to exclude a {@link
|
||||
* MethodSecurityInterceptor} from public (ie non-secure) methods.
|
||||
*
|
||||
* <p>
|
||||
* Because the AOP framework caches advice calculations, this is normally
|
||||
* faster than just letting the <code>MethodSecurityInterceptor</code> run and
|
||||
* find out itself that it has no work to do.
|
||||
* </p>
|
||||
*
|
||||
* <p>
|
||||
* This class also allows the use of Spring's
|
||||
* <code>DefaultAdvisorAutoProxyCreator</code>, which makes configuration
|
||||
* easier than setup a <code>ProxyFactoryBean</code> for each object requiring
|
||||
* security. Note that autoproxying is not supported for BeanFactory
|
||||
* implementations, as post-processing is automatic only for application
|
||||
* contexts.
|
||||
* </p>
|
||||
*
|
||||
* <p>
|
||||
* Based on Spring's TransactionAttributeSourceAdvisor.
|
||||
* </p>
|
||||
*
|
||||
* @author Ben Alex
|
||||
* @version $Id$
|
||||
*/
|
||||
public class MethodDefinitionSourceAdvisor
|
||||
extends StaticMethodMatcherPointcutAdvisor {
|
||||
//~ Instance fields ========================================================
|
||||
|
||||
private MethodDefinitionSource transactionAttributeSource;
|
||||
|
||||
//~ Constructors ===========================================================
|
||||
|
||||
public MethodDefinitionSourceAdvisor(MethodSecurityInterceptor advice) {
|
||||
super(advice);
|
||||
|
||||
if (advice.getObjectDefinitionSource() == null) {
|
||||
throw new AopConfigException(
|
||||
"Cannot construct a MethodDefinitionSourceAdvisor using a "
|
||||
+ "MethodSecurityInterceptor that has no ObjectDefinitionSource configured");
|
||||
}
|
||||
|
||||
this.transactionAttributeSource = advice.getObjectDefinitionSource();
|
||||
}
|
||||
|
||||
//~ Methods ================================================================
|
||||
|
||||
public boolean matches(Method m, Class targetClass) {
|
||||
MethodInvocation methodInvocation = new InternalMethodInvocation(m);
|
||||
|
||||
return (this.transactionAttributeSource.getAttributes(methodInvocation) != null);
|
||||
}
|
||||
|
||||
//~ Inner Classes ==========================================================
|
||||
|
||||
/**
|
||||
* Represents a <code>MethodInvocation</code>.
|
||||
*
|
||||
* <p>
|
||||
* Required as <code>MethodDefinitionSource</code> only supports lookup of
|
||||
* configuration attributes for <code>MethodInvocation</code>s.
|
||||
* </p>
|
||||
*/
|
||||
private class InternalMethodInvocation implements MethodInvocation {
|
||||
Method method;
|
||||
|
||||
public InternalMethodInvocation(Method method) {
|
||||
this.method = method;
|
||||
}
|
||||
|
||||
private InternalMethodInvocation() {}
|
||||
|
||||
public Object[] getArguments() {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
|
||||
public Method getMethod() {
|
||||
return this.method;
|
||||
}
|
||||
|
||||
public AccessibleObject getStaticPart() {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
|
||||
public Object getThis() {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
|
||||
public Object proceed() throws Throwable {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,98 @@
|
|||
/* Copyright 2004 Acegi Technology Pty Limited
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
package net.sf.acegisecurity.intercept.method;
|
||||
|
||||
import junit.framework.TestCase;
|
||||
|
||||
import net.sf.acegisecurity.TargetObject;
|
||||
|
||||
import org.springframework.aop.framework.AopConfigException;
|
||||
|
||||
import java.lang.reflect.Method;
|
||||
|
||||
|
||||
/**
|
||||
* Tests {@link MethodDefinitionSourceAdvisor}.
|
||||
*
|
||||
* @author Ben Alex
|
||||
* @version $Id$
|
||||
*/
|
||||
public class MethodDefinitionSourceAdvisorTests extends TestCase {
|
||||
//~ Constructors ===========================================================
|
||||
|
||||
public MethodDefinitionSourceAdvisorTests() {
|
||||
super();
|
||||
}
|
||||
|
||||
public MethodDefinitionSourceAdvisorTests(String arg0) {
|
||||
super(arg0);
|
||||
}
|
||||
|
||||
//~ Methods ================================================================
|
||||
|
||||
public final void setUp() throws Exception {
|
||||
super.setUp();
|
||||
}
|
||||
|
||||
public static void main(String[] args) {
|
||||
junit.textui.TestRunner.run(MethodDefinitionSourceAdvisorTests.class);
|
||||
}
|
||||
|
||||
public void testAdvisorReturnsFalseWhenMethodInvocationNotDefined()
|
||||
throws Exception {
|
||||
Class clazz = TargetObject.class;
|
||||
Method method = clazz.getMethod("makeLowerCase",
|
||||
new Class[] {String.class});
|
||||
|
||||
MethodDefinitionSourceAdvisor advisor = new MethodDefinitionSourceAdvisor(getInterceptor());
|
||||
assertFalse(advisor.matches(method, clazz));
|
||||
}
|
||||
|
||||
public void testAdvisorReturnsTrueWhenMethodInvocationIsDefined()
|
||||
throws Exception {
|
||||
Class clazz = TargetObject.class;
|
||||
Method method = clazz.getMethod("countLength",
|
||||
new Class[] {String.class});
|
||||
|
||||
MethodDefinitionSourceAdvisor advisor = new MethodDefinitionSourceAdvisor(getInterceptor());
|
||||
assertTrue(advisor.matches(method, clazz));
|
||||
}
|
||||
|
||||
public void testDetectsImproperlyConfiguredAdvice() {
|
||||
MethodSecurityInterceptor msi = new MethodSecurityInterceptor();
|
||||
|
||||
try {
|
||||
new MethodDefinitionSourceAdvisor(msi);
|
||||
fail(
|
||||
"Should have detected null ObjectDefinitionSource and thrown AopConfigException");
|
||||
} catch (AopConfigException expected) {
|
||||
assertTrue(true);
|
||||
}
|
||||
}
|
||||
|
||||
private MethodSecurityInterceptor getInterceptor() {
|
||||
MethodDefinitionSourceEditor editor = new MethodDefinitionSourceEditor();
|
||||
editor.setAsText(
|
||||
"net.sf.acegisecurity.TargetObject.countLength=ROLE_NOT_USED");
|
||||
|
||||
MethodDefinitionMap map = (MethodDefinitionMap) editor.getValue();
|
||||
|
||||
MethodSecurityInterceptor msi = new MethodSecurityInterceptor();
|
||||
msi.setObjectDefinitionSource(map);
|
||||
|
||||
return msi;
|
||||
}
|
||||
}
|
|
@ -75,20 +75,23 @@
|
|||
|
||||
<bean id="bankService" class="sample.attributes.BankServiceImpl"/>
|
||||
|
||||
<bean id="autoProxyCreator" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
|
||||
<!-- names of the interceptors that will be applied by the proxy -->
|
||||
<property name="interceptorNames">
|
||||
<list>
|
||||
<value>securityInterceptor</value>
|
||||
</list>
|
||||
</property>
|
||||
<!--
|
||||
This bean is a postprocessor that will automatically apply relevant advisors
|
||||
to any bean in child factories.
|
||||
-->
|
||||
<bean id="autoproxy"
|
||||
class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator">
|
||||
</bean>
|
||||
|
||||
<!-- the bean names to automatically generate proxies for -->
|
||||
<property name="beanNames">
|
||||
<list>
|
||||
<idref local="bankService"/>
|
||||
</list>
|
||||
</property>
|
||||
<!--
|
||||
AOP advisor that will automatically wire the MethodSecurityInterceptor (above)
|
||||
into BankServiceImpl (above). The configuration attributes used are obtained
|
||||
from the securityInterceptor.objectDefinitionSouce, which in the
|
||||
above configuration is a Commons Attributes-based source.
|
||||
-->
|
||||
<bean id="methodSecurityAdvisor"
|
||||
class="net.sf.acegisecurity.intercept.method.MethodDefinitionSourceAdvisor"
|
||||
autowire="constructor" >
|
||||
</bean>
|
||||
|
||||
</beans>
|
||||
|
|
Loading…
Reference in New Issue