All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.xsocket.connection.HandlerChain Maven / Gradle / Ivy

/*
 * Copyright (c) xlightweb.org, 2006 - 2010. All rights reserved.
 *
 *  This library is free software; you can redistribute it and/or
 *  modify it under the terms of the GNU Lesser General Public
 *  License as published by the Free Software Foundation; either
 *  version 2.1 of the License, or (at your option) any later version.
 *
 *  This library is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 *  Lesser General Public License for more details.
 *
 *  You should have received a copy of the GNU Lesser General Public
 *  License along with this library; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 * Please refer to the LGPL license at: http://www.gnu.org/copyleft/lesser.txt
 * The latest copy of this software may be found on http://www.xsocket.org/
 */
package org.xsocket.connection;

import java.io.IOException;
import java.lang.ref.WeakReference;
import java.nio.BufferUnderflowException;
import java.util.ArrayList;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;

import org.xsocket.ILifeCycle;
import org.xsocket.MaxReadSizeExceededException;
import org.xsocket.Resource;




/**
 * Implements a handler chain. Each handler of the chain will be called (in the registering order),
 * until one handler signal by the return value true, that the event has been handled. In 
 * this case the remaining handlers will not be called.  

* * Nested chains is not supported yet * *
* E.g. *
 *   ...
 *   HandlerChain tcpBasedSpamfilter = new HandlerChain();
 *   tcpBasedSpamfilter.addLast(new BlackIPFilter()); 
 *   tcpBasedSpamfilter.addLast(new FirstConnectRefuseFilter());
 *   
 *   HandlerChain mainChain = new HandlerChain();
 *   mainChain.addLast(tcpBasedSpamfilter);
 *   mainChain.addLast(new SmtpProtocolHandler());
 *   
 *   IMultithreadedServer smtpServer = new MultithreadedServer(port, mainChain);
 *   StreamUtils.start(server);
 *   ...
 *   
 * 
* * * @author [email protected] */ public final class HandlerChain implements IHandler, IConnectHandler, IDataHandler, IDisconnectHandler, IConnectionTimeoutHandler, IIdleTimeoutHandler, ILifeCycle { private static final Logger LOG = Logger.getLogger(HandlerChain.class.getName()); @Resource private Server server = null; private final List> enclosingChains = new ArrayList>(); private final HandlerInfo handlerInfo = new HandlerInfo(); // handlers private final List handlers = new ArrayList(); private boolean isUnsynchronized = false; private final List lifeCycleChain = new ArrayList(); private boolean isOnConnectPathMultithreaded = false; private final List connectHandlerChain = new ArrayList(); private boolean isOnDataPathMultithreaded = false; private final List dataHandlerChain = new ArrayList(); private boolean isOnDisconnectPathMultithreaded = false; private final List disconnectHandlerChain = new ArrayList(); private boolean isOnIdleTimeoutPathMultithreaded = false; private final List idleTimeoutHandlerChain = new ArrayList(); private boolean isOnConnectionTimeoutPathMultithreaded = false; private final List connectionTimeoutHandlerChain = new ArrayList(); /* * the execution mode will NOT be determined by annotations. In contrast to other handler * this handler tpye is well known. To determine the execution mode the getHandlerInfo * will used */ /** * constructor * */ public HandlerChain() { } IHandlerInfo getHandlerInfo() { return handlerInfo; } /** * constructor * * @param handlers the initial handlers */ public HandlerChain(List handlers) { for (IHandler hdl : handlers) { addLast(hdl); } } public void onInit() { for (IHandler handler : handlers) { ConnectionUtils.injectServerField(server, handler); } for (ILifeCycle lifeCycle : lifeCycleChain) { lifeCycle.onInit(); } } public void onDestroy() throws IOException { for (ILifeCycle lifeCycle : lifeCycleChain) { lifeCycle.onDestroy(); } } /** * add a handler to the end og the chain * * @param handler the handler to add */ public void addLast(IHandler handler) { if (handler instanceof HandlerChain) { ((HandlerChain) handler).registerChildChain(this); } handlers.add(handler); computePath(); } private void registerChildChain(HandlerChain handlerChain) { enclosingChains.add(new WeakReference((handlerChain))); } private void computePath() { lifeCycleChain.clear(); connectHandlerChain.clear(); isOnConnectPathMultithreaded = false; dataHandlerChain.clear(); isOnDataPathMultithreaded = false; disconnectHandlerChain.clear(); isOnDisconnectPathMultithreaded = false; idleTimeoutHandlerChain.clear(); isOnIdleTimeoutPathMultithreaded = false; connectionTimeoutHandlerChain.clear(); isOnConnectionTimeoutPathMultithreaded = false; isUnsynchronized = true; for (IHandler handler : handlers) { IHandlerInfo handlerInfo = ConnectionUtils.getHandlerInfo(handler); isUnsynchronized = isUnsynchronized && handlerInfo.isUnsynchronized(); if (handlerInfo.isLifeCycle()) { lifeCycleChain.add((ILifeCycle) handler); } // nested chain? if (handler instanceof HandlerChain) { IHandlerInfo nestedInfo = ((HandlerChain) handler).getHandlerInfo(); isOnConnectPathMultithreaded = isOnConnectPathMultithreaded || nestedInfo.isConnectHandlerMultithreaded(); isOnDataPathMultithreaded = isOnDataPathMultithreaded || nestedInfo.isDataHandlerMultithreaded(); isOnDisconnectPathMultithreaded = isOnDisconnectPathMultithreaded || nestedInfo.isDisconnectHandlerMultithreaded(); isOnIdleTimeoutPathMultithreaded = isOnIdleTimeoutPathMultithreaded || nestedInfo.isIdleTimeoutHandlerMultithreaded(); isOnConnectionTimeoutPathMultithreaded = isOnConnectionTimeoutPathMultithreaded || nestedInfo.isConnectionTimeoutHandlerMultithreaded(); dataHandlerChain.add((IDataHandler) handler); disconnectHandlerChain.add((IDisconnectHandler) handler); idleTimeoutHandlerChain.add((IIdleTimeoutHandler) handler); connectionTimeoutHandlerChain.add((IConnectionTimeoutHandler) handler); // ... no } else { if (handlerInfo.isConnectHandler()) { connectHandlerChain.add((IConnectHandler) handler); isOnConnectPathMultithreaded = isOnConnectPathMultithreaded || handlerInfo.isConnectHandlerMultithreaded(); } if (handlerInfo.isDataHandler()) { dataHandlerChain.add((IDataHandler) handler); isOnDataPathMultithreaded = isOnDataPathMultithreaded || handlerInfo.isDataHandlerMultithreaded(); } if (handlerInfo.isDisconnectHandler()) { disconnectHandlerChain.add((IDisconnectHandler) handler); isOnDisconnectPathMultithreaded = isOnDisconnectPathMultithreaded || handlerInfo.isDisconnectHandlerMultithreaded(); } if (handlerInfo.isIdleTimeoutHandler()) { idleTimeoutHandlerChain.add((IIdleTimeoutHandler) handler); isOnIdleTimeoutPathMultithreaded = isOnIdleTimeoutPathMultithreaded || handlerInfo.isIdleTimeoutHandlerMultithreaded(); } if (handlerInfo.isConnectionTimeoutHandler()) { connectionTimeoutHandlerChain.add((IConnectionTimeoutHandler) handler); isOnConnectionTimeoutPathMultithreaded = isOnConnectionTimeoutPathMultithreaded || handlerInfo.isConnectionTimeoutHandlerMultithreaded(); } } } for (WeakReference handlerChainRef : enclosingChains) { HandlerChain handlerChain = handlerChainRef.get(); if (handlerChain != null) { handlerChain.computePath(); } } } /** * {@inheritDoc} */ public boolean onConnect(final INonBlockingConnection connection) throws IOException, BufferUnderflowException, MaxReadSizeExceededException { if (connectHandlerChain.isEmpty()) { return false; } for (IConnectHandler connectHandler : connectHandlerChain) { boolean result = connectHandler.onConnect(connection); if (result == true) { return true; } } return true; } /** * {@inheritDoc} */ public boolean onData(final INonBlockingConnection connection) throws IOException { if (dataHandlerChain.isEmpty()) { return false; } for (IDataHandler dataHandler : dataHandlerChain) { boolean result = dataHandler.onData(connection); if (result == true) { return true; } } return true; } /** * {@inheritDoc} */ public boolean onDisconnect(final INonBlockingConnection connection) throws IOException, BufferUnderflowException, MaxReadSizeExceededException { if (disconnectHandlerChain.isEmpty()) { return false; } for (IDisconnectHandler disconnectHandler : disconnectHandlerChain) { boolean result = disconnectHandler.onDisconnect(connection); if (result == true) { return true; } } return true; } /** * {@inheritDoc} */ public boolean onIdleTimeout(final INonBlockingConnection connection) throws IOException, BufferUnderflowException, MaxReadSizeExceededException { if (idleTimeoutHandlerChain.isEmpty()) { return false; } for (IIdleTimeoutHandler idleTimeoutHandler : idleTimeoutHandlerChain) { boolean result = idleTimeoutHandler.onIdleTimeout(connection); if (result == true) { return true; } } if (LOG.isLoggable(Level.FINE)) { LOG.fine("[" + connection.getId() + "] closing connection because idle timeout has been occured and timeout handler returns true)"); } connection.close(); return true; } /** * {@inheritDoc} */ public boolean onConnectionTimeout(final INonBlockingConnection connection) throws IOException, BufferUnderflowException, MaxReadSizeExceededException { if (connectionTimeoutHandlerChain.isEmpty()) { return false; } for (IConnectionTimeoutHandler connectionTimeoutHandler : connectionTimeoutHandlerChain) { boolean result = connectionTimeoutHandler.onConnectionTimeout(connection); if (result == true) { return true; } } if (LOG.isLoggable(Level.FINE)) { LOG.fine("[" + connection.getId() + "] closing connection because coonection timeout has been occured and timeout handler returns true)"); } connection.close(); return true; } private class HandlerInfo implements IHandlerInfo { public boolean isConnectionScoped() { return false; } public boolean isConnectExceptionHandler() { return false; } public boolean isConnectExceptionHandlerMultithreaded() { return false; } public boolean isConnectHandler() { return true; } public boolean isConnectHandlerMultithreaded() { return isOnConnectionTimeoutPathMultithreaded; } public boolean isConnectionTimeoutHandler() { return true; } public boolean isConnectionTimeoutHandlerMultithreaded() { return isOnConnectionTimeoutPathMultithreaded; } public boolean isDataHandler() { return true; } public boolean isDataHandlerMultithreaded() { return isOnDataPathMultithreaded; } public boolean isDisconnectHandler() { return true; } public boolean isDisconnectHandlerMultithreaded() { return isOnDisconnectPathMultithreaded; } public boolean isIdleTimeoutHandler() { return true; } public boolean isIdleTimeoutHandlerMultithreaded() { return isOnIdleTimeoutPathMultithreaded; } public boolean isLifeCycle() { return true; } public boolean isUnsynchronized() { return isUnsynchronized; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy