/*
 * Copyright 2002-2018 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
 *
 *      https://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.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;

import com.caucho.hessian.io.AbstractHessianInput;
import com.caucho.hessian.io.AbstractHessianOutput;
import com.caucho.hessian.io.Hessian2Input;
import com.caucho.hessian.io.Hessian2Output;
import com.caucho.hessian.io.HessianDebugInputStream;
import com.caucho.hessian.io.HessianDebugOutputStream;
import com.caucho.hessian.io.HessianInput;
import com.caucho.hessian.io.HessianOutput;
import com.caucho.hessian.io.HessianRemoteResolver;
import com.caucho.hessian.io.SerializerFactory;
import com.caucho.hessian.server.HessianSkeleton;
import org.apache.commons.logging.Log;

import org.springframework.beans.factory.InitializingBean;
import org.springframework.lang.Nullable;
import org.springframework.remoting.support.RemoteExporter;
import org.springframework.util.Assert;
import org.springframework.util.CommonsLogWriter;

General stream-based protocol exporter for a Hessian endpoint.

Hessian is a slim, binary RPC protocol. For information on Hessian, see the Hessian website. Note: As of Spring 4.0, this exporter requires Hessian 4.0 or above.

Author:Juergen Hoeller
See Also:
Since:2.5.1
Deprecated:as of 5.3 (phasing out serialization-based remoting)
/** * General stream-based protocol exporter for a Hessian endpoint. * * <p>Hessian is a slim, binary RPC protocol. * For information on Hessian, see the * <a href="http://hessian.caucho.com">Hessian website</a>. * <b>Note: As of Spring 4.0, this exporter requires Hessian 4.0 or above.</b> * * @author Juergen Hoeller * @since 2.5.1 * @see #invoke(java.io.InputStream, java.io.OutputStream) * @see HessianServiceExporter * @deprecated as of 5.3 (phasing out serialization-based remoting) */
@Deprecated public class HessianExporter extends RemoteExporter implements InitializingBean {
The content type for hessian (application/x-hessian).
/** * The content type for hessian ({@code application/x-hessian}). */
public static final String CONTENT_TYPE_HESSIAN = "application/x-hessian"; private SerializerFactory serializerFactory = new SerializerFactory(); @Nullable private HessianRemoteResolver remoteResolver; @Nullable private Log debugLogger; @Nullable private HessianSkeleton skeleton;
Specify the Hessian SerializerFactory to use.

This will typically be passed in as an inner bean definition of type com.caucho.hessian.io.SerializerFactory, with custom bean property values applied.

/** * Specify the Hessian SerializerFactory to use. * <p>This will typically be passed in as an inner bean definition * of type {@code com.caucho.hessian.io.SerializerFactory}, * with custom bean property values applied. */
public void setSerializerFactory(@Nullable SerializerFactory serializerFactory) { this.serializerFactory = (serializerFactory != null ? serializerFactory : new SerializerFactory()); }
Set whether to send the Java collection type for each serialized collection. Default is "true".
/** * Set whether to send the Java collection type for each serialized * collection. Default is "true". */
public void setSendCollectionType(boolean sendCollectionType) { this.serializerFactory.setSendCollectionType(sendCollectionType); }
Set whether to allow non-serializable types as Hessian arguments and return values. Default is "true".
/** * Set whether to allow non-serializable types as Hessian arguments * and return values. Default is "true". */
public void setAllowNonSerializable(boolean allowNonSerializable) { this.serializerFactory.setAllowNonSerializable(allowNonSerializable); }
Specify a custom HessianRemoteResolver to use for resolving remote object references.
/** * Specify a custom HessianRemoteResolver to use for resolving remote * object references. */
public void setRemoteResolver(HessianRemoteResolver remoteResolver) { this.remoteResolver = remoteResolver; }
Set whether Hessian's debug mode should be enabled, logging to this exporter's Commons Logging log. Default is "false".
See Also:
  • setDebug.setDebug
/** * Set whether Hessian's debug mode should be enabled, logging to * this exporter's Commons Logging log. Default is "false". * @see com.caucho.hessian.client.HessianProxyFactory#setDebug */
public void setDebug(boolean debug) { this.debugLogger = (debug ? logger : null); } @Override public void afterPropertiesSet() { prepare(); }
Initialize this exporter.
/** * Initialize this exporter. */
public void prepare() { checkService(); checkServiceInterface(); this.skeleton = new HessianSkeleton(getProxyForService(), getServiceInterface()); }
Perform an invocation on the exported object.
Params:
  • inputStream – the request stream
  • outputStream – the response stream
Throws:
/** * Perform an invocation on the exported object. * @param inputStream the request stream * @param outputStream the response stream * @throws Throwable if invocation failed */
public void invoke(InputStream inputStream, OutputStream outputStream) throws Throwable { Assert.notNull(this.skeleton, "Hessian exporter has not been initialized"); doInvoke(this.skeleton, inputStream, outputStream); }
Actually invoke the skeleton with the given streams.
Params:
  • skeleton – the skeleton to invoke
  • inputStream – the request stream
  • outputStream – the response stream
Throws:
/** * Actually invoke the skeleton with the given streams. * @param skeleton the skeleton to invoke * @param inputStream the request stream * @param outputStream the response stream * @throws Throwable if invocation failed */
protected void doInvoke(HessianSkeleton skeleton, InputStream inputStream, OutputStream outputStream) throws Throwable { ClassLoader originalClassLoader = overrideThreadContextClassLoader(); try { InputStream isToUse = inputStream; OutputStream osToUse = outputStream; if (this.debugLogger != null && this.debugLogger.isDebugEnabled()) { try (PrintWriter debugWriter = new PrintWriter(new CommonsLogWriter(this.debugLogger))){ @SuppressWarnings("resource") HessianDebugInputStream dis = new HessianDebugInputStream(inputStream, debugWriter); @SuppressWarnings("resource") HessianDebugOutputStream dos = new HessianDebugOutputStream(outputStream, debugWriter); dis.startTop2(); dos.startTop2(); isToUse = dis; osToUse = dos; } } if (!isToUse.markSupported()) { isToUse = new BufferedInputStream(isToUse); isToUse.mark(1); } int code = isToUse.read(); int major; int minor; AbstractHessianInput in; AbstractHessianOutput out; if (code == 'H') { // Hessian 2.0 stream major = isToUse.read(); minor = isToUse.read(); if (major != 0x02) { throw new IOException("Version " + major + '.' + minor + " is not understood"); } in = new Hessian2Input(isToUse); out = new Hessian2Output(osToUse); in.readCall(); } else if (code == 'C') { // Hessian 2.0 call... for some reason not handled in HessianServlet! isToUse.reset(); in = new Hessian2Input(isToUse); out = new Hessian2Output(osToUse); in.readCall(); } else if (code == 'c') { // Hessian 1.0 call major = isToUse.read(); minor = isToUse.read(); in = new HessianInput(isToUse); if (major >= 2) { out = new Hessian2Output(osToUse); } else { out = new HessianOutput(osToUse); } } else { throw new IOException("Expected 'H'/'C' (Hessian 2.0) or 'c' (Hessian 1.0) in hessian input at " + code); } in.setSerializerFactory(this.serializerFactory); out.setSerializerFactory(this.serializerFactory); if (this.remoteResolver != null) { in.setRemoteResolver(this.remoteResolver); } try { skeleton.invoke(in, out); } finally { try { in.close(); isToUse.close(); } catch (IOException ex) { // ignore } try { out.close(); osToUse.close(); } catch (IOException ex) { // ignore } } } finally { resetThreadContextClassLoader(originalClassLoader); } } }