2
0
Эх сурвалжийг харах

Added MethodDefinitionSourceAdvisor for performance and autoproxying.

Ben Alex 21 жил өмнө
parent
commit
8d973af603

+ 5 - 0
changelog.txt

@@ -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)
 -------------------------------------
 

+ 120 - 0
core/src/main/java/org/acegisecurity/intercept/method/MethodDefinitionSourceAdvisor.java

@@ -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();
+        }
+    }
+}

+ 98 - 0
core/src/test/java/org/acegisecurity/intercept/method/MethodDefinitionSourceAdvisorTests.java

@@ -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;
+    }
+}

+ 17 - 14
samples/attributes/src/applicationContext.xml

@@ -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>
-    </bean>
+	<!--
+		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>