001    /**
002     * Copyright 2005-2012 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.springframework.cache.interceptor;
017    
018    import org.aopalliance.aop.Advice;
019    import org.springframework.aop.Pointcut;
020    import org.springframework.aop.framework.ProxyFactory;
021    import org.springframework.aop.support.AbstractPointcutAdvisor;
022    import org.springframework.aop.support.AopUtils;
023    import org.springframework.cache.CacheManager;
024    import org.springframework.cache.annotation.AnnotationCacheOperationSource;
025    
026    /**
027     * Methods for creating caching proxies.
028     */
029    public final class CacheProxy {
030        /*
031            WARNING!  This class uses internal spring code and could be subject to breaking
032            changes between Spring releases.
033    
034            This class is needed because Spring does not yet have a way to apply caching advise to
035            existing objects which is required for our remote service proxies.
036    
037            Hopefully, Spring will provide this capability in the future and we can remove this class.
038        */
039    
040    
041        private CacheProxy() {
042            throw new IllegalArgumentException("do not call");
043        }
044    
045        /**
046         * This creates a caching proxy around a existing object if the proxy can be applied.
047         *
048         * @param o the object to proxy.  cannot be null.
049         * @param cacheManager the cacheManager to handle the caching operations.  cannot be null.
050         * @return a proxy.  will not return null.
051         * @throws IllegalArgumentException if the o or cacheManager is null
052         */
053        public static <T> T createCacheProxy(T o, CacheManager cacheManager) {
054            if (o == null) {
055                throw new IllegalArgumentException("o is null");
056            }
057    
058            if (cacheManager == null) {
059                throw new IllegalArgumentException("cacheManager is null");
060            }
061    
062            //this method is following the logic found in AnnotationDrivenCacheBeanDefinitionParser.AopAutoProxyConfigurer
063    
064            final AnnotationCacheOperationSource source = new AnnotationCacheOperationSource();
065    
066            final CacheInterceptor interceptor = new CacheInterceptor();
067            interceptor.setCacheManager(cacheManager);
068            interceptor.setCacheOperationSources(source);
069            //important!  must call afterPropertiesSet or does not execute caching logic
070            interceptor.afterPropertiesSet();
071    
072            final ProxyPointcut pointcut = new ProxyPointcut(source);
073            final ProxyPointCutAdvisor advisor = new ProxyPointCutAdvisor(pointcut, interceptor);
074    
075            if (AopUtils.canApply(advisor, o.getClass())) {
076                final ProxyFactory proxyFactory = new ProxyFactory(o);
077                proxyFactory.addAdvisor(advisor);
078                return (T) proxyFactory.getProxy();
079            }
080            return o;
081        }
082    
083        private static class ProxyPointcut extends CacheOperationSourcePointcut {
084            private static final long serialVersionUID = 6050508570006106939L;
085    
086            private final AnnotationCacheOperationSource source;
087    
088            private ProxyPointcut(AnnotationCacheOperationSource source) {
089                this.source = source;
090            }
091    
092            @Override
093            protected CacheOperationSource getCacheOperationSource() {
094                return source;
095            }
096        }
097    
098        private static class ProxyPointCutAdvisor extends AbstractPointcutAdvisor {
099    
100            private static final long serialVersionUID = 6050508570006106939L;
101    
102            private final CacheOperationSourcePointcut pointcut;
103            private final CacheInterceptor interceptor;
104    
105            private ProxyPointCutAdvisor(CacheOperationSourcePointcut pointcut, CacheInterceptor interceptor) {
106                this.pointcut = pointcut;
107                this.interceptor = interceptor;
108            }
109    
110            @Override
111            public Pointcut getPointcut() {
112                return pointcut;
113            }
114    
115            @Override
116            public Advice getAdvice() {
117                return interceptor;
118            }
119        }
120    }