/* * Copyright 2002-2008 the original author or authors. * * 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 org.springframework.remoting.caucho; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.UndeclaredThrowableException; import java.net.ConnectException; import java.net.MalformedURLException; import com.caucho.burlap.client.BurlapProxyFactory; import com.caucho.burlap.client.BurlapRuntimeException; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; import org.springframework.remoting.RemoteAccessException; import org.springframework.remoting.RemoteConnectFailureException; import org.springframework.remoting.RemoteLookupFailureException; import org.springframework.remoting.RemoteProxyFailureException; import org.springframework.remoting.support.UrlBasedRemoteAccessor; import org.springframework.util.Assert; /** * {@link org.aopalliance.intercept.MethodInterceptor} for accessing a Burlap service. * Supports authentication via username and password. * The service URL must be an HTTP URL exposing a Burlap service. * * <p>Burlap is a slim, XML-based RPC protocol. * For information on Burlap, see the * <a href="http://www.caucho.com/burlap">Burlap website</a> * * <p>Note: There is no requirement for services accessed with this proxy factory * to have been exported using Spring's {@link BurlapServiceExporter}, as there is * no special handling involved. As a consequence, you can also access services that * have been exported using Caucho's {@link com.caucho.burlap.server.BurlapServlet}. * * @author Juergen Hoeller * @since 29.09.2003 * @see #setServiceInterface * @see #setServiceUrl * @see #setUsername * @see #setPassword * @see BurlapServiceExporter * @see BurlapProxyFactoryBean * @see com.caucho.burlap.client.BurlapProxyFactory * @see com.caucho.burlap.server.BurlapServlet */ public class BurlapClientInterceptor extends UrlBasedRemoteAccessor implements MethodInterceptor { private BurlapProxyFactory proxyFactory = new BurlapProxyFactory(); private Object burlapProxy; /** * Set the BurlapProxyFactory instance to use. * If not specified, a default BurlapProxyFactory will be created. * <p>Allows to use an externally configured factory instance, * in particular a custom BurlapProxyFactory subclass. */ public void setProxyFactory(BurlapProxyFactory proxyFactory) { this.proxyFactory = (proxyFactory != null ? proxyFactory : new BurlapProxyFactory()); } /** * Set the username that this factory should use to access the remote service. * Default is none. * <p>The username will be sent by Burlap via HTTP Basic Authentication. * @see com.caucho.burlap.client.BurlapProxyFactory#setUser */ public void setUsername(String username) { this.proxyFactory.setUser(username); } /** * Set the password that this factory should use to access the remote service. * Default is none. * <p>The password will be sent by Burlap via HTTP Basic Authentication. * @see com.caucho.burlap.client.BurlapProxyFactory#setPassword */ public void setPassword(String password) { this.proxyFactory.setPassword(password); } /** * Set whether overloaded methods should be enabled for remote invocations. * Default is "false". * @see com.caucho.burlap.client.BurlapProxyFactory#setOverloadEnabled */ public void setOverloadEnabled(boolean overloadEnabled) { this.proxyFactory.setOverloadEnabled(overloadEnabled); } public void afterPropertiesSet() { super.afterPropertiesSet(); prepare(); } /** * Initialize the Burlap proxy for this interceptor. * @throws RemoteLookupFailureException if the service URL is invalid */ public void prepare() throws RemoteLookupFailureException { try { this.burlapProxy = createBurlapProxy(this.proxyFactory); } catch (MalformedURLException ex) { throw new RemoteLookupFailureException("Service URL [" + getServiceUrl() + "] is invalid", ex); } } /** * Create the Burlap proxy that is wrapped by this interceptor. * @param proxyFactory the proxy factory to use * @return the Burlap proxy * @throws MalformedURLException if thrown by the proxy factory * @see com.caucho.burlap.client.BurlapProxyFactory#create */ protected Object createBurlapProxy(BurlapProxyFactory proxyFactory) throws MalformedURLException { Assert.notNull(getServiceInterface(), "Property 'serviceInterface' is required"); return proxyFactory.create(getServiceInterface(), getServiceUrl()); } public Object invoke(MethodInvocation invocation) throws Throwable { if (this.burlapProxy == null) { throw new IllegalStateException("BurlapClientInterceptor is not properly initialized - " + "invoke 'prepare' before attempting any operations"); } ClassLoader originalClassLoader = overrideThreadContextClassLoader(); try { return invocation.getMethod().invoke(this.burlapProxy, invocation.getArguments()); } catch (InvocationTargetException ex) { if (ex.getTargetException() instanceof BurlapRuntimeException) { BurlapRuntimeException bre = (BurlapRuntimeException) ex.getTargetException(); Throwable rootCause = (bre.getRootCause() != null ? bre.getRootCause() : bre); throw convertBurlapAccessException(rootCause); } else if (ex.getTargetException() instanceof UndeclaredThrowableException) { UndeclaredThrowableException utex = (UndeclaredThrowableException) ex.getTargetException(); throw convertBurlapAccessException(utex.getUndeclaredThrowable()); } throw ex.getTargetException(); } catch (Throwable ex) { throw new RemoteProxyFailureException( "Failed to invoke Burlap proxy for remote service [" + getServiceUrl() + "]", ex); } finally { resetThreadContextClassLoader(originalClassLoader); } } /** * Convert the given Burlap access exception to an appropriate * Spring RemoteAccessException. * @param ex the exception to convert * @return the RemoteAccessException to throw */ protected RemoteAccessException convertBurlapAccessException(Throwable ex) { if (ex instanceof ConnectException) { return new RemoteConnectFailureException( "Cannot connect to Burlap remote service at [" + getServiceUrl() + "]", ex); } else { return new RemoteAccessException( "Cannot access Burlap remote service at [" + getServiceUrl() + "]", ex); } } }