/* * 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.audit; import java.lang.reflect.Method; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import org.alfresco.service.PublicService; import org.aopalliance.intercept.MethodInvocation; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.springframework.beans.BeansException; import org.springframework.beans.factory.config.BeanFactoryPostProcessor; import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; /** * Identify public services by method invocation. Look ups are cached on a thread local as they are quite expensive. All public service names end with "Service" and start with * capital letter. This pattern is used to filter bean names. TODO: Look at pulling out all the mappings at start up. * * @author Andy Hind */ public class PublicServiceIdentifierImpl implements PublicServiceIdentifier, BeanFactoryPostProcessor { private static Log s_logger = LogFactory.getLog(PublicServiceIdentifierImpl.class); private static ThreadLocal> methodToServiceMap = new ThreadLocal>(); private ConfigurableListableBeanFactory beanFactory; public PublicServiceIdentifierImpl() { super(); } public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { this.beanFactory = beanFactory; } public String getPublicServiceName(MethodInvocation mi) { return getServiceName(mi); } /** * Cache service name look up. * * @param mi * @return * @throws BeansException */ private String getServiceName(MethodInvocation mi) throws BeansException { if (methodToServiceMap.get() == null) { methodToServiceMap.set(new HashMap()); } Method method = mi.getMethod(); String serviceName = methodToServiceMap.get().get(method); if (serviceName == null) { serviceName = getServiceNameImpl(mi); methodToServiceMap.get().put(method, serviceName); } else { if (s_logger.isDebugEnabled()) { s_logger.debug("Cached look up for " + serviceName + "." + method.getName()); } } return serviceName; } /** * Do the look up by interface type. * * @param mi * @return * @throws BeansException */ private String getServiceNameImpl(MethodInvocation mi) throws BeansException { Class clazz = mi.getThis().getClass(); while (clazz != null) { Class[] interfaces = clazz.getInterfaces(); for (Class iFace : interfaces) { Class publicServiceInterface = findPublicService(iFace); if (publicServiceInterface != null) { Map beans = beanFactory.getBeansOfType(publicServiceInterface); Iterator iter = beans.entrySet().iterator(); while (iter.hasNext()) { Map.Entry entry = (Map.Entry) iter.next(); String serviceName = (String) entry.getKey(); if ((serviceName.endsWith("Service")) && (Character.isUpperCase(serviceName.charAt(0))) && !serviceName.equals("DescriptorService")) { return serviceName; } } } } clazz = clazz.getSuperclass(); } return null; } /** * We use a marker annotation to identify public interfaces. * The interfaces have to be walked to determine if a public interface is implemented. * * Only one public service interface is expected. * * @param clazz * @return */ @SuppressWarnings("unchecked") private Class findPublicService(Class clazz) { if (!clazz.isInterface()) { return null; } if (clazz.isAnnotationPresent(PublicService.class)) { return clazz; } Class[] classes = clazz.getInterfaces(); for(Class implemented: classes) { Class answer = findPublicService(implemented); if(answer != null) { return answer; } } return null; } }