Package org.springframework.remoting.caucho

Source Code of org.springframework.remoting.caucho.HessianExporter

/*
* Copyright 2002-2013 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.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.remoting.support.RemoteExporter;
import org.springframework.util.Assert;
import org.springframework.util.CommonsLogWriter;

/**
* 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://www.caucho.com/hessian">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
* @see SimpleHessianServiceExporter
*/
public class HessianExporter extends RemoteExporter implements InitializingBean {

  public static final String CONTENT_TYPE_HESSIAN = "application/x-hessian";


  private SerializerFactory serializerFactory = new SerializerFactory();

  private HessianRemoteResolver remoteResolver;

  private Log debugLogger;

  private HessianSkeleton skeleton;


  /**
   * 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(SerializerFactory serializerFactory) {
    this.serializerFactory = (serializerFactory != null ? serializerFactory : new SerializerFactory());
  }

  /**
   * 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".
   */
  public void setAllowNonSerializable(boolean allowNonSerializable) {
    this.serializerFactory.setAllowNonSerializable(allowNonSerializable);
  }

  /**
   * 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 com.caucho.hessian.client.HessianProxyFactory#setDebug
   */
  public void setDebug(boolean debug) {
    this.debugLogger = (debug ? logger : null);
  }


  @Override
  public void afterPropertiesSet() {
    prepare();
  }

  /**
   * Initialize this exporter.
   */
  public void prepare() {
    checkService();
    checkServiceInterface();
    this.skeleton = new HessianSkeleton(getProxyForService(), getServiceInterface());
  }


  /**
   * 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.
   * @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()) {
        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);
      }

      if (this.serializerFactory != null) {
        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);
    }
  }

}
TOP

Related Classes of org.springframework.remoting.caucho.HessianExporter

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.