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

org.xsocket.connection.IoAcceptor 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.net.BindException;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.logging.Level;
import java.util.logging.Logger;

import javax.net.ssl.SSLContext;

import org.xsocket.DataConverter;




/**
 * The acceptor is responsible to accept new incoming connections, and
 * register these on the dispatcher.

* * @author [email protected] */ final class IoAcceptor { private static final Logger LOG = Logger.getLogger(IoAcceptor.class.getName()); private static final String SO_RCVBUF = IServer.SO_RCVBUF; private static final String SO_REUSEADDR = IServer.SO_REUSEADDR; @SuppressWarnings("unchecked") private static final Map SUPPORTED_OPTIONS = new HashMap(); static { SUPPORTED_OPTIONS.put(SO_RCVBUF, Integer.class); SUPPORTED_OPTIONS.put(SO_REUSEADDR, Boolean.class); } // io handler private IIoAcceptorCallback callback; // open flag private final AtomicBoolean isOpen = new AtomicBoolean(true); // Socket private final ServerSocketChannel serverChannel; // SSL private final boolean sslOn; private final SSLContext sslContext; // dispatcher pool private final IoSocketDispatcherPool dispatcherPool; // statistics private long acceptedConnections; private long lastRequestAccpetedRate = System.currentTimeMillis(); public IoAcceptor(IIoAcceptorCallback callback, InetSocketAddress address, int backlog, boolean isReuseAddress) throws IOException { this(callback, address, backlog, null, false, isReuseAddress); } public IoAcceptor(IIoAcceptorCallback callback, InetSocketAddress address, int backlog, SSLContext sslContext, boolean sslOn, boolean isReuseAddress) throws IOException { this.callback = callback; this.sslContext = sslContext; this.sslOn = sslOn; LOG.fine("try to bind server on " + address); // create a new server socket serverChannel = ServerSocketChannel.open(); assert (serverChannel != null); serverChannel.configureBlocking(true); serverChannel.socket().setSoTimeout(0); // accept method never times out serverChannel.socket().setReuseAddress(isReuseAddress); try { serverChannel.socket().bind(address, backlog); dispatcherPool = new IoSocketDispatcherPool("Srv" + getLocalPort(), IoProvider.getServerDispatcherInitialSize()); } catch (BindException be) { serverChannel.close(); LOG.warning("could not bind server to " + address + ". Reason: " + DataConverter.toString(be)); throw be; } } void setOption(String name, Object value) throws IOException { if (name.equals(SO_RCVBUF)) { serverChannel.socket().setReceiveBufferSize((Integer) value); } else if (name.equals(SO_REUSEADDR)) { serverChannel.socket().setReuseAddress((Boolean) value); } else { LOG.warning("option " + name + " is not supproted for " + this.getClass().getName()); } } boolean isSSLSupported() { return (sslContext != null); } boolean isSSLOn() { return sslOn; } /** * {@inheritDoc} */ public Object getOption(String name) throws IOException { if (name.equals(SO_RCVBUF)) { return serverChannel.socket().getReceiveBufferSize(); } else if (name.equals(SO_REUSEADDR)) { return serverChannel.socket().getReuseAddress(); } else { LOG.warning("option " + name + " is not supproted for " + this.getClass().getName()); return null; } } @SuppressWarnings("unchecked") public Map getOptions() { return Collections.unmodifiableMap(SUPPORTED_OPTIONS); } IoSocketDispatcherPool getDispatcherPool() { return dispatcherPool; } /** * {@inheritDoc} */ public InetAddress getLocalAddress() { return serverChannel.socket().getInetAddress(); } /** * {@inheritDoc} */ public int getLocalPort() { return serverChannel.socket().getLocalPort(); } /** * {@inheritDoc} */ public void listen() throws IOException { callback.onConnected(); accept(); } private void accept() { while (isOpen.get()) { try { // blocking accept call SocketChannel channel = serverChannel.accept(); // create IoSocketHandler IoSocketDispatcher dispatcher = dispatcherPool.nextDispatcher(); IoChainableHandler ioHandler = ConnectionUtils.getIoProvider().createIoHandler(false, dispatcher, channel, sslContext, sslOn); // notify call back callback.onConnectionAccepted(ioHandler); acceptedConnections++; } catch (Exception e) { // if acceptor is running (.close() causes that any // thread currently blocked in accept() will throw a SocketException) if (serverChannel.isOpen()) { LOG.warning("error occured while accepting connection: " + DataConverter.toString(e)); } } } } /** * {@inheritDoc} */ public void close() throws IOException { if (isOpen.get()) { isOpen.set(false); if (LOG.isLoggable(Level.FINE)) { LOG.fine("closing acceptor"); } try { // closes the server socket serverChannel.close(); } catch (Exception e) { // eat and log exception if (LOG.isLoggable(Level.FINE)) { LOG.fine("error occured by closing " + e.toString()); } } dispatcherPool.close(); callback.onDisconnected(); callback = null; // unset reference to server } } void setDispatcherSize(int size) { dispatcherPool.setDispatcherSize(size); } int getDispatcherSize() { return dispatcherPool.getDispatcherSize(); } boolean getReceiveBufferIsDirect() { return dispatcherPool.getReceiveBufferIsDirect(); } void setReceiveBufferIsDirect(boolean isDirect) { dispatcherPool.setReceiveBufferIsDirect(isDirect); } boolean isReceiveBufferPreallocationMode() { return dispatcherPool.isReceiveBufferPreallocationMode(); } void setReceiveBufferPreallocationMode(boolean mode) { dispatcherPool.setReceiveBufferPreallocationMode(mode); } void setReceiveBufferPreallocatedMinSize(Integer minSize) { dispatcherPool.setReceiveBufferPreallocatedMinSize(minSize); } Integer getReceiveBufferPreallocatedMinSize() { return dispatcherPool.getReceiveBufferPreallocatedMinSize(); } /** * get the size of the preallocation buffer, * for reading incoming data * * @return preallocation buffer size */ Integer getReceiveBufferPreallocationSize() { return dispatcherPool.getReceiveBufferPreallocationSize(); } /** * set the size of the preallocation buffer, * for reading incoming data * * @param size the preallocation buffer size */ void setReceiveBufferPreallocationSize(int size) { dispatcherPool.setReceiveBufferPreallocationSize(size); } double getAcceptedRateCountPerSec() { double rate = 0; long elapsed = System.currentTimeMillis() - lastRequestAccpetedRate; if (acceptedConnections == 0) { rate = 0; } else if (elapsed == 0) { rate = Integer.MAX_VALUE; } else { rate = (((double) (acceptedConnections * 1000)) / elapsed); } lastRequestAccpetedRate = System.currentTimeMillis(); acceptedConnections = 0; return rate; } long getSendRateBytesPerSec() { return dispatcherPool.getSendRateBytesPerSec(); } long getReceiveRateBytesPerSec() { return dispatcherPool.getReceiveRateBytesPerSec(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy