src.openwfe.org.engine.impl.dispatch.SocketDispatcher Maven / Gradle / Ivy
/*
* Copyright (c) 2001-2006, John Mettraux, OpenWFE.org
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* . Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* . Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* . Neither the name of the "OpenWFE" nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
* $Id: SocketDispatcher.java 2718 2006-06-02 05:24:52Z jmettraux $
*/
//
// SocketDispatcher.java
//
// [email protected]
//
// generated with
// jtmpl 1.0.04 20.11.2001 John Mettraux ([email protected])
//
package openwfe.org.engine.impl.dispatch;
import openwfe.org.Utils;
import openwfe.org.MapUtils;
import openwfe.org.ReflectionUtils;
import openwfe.org.ServiceException;
import openwfe.org.ApplicationContext;
import openwfe.org.time.Time;
import openwfe.org.engine.listen.reply.WarningReply;
import openwfe.org.engine.listen.reply.ListenerReply;
import openwfe.org.engine.workitem.WorkItem;
import openwfe.org.engine.workitem.InFlowWorkItem;
import openwfe.org.engine.workitem.WorkItemCoder;
import openwfe.org.engine.workitem.CodingException;
import openwfe.org.engine.dispatch.DispatchingException;
import openwfe.org.engine.dispatch.FatalDispatchingException;
import openwfe.org.engine.dispatch.AbstractWorkItemDispatcher;
/**
* It dispatches workitems through sockets
*
* CVS Info :
*
$Author: jmettraux $
*
$Date: 2006-06-02 07:24:52 +0200 (Fri, 02 Jun 2006) $
*
$Id: SocketDispatcher.java 2718 2006-06-02 05:24:52Z jmettraux $
*
* @author [email protected]
*/
public class SocketDispatcher
extends AbstractWorkItemDispatcher
{
private final static org.apache.log4j.Logger log = org.apache.log4j.Logger
.getLogger(SocketDispatcher.class.getName());
//
// CONSTANTS (definitions)
/**
* 'host' towards which the workitem will be dispatched.
*/
public final static String P_HOST
= "host";
/**
* 'port' where something listening for the workitem.
*/
public final static String P_PORT
= "port";
private final static int DEFAULT_PORT = 7007;
/**
* 'socketTimeout', the classical socket connection parameter.
*/
public final static String P_SOCKET_TIMEOUT
= "socketTimeout";
/**
* 'socketFactory', one can specify a custom socket factory class name with
* this parameter.
*/
public final static String P_SOCKET_FACTORY
= "socketFactory";
/**
* 'retryCount', how many times should the dispatcher attempt to deliver
* the workitem before calling it quit ?
*/
public final static String P_RETRY_COUNT
= "retryCount";
/*
public final static String RETRY_FREQUENCY
= "retryFrequency";
*/
//
// FIELDS
private String targetHost = null;
private int targetPort = -1;
private int retryCount = 2;
//protected long retryFrequency = 300000; // 5 minutes
private int socketTimeout = 120000;
private javax.net.SocketFactory socketFactory = null;
//
// CONSTRUCTORS
public void init
(final String serviceName,
final ApplicationContext context,
final java.util.Map serviceParams)
throws
ServiceException
{
super.init(serviceName, context, serviceParams);
//
// host
this.targetHost = (String)serviceParams.get(P_HOST);
if (this.targetHost == null) this.targetHost = "127.0.0.1";
//
// port
this.targetPort =
MapUtils.getAsInt(serviceParams, P_PORT, DEFAULT_PORT);
//
// retryCount
this.retryCount =
MapUtils.getAsInt(serviceParams, P_RETRY_COUNT, 3);
/*
//
// retryFrequency
try
{
this.retryFrequency =
Long.parseLong((String)serviceParams.get(RETRY_FREQUENCY));
}
catch (Exception e)
{
}
*/
//
// socket timeout
this.socketTimeout =
MapUtils.getAsInt(serviceParams, P_SOCKET_TIMEOUT, 120000);
//
// socketFactory
String sFactoryClass = (String)serviceParams.get(P_SOCKET_FACTORY);
try
{
/*
Class factoryClass = Class.forName(sFactoryClass);
java.lang.reflect.Method getDefault = factoryClass
.getDeclaredMethod("getDefault", new Class[] {});
this.socketFactory = (javax.net.SocketFactory)getDefault
.invoke(null, new Object[] {});
*/
this.socketFactory =
(javax.net.SocketFactory)ReflectionUtils.invokeStatic
(Class.forName(sFactoryClass),
"getDefault",
new Class[] {},
new Object[] {});
}
catch (final Exception e)
{
if (sFactoryClass != null)
{
log.warn
("Failed to use custom SocketFactory '"+sFactoryClass+"'",
e);
}
this.socketFactory = javax.net.SocketFactory.getDefault();
if (log.isDebugEnabled())
{
log.debug
("Using default socket factory '"+
this.socketFactory.getClass().getName()+"'");
}
}
}
//
// METHODS
/**
* The bulk of the encoding work taken out of the dispatch method
*/
protected byte[] encodeWorkitem
(final WorkItemCoder coder, final WorkItem wi)
throws
DispatchingException
{
final long start = System.currentTimeMillis();
try
{
final java.io.ByteArrayOutputStream baos =
new java.io.ByteArrayOutputStream();
//
// set dispatch time
if (wi instanceof InFlowWorkItem)
{
((InFlowWorkItem)wi)
.setDispatchTime(Time.toIsoDate());
}
//
// encode workitem
final byte[] encodedWorkItem = (byte[])coder
.encode(wi, getContext(), getParams());
//
// advertise encoder used
final String ad =
getWorkItemCoderName() + " " + encodedWorkItem.length + "\n\n";
baos.write(ad.getBytes(Utils.getEncoding()));
//log.debug("advertise() coderName >"+getWorkItemCoderName()+"<");
//
// write workitem
baos.write(encodedWorkItem);
baos.flush();
if (log.isDebugEnabled())
{
log.debug
("encodeWorkitem() took "+
(System.currentTimeMillis() - start));
}
return baos.toByteArray();
}
catch (final Exception e)
{
throw new DispatchingException
("Failed to encode workitem", e);
}
}
/**
* Dispatches the workitem over a socket connection.
*/
public Object dispatch (final WorkItem wi)
throws DispatchingException
{
if (log.isDebugEnabled())
log.debug("dispatch() to "+this.targetHost+":"+this.targetPort);
final long start = System.currentTimeMillis();
ListenerReply reply = null;
final WorkItemCoder coder = instantiateEncoder();
byte[] encodedWorkitem = encodeWorkitem(coder, wi);
//Utils.dump("dispatch_", encodedWorkitem);
for (int i=0; i= this.retryCount-1)
{
// flow will get frozen
throw new DispatchingException
("last dispatch attempt failed", t);
}
Thread.currentThread().yield();
// yield priority to other threads...
/*
try
{
Thread.sleep(1 * 1000);
}
catch (final InterruptedException ie)
{
// ignore
}
*/
}
finally
{
try
{
os.flush();
}
catch (final Throwable t)
{
}
// separating them both.
try
{
socket.close();
// close connection
}
catch (final Throwable t)
{
}
}
}
if (log.isDebugEnabled())
log.debug("dispatch() took "+(System.currentTimeMillis() - start));
//
// handle reply
return handleReply(reply);
// this method from parent class will throw an exception
// if the reply is fatal
}
}