Fine-tune HTTP/RMI Invoker exception handling

Issue: SPR-15684
This commit is contained in:
Juergen Hoeller 2017-06-20 16:50:35 +02:00
parent c244f33f84
commit 535103cd52
12 changed files with 61 additions and 55 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2016 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -35,6 +35,7 @@ import org.springframework.remoting.RemoteLookupFailureException;
import org.springframework.remoting.support.DefaultRemoteInvocationFactory; import org.springframework.remoting.support.DefaultRemoteInvocationFactory;
import org.springframework.remoting.support.RemoteInvocation; import org.springframework.remoting.support.RemoteInvocation;
import org.springframework.remoting.support.RemoteInvocationFactory; import org.springframework.remoting.support.RemoteInvocationFactory;
import org.springframework.util.Assert;
/** /**
* {@link org.aopalliance.intercept.MethodInterceptor} for accessing RMI services * {@link org.aopalliance.intercept.MethodInterceptor} for accessing RMI services
@ -97,9 +98,8 @@ public class JndiRmiClientInterceptor extends JndiObjectLocator implements Metho
* but can also be optional if the lookup returns a typed stub. * but can also be optional if the lookup returns a typed stub.
*/ */
public void setServiceInterface(Class<?> serviceInterface) { public void setServiceInterface(Class<?> serviceInterface) {
if (serviceInterface != null && !serviceInterface.isInterface()) { Assert.notNull(serviceInterface, "'serviceInterface' must not be null");
throw new IllegalArgumentException("'serviceInterface' must be an interface"); Assert.isTrue(serviceInterface.isInterface(), "'serviceInterface' must be an interface");
}
this.serviceInterface = serviceInterface; this.serviceInterface = serviceInterface;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2012 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -21,6 +21,7 @@ import javax.naming.NamingException;
import org.springframework.aop.framework.ProxyFactory; import org.springframework.aop.framework.ProxyFactory;
import org.springframework.beans.factory.BeanClassLoaderAware; import org.springframework.beans.factory.BeanClassLoaderAware;
import org.springframework.beans.factory.FactoryBean; import org.springframework.beans.factory.FactoryBean;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils; import org.springframework.util.ClassUtils;
/** /**
@ -77,10 +78,9 @@ public class JndiRmiProxyFactoryBean extends JndiRmiClientInterceptor
@Override @Override
public void afterPropertiesSet() throws NamingException { public void afterPropertiesSet() throws NamingException {
super.afterPropertiesSet(); super.afterPropertiesSet();
if (getServiceInterface() == null) { Class<?> ifc = getServiceInterface();
throw new IllegalArgumentException("Property 'serviceInterface' is required"); Assert.notNull(ifc, "Property 'serviceInterface' is required");
} this.serviceProxy = new ProxyFactory(ifc, this).getProxy(this.beanClassLoader);
this.serviceProxy = new ProxyFactory(getServiceInterface(), this).getProxy(this.beanClassLoader);
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2012 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -107,7 +107,9 @@ public abstract class RemoteInvocationSerializingExporter extends RemoteInvocati
} }
protected final Object getProxy() { protected final Object getProxy() {
Assert.notNull(this.proxy, ClassUtils.getShortName(getClass()) + " has not been initialized"); if (this.proxy == null) {
throw new IllegalStateException(ClassUtils.getShortName(getClass()) + " has not been initialized");
}
return this.proxy; return this.proxy;
} }
@ -142,7 +144,7 @@ public abstract class RemoteInvocationSerializingExporter extends RemoteInvocati
Object obj = ois.readObject(); Object obj = ois.readObject();
if (!(obj instanceof RemoteInvocation)) { if (!(obj instanceof RemoteInvocation)) {
throw new RemoteException("Deserialized object needs to be assignable to type [" + throw new RemoteException("Deserialized object needs to be assignable to type [" +
RemoteInvocation.class.getName() + "]: " + obj); RemoteInvocation.class.getName() + "]: " + ClassUtils.getDescriptiveType(obj));
} }
return (RemoteInvocation) obj; return (RemoteInvocation) obj;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2016 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -19,6 +19,7 @@ package org.springframework.remoting.rmi;
import org.springframework.aop.framework.ProxyFactory; import org.springframework.aop.framework.ProxyFactory;
import org.springframework.beans.factory.BeanClassLoaderAware; import org.springframework.beans.factory.BeanClassLoaderAware;
import org.springframework.beans.factory.FactoryBean; import org.springframework.beans.factory.FactoryBean;
import org.springframework.util.Assert;
/** /**
* {@link FactoryBean} for RMI proxies, supporting both conventional RMI services * {@link FactoryBean} for RMI proxies, supporting both conventional RMI services
@ -65,10 +66,9 @@ public class RmiProxyFactoryBean extends RmiClientInterceptor implements Factory
@Override @Override
public void afterPropertiesSet() { public void afterPropertiesSet() {
super.afterPropertiesSet(); super.afterPropertiesSet();
if (getServiceInterface() == null) { Class<?> ifc = getServiceInterface();
throw new IllegalArgumentException("Property 'serviceInterface' is required"); Assert.notNull(ifc, "Property 'serviceInterface' is required");
} this.serviceProxy = new ProxyFactory(ifc, this).getProxy(getBeanClassLoader());
this.serviceProxy = new ProxyFactory(getServiceInterface(), this).getProxy(getBeanClassLoader());
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2012 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -16,6 +16,8 @@
package org.springframework.remoting.support; package org.springframework.remoting.support;
import org.springframework.util.Assert;
/** /**
* Abstract base class for classes that access a remote service. * Abstract base class for classes that access a remote service.
* Provides a "serviceInterface" bean property. * Provides a "serviceInterface" bean property.
@ -46,9 +48,8 @@ public abstract class RemoteAccessor extends RemotingSupport {
* but can also be optional if the lookup returns a typed proxy. * but can also be optional if the lookup returns a typed proxy.
*/ */
public void setServiceInterface(Class<?> serviceInterface) { public void setServiceInterface(Class<?> serviceInterface) {
if (serviceInterface != null && !serviceInterface.isInterface()) { Assert.notNull(serviceInterface, "'serviceInterface' must not be null");
throw new IllegalArgumentException("'serviceInterface' must be an interface"); Assert.isTrue(serviceInterface.isInterface(), "'serviceInterface' must be an interface");
}
this.serviceInterface = serviceInterface; this.serviceInterface = serviceInterface;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2012 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -19,6 +19,7 @@ package org.springframework.remoting.support;
import org.springframework.aop.framework.ProxyFactory; import org.springframework.aop.framework.ProxyFactory;
import org.springframework.aop.framework.adapter.AdvisorAdapterRegistry; import org.springframework.aop.framework.adapter.AdvisorAdapterRegistry;
import org.springframework.aop.framework.adapter.GlobalAdvisorAdapterRegistry; import org.springframework.aop.framework.adapter.GlobalAdvisorAdapterRegistry;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils; import org.springframework.util.ClassUtils;
/** /**
@ -63,9 +64,8 @@ public abstract class RemoteExporter extends RemotingSupport {
* The interface must be suitable for the particular service and remoting strategy. * The interface must be suitable for the particular service and remoting strategy.
*/ */
public void setServiceInterface(Class<?> serviceInterface) { public void setServiceInterface(Class<?> serviceInterface) {
if (serviceInterface != null && !serviceInterface.isInterface()) { Assert.notNull(serviceInterface, "'serviceInterface' must not be null");
throw new IllegalArgumentException("'serviceInterface' must be an interface"); Assert.isTrue(serviceInterface.isInterface(), "'serviceInterface' must be an interface");
}
this.serviceInterface = serviceInterface; this.serviceInterface = serviceInterface;
} }
@ -89,7 +89,7 @@ public abstract class RemoteExporter extends RemotingSupport {
* @see RemoteInvocationTraceInterceptor * @see RemoteInvocationTraceInterceptor
*/ */
public void setRegisterTraceInterceptor(boolean registerTraceInterceptor) { public void setRegisterTraceInterceptor(boolean registerTraceInterceptor) {
this.registerTraceInterceptor = Boolean.valueOf(registerTraceInterceptor); this.registerTraceInterceptor = registerTraceInterceptor;
} }
/** /**
@ -110,9 +110,7 @@ public abstract class RemoteExporter extends RemotingSupport {
* @see #setService * @see #setService
*/ */
protected void checkService() throws IllegalArgumentException { protected void checkService() throws IllegalArgumentException {
if (getService() == null) { Assert.notNull(getService(), "Property 'service' is required");
throw new IllegalArgumentException("Property 'service' is required");
}
} }
/** /**
@ -123,10 +121,9 @@ public abstract class RemoteExporter extends RemotingSupport {
*/ */
protected void checkServiceInterface() throws IllegalArgumentException { protected void checkServiceInterface() throws IllegalArgumentException {
Class<?> serviceInterface = getServiceInterface(); Class<?> serviceInterface = getServiceInterface();
Assert.notNull(serviceInterface, "Property 'serviceInterface' is required");
Object service = getService(); Object service = getService();
if (serviceInterface == null) {
throw new IllegalArgumentException("Property 'serviceInterface' is required");
}
if (service instanceof String) { if (service instanceof String) {
throw new IllegalArgumentException("Service [" + service + "] is a String " + throw new IllegalArgumentException("Service [" + service + "] is a String " +
"rather than an actual service reference: Have you accidentally specified " + "rather than an actual service reference: Have you accidentally specified " +
@ -153,20 +150,23 @@ public abstract class RemoteExporter extends RemotingSupport {
protected Object getProxyForService() { protected Object getProxyForService() {
checkService(); checkService();
checkServiceInterface(); checkServiceInterface();
ProxyFactory proxyFactory = new ProxyFactory(); ProxyFactory proxyFactory = new ProxyFactory();
proxyFactory.addInterface(getServiceInterface()); proxyFactory.addInterface(getServiceInterface());
if (this.registerTraceInterceptor != null ?
this.registerTraceInterceptor.booleanValue() : this.interceptors == null) { if (this.registerTraceInterceptor != null ? this.registerTraceInterceptor : this.interceptors == null) {
proxyFactory.addAdvice(new RemoteInvocationTraceInterceptor(getExporterName())); proxyFactory.addAdvice(new RemoteInvocationTraceInterceptor(getExporterName()));
} }
if (this.interceptors != null) { if (this.interceptors != null) {
AdvisorAdapterRegistry adapterRegistry = GlobalAdvisorAdapterRegistry.getInstance(); AdvisorAdapterRegistry adapterRegistry = GlobalAdvisorAdapterRegistry.getInstance();
for (int i = 0; i < this.interceptors.length; i++) { for (Object interceptor : this.interceptors) {
proxyFactory.addAdvisor(adapterRegistry.wrap(this.interceptors[i])); proxyFactory.addAdvisor(adapterRegistry.wrap(interceptor));
} }
} }
proxyFactory.setTarget(getService()); proxyFactory.setTarget(getService());
proxyFactory.setOpaque(true); proxyFactory.setOpaque(true);
return proxyFactory.getProxy(getBeanClassLoader()); return proxyFactory.getProxy(getBeanClassLoader());
} }

View File

@ -59,9 +59,7 @@ public class JmsInvokerProxyFactoryBean extends JmsInvokerClientInterceptor
*/ */
public void setServiceInterface(Class<?> serviceInterface) { public void setServiceInterface(Class<?> serviceInterface) {
Assert.notNull(serviceInterface, "'serviceInterface' must not be null"); Assert.notNull(serviceInterface, "'serviceInterface' must not be null");
if (!serviceInterface.isInterface()) { Assert.isTrue(serviceInterface.isInterface(), "'serviceInterface' must be an interface");
throw new IllegalArgumentException("'serviceInterface' must be an interface");
}
this.serviceInterface = serviceInterface; this.serviceInterface = serviceInterface;
} }
@ -73,9 +71,7 @@ public class JmsInvokerProxyFactoryBean extends JmsInvokerClientInterceptor
@Override @Override
public void afterPropertiesSet() { public void afterPropertiesSet() {
super.afterPropertiesSet(); super.afterPropertiesSet();
if (this.serviceInterface == null) { Assert.notNull(this.serviceInterface, "Property 'serviceInterface' is required");
throw new IllegalArgumentException("Property 'serviceInterface' is required");
}
this.serviceProxy = new ProxyFactory(this.serviceInterface, this).getProxy(this.beanClassLoader); this.serviceProxy = new ProxyFactory(this.serviceInterface, this).getProxy(this.beanClassLoader);
} }

View File

@ -33,6 +33,7 @@ import org.springframework.remoting.rmi.CodebaseAwareObjectInputStream;
import org.springframework.remoting.support.RemoteInvocation; import org.springframework.remoting.support.RemoteInvocation;
import org.springframework.remoting.support.RemoteInvocationResult; import org.springframework.remoting.support.RemoteInvocationResult;
import org.springframework.util.Assert; import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
/** /**
* Abstract base implementation of the HttpInvokerRequestExecutor interface. * Abstract base implementation of the HttpInvokerRequestExecutor interface.
@ -291,7 +292,7 @@ public abstract class AbstractHttpInvokerRequestExecutor implements HttpInvokerR
Object obj = ois.readObject(); Object obj = ois.readObject();
if (!(obj instanceof RemoteInvocationResult)) { if (!(obj instanceof RemoteInvocationResult)) {
throw new RemoteException("Deserialized object needs to be assignable to type [" + throw new RemoteException("Deserialized object needs to be assignable to type [" +
RemoteInvocationResult.class.getName() + "]: " + obj); RemoteInvocationResult.class.getName() + "]: " + ClassUtils.getDescriptiveType(obj));
} }
return (RemoteInvocationResult) obj; return (RemoteInvocationResult) obj;
} }

View File

@ -212,10 +212,12 @@ public class HttpComponentsHttpInvokerRequestExecutor extends AbstractHttpInvoke
*/ */
protected HttpPost createHttpPost(HttpInvokerClientConfiguration config) throws IOException { protected HttpPost createHttpPost(HttpInvokerClientConfiguration config) throws IOException {
HttpPost httpPost = new HttpPost(config.getServiceUrl()); HttpPost httpPost = new HttpPost(config.getServiceUrl());
RequestConfig requestConfig = createRequestConfig(config); RequestConfig requestConfig = createRequestConfig(config);
if (requestConfig != null) { if (requestConfig != null) {
httpPost.setConfig(requestConfig); httpPost.setConfig(requestConfig);
} }
LocaleContext localeContext = LocaleContextHolder.getLocaleContext(); LocaleContext localeContext = LocaleContextHolder.getLocaleContext();
if (localeContext != null) { if (localeContext != null) {
Locale locale = localeContext.getLocale(); Locale locale = localeContext.getLocale();
@ -223,9 +225,11 @@ public class HttpComponentsHttpInvokerRequestExecutor extends AbstractHttpInvoke
httpPost.addHeader(HTTP_HEADER_ACCEPT_LANGUAGE, StringUtils.toLanguageTag(locale)); httpPost.addHeader(HTTP_HEADER_ACCEPT_LANGUAGE, StringUtils.toLanguageTag(locale));
} }
} }
if (isAcceptGzipEncoding()) { if (isAcceptGzipEncoding()) {
httpPost.addHeader(HTTP_HEADER_ACCEPT_ENCODING, ENCODING_GZIP); httpPost.addHeader(HTTP_HEADER_ACCEPT_ENCODING, ENCODING_GZIP);
} }
return httpPost; return httpPost;
} }
@ -253,6 +257,7 @@ public class HttpComponentsHttpInvokerRequestExecutor extends AbstractHttpInvoke
if (this.requestConfig == null) { // nothing to merge if (this.requestConfig == null) { // nothing to merge
return defaultRequestConfig; return defaultRequestConfig;
} }
RequestConfig.Builder builder = RequestConfig.copy(defaultRequestConfig); RequestConfig.Builder builder = RequestConfig.copy(defaultRequestConfig);
int connectTimeout = this.requestConfig.getConnectTimeout(); int connectTimeout = this.requestConfig.getConnectTimeout();
if (connectTimeout >= 0) { if (connectTimeout >= 0) {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2016 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -18,6 +18,7 @@ package org.springframework.remoting.httpinvoker;
import org.springframework.aop.framework.ProxyFactory; import org.springframework.aop.framework.ProxyFactory;
import org.springframework.beans.factory.FactoryBean; import org.springframework.beans.factory.FactoryBean;
import org.springframework.util.Assert;
/** /**
* {@link FactoryBean} for HTTP invoker proxies. Exposes the proxied service * {@link FactoryBean} for HTTP invoker proxies. Exposes the proxied service
@ -60,10 +61,9 @@ public class HttpInvokerProxyFactoryBean extends HttpInvokerClientInterceptor im
@Override @Override
public void afterPropertiesSet() { public void afterPropertiesSet() {
super.afterPropertiesSet(); super.afterPropertiesSet();
if (getServiceInterface() == null) { Class<?> ifc = getServiceInterface();
throw new IllegalArgumentException("Property 'serviceInterface' is required"); Assert.notNull(ifc, "Property 'serviceInterface' is required");
} this.serviceProxy = new ProxyFactory(ifc, this).getProxy(getBeanClassLoader());
this.serviceProxy = new ProxyFactory(getServiceInterface(), this).getProxy(getBeanClassLoader());
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2014 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -107,7 +107,8 @@ public class SimpleHttpInvokerRequestExecutor extends AbstractHttpInvokerRequest
protected HttpURLConnection openConnection(HttpInvokerClientConfiguration config) throws IOException { protected HttpURLConnection openConnection(HttpInvokerClientConfiguration config) throws IOException {
URLConnection con = new URL(config.getServiceUrl()).openConnection(); URLConnection con = new URL(config.getServiceUrl()).openConnection();
if (!(con instanceof HttpURLConnection)) { if (!(con instanceof HttpURLConnection)) {
throw new IOException("Service URL [" + config.getServiceUrl() + "] is not an HTTP URL"); throw new IOException(
"Service URL [" + config.getServiceUrl() + "] does not resolve to an HTTP connection");
} }
return (HttpURLConnection) con; return (HttpURLConnection) con;
} }
@ -130,6 +131,7 @@ public class SimpleHttpInvokerRequestExecutor extends AbstractHttpInvokerRequest
if (this.readTimeout >= 0) { if (this.readTimeout >= 0) {
connection.setReadTimeout(this.readTimeout); connection.setReadTimeout(this.readTimeout);
} }
connection.setDoOutput(true); connection.setDoOutput(true);
connection.setRequestMethod(HTTP_METHOD_POST); connection.setRequestMethod(HTTP_METHOD_POST);
connection.setRequestProperty(HTTP_HEADER_CONTENT_TYPE, getContentType()); connection.setRequestProperty(HTTP_HEADER_CONTENT_TYPE, getContentType());
@ -142,6 +144,7 @@ public class SimpleHttpInvokerRequestExecutor extends AbstractHttpInvokerRequest
connection.setRequestProperty(HTTP_HEADER_ACCEPT_LANGUAGE, StringUtils.toLanguageTag(locale)); connection.setRequestProperty(HTTP_HEADER_ACCEPT_LANGUAGE, StringUtils.toLanguageTag(locale));
} }
} }
if (isAcceptGzipEncoding()) { if (isAcceptGzipEncoding()) {
connection.setRequestProperty(HTTP_HEADER_ACCEPT_ENCODING, ENCODING_GZIP); connection.setRequestProperty(HTTP_HEADER_ACCEPT_ENCODING, ENCODING_GZIP);
} }

View File

@ -278,16 +278,14 @@ public class JaxWsPortClientInterceptor extends LocalJaxWsServiceFactory
* Set the interface of the service that this factory should create a proxy for. * Set the interface of the service that this factory should create a proxy for.
*/ */
public void setServiceInterface(Class<?> serviceInterface) { public void setServiceInterface(Class<?> serviceInterface) {
if (!serviceInterface.isInterface()) { Assert.notNull(serviceInterface, "'serviceInterface' must not be null");
throw new IllegalArgumentException("'serviceInterface' must be an interface"); Assert.isTrue(serviceInterface.isInterface(), "'serviceInterface' must be an interface");
}
this.serviceInterface = serviceInterface; this.serviceInterface = serviceInterface;
} }
/** /**
* Return the interface of the service that this factory should create a proxy for. * Return the interface of the service that this factory should create a proxy for.
*/ */
@Nullable
public Class<?> getServiceInterface() { public Class<?> getServiceInterface() {
return this.serviceInterface; return this.serviceInterface;
} }