Package com.sun.sgs.impl.io

Source Code of com.sun.sgs.impl.io.SocketAcceptor

/*
* Copyright (c) 2007-2009, Sun Microsystems, Inc.
*
* 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 Sun Microsystems, Inc. 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.
*/

package com.sun.sgs.impl.io;

import java.io.IOException;
import java.net.SocketAddress;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;

import org.apache.mina.common.IoAcceptor;
import org.apache.mina.common.IoSession;

import com.sun.sgs.impl.sharedutil.LoggerWrapper;
import com.sun.sgs.io.ServerEndpoint;
import com.sun.sgs.io.AcceptorListener;
import com.sun.sgs.io.Acceptor;
import com.sun.sgs.io.ConnectionListener;

/**
* This is an implementation of an {@link Acceptor} that uses a MINA
* {@link IoAcceptor} to accept incoming connections.
* <p>
* Its constructor is package-private, so use
* {@link ServerEndpoint#createAcceptor} to create an instance.
* This implementation is thread-safe.
*/
class SocketAcceptor implements Acceptor<SocketAddress> {

    /** The logger for this class. */
    private static final LoggerWrapper logger =
        new LoggerWrapper(Logger.getLogger(SocketAcceptor.class.getName()));
   
    /** The associated MINA acceptor that handles the binding. */
    private final IoAcceptor acceptor;

    /** The endpoint on which to listen. */
    private final ServerSocketEndpoint endpoint;

    /** Whether this acceptor has been shutdown. */
    private volatile boolean shutdown = false;

    /**
     * Constructs a {@code SocketAcceptor} with the given MINA
     * {@link IoAcceptor}.
     *
     * @param endpoint the local address to which to listen
     * @param acceptor the MINA {@code IoAcceptor} to use for the underlying
     *        IO processing
     */
    SocketAcceptor(ServerSocketEndpoint endpoint, IoAcceptor acceptor) {
        this.endpoint = endpoint;
        this.acceptor = acceptor;
    }

    /**
     * {@inheritDoc}
     * <p>
     * This implementation ensures that only complete messages are
     * delivered on the connections that it accepts.
     */
    public void listen(AcceptorListener listener)
        throws IOException
    {
        synchronized (this) {
            checkShutdown();
            AcceptHandler acceptHandler = new AcceptHandler(listener);
            acceptor.bind(endpoint.getAddress(), acceptHandler);
        }
        logger.log(Level.FINE, "listening on {0}", getBoundEndpoint());
    }

    /**
     * {@inheritDoc}
     */
    public ServerSocketEndpoint getEndpoint() {
        return endpoint;
    }

    /**
     * {@inheritDoc}
     */
    public ServerSocketEndpoint getBoundEndpoint() {
        synchronized (this) {
            checkShutdown();

            Set<?> boundAddresses = acceptor.getManagedServiceAddresses();
            if (boundAddresses.size() != 1) {
                logger.log(Level.WARNING,
                           "Expected 1 bound address, got {0}",
                           boundAddresses.size());
            }
            SocketAddress sockAddr =
                (SocketAddress) boundAddresses.iterator().next();
            return new ServerSocketEndpoint(sockAddr,
                                      endpoint.getTransportType(),
                                      endpoint.getExecutor(),
                                      endpoint.getNumProcessors());
        }
    }

    /**
     * {@inheritDoc}
     */
    public void shutdown() {
        synchronized (this) {
            // TODO currently allow multiple calls to shutdown; should we
            // only allow one? -JM
            shutdown = true;
            acceptor.unbindAll();
        }
    }

    /**
     * Check whether this acceptor has been shutdown, throwing
     * IllegalStateException if it has.
     *
     * @throws IllegalStateException if this acceptor has been shutdown.
     */
    private void checkShutdown() {
        if (shutdown) {
            throw new IllegalStateException("Acceptor has been shutdown");
        }
    }
   
    /**
     * Internal adaptor class to handle events from the acceptor itself.
     */
    static final class AcceptHandler extends SocketConnectionListener {
       
        /** The AcceptorListener for our parent Acceptor. */
        private final AcceptorListener acceptorListener;
       
        /**
         * Constructs a new {@code AcceptHandler} with an
         * {@code AcceptorListener} that will be notified as new
         * connections arrive.
         *
         * @param listener the listener to be notified of incoming
         *        connections
         */
        public AcceptHandler(AcceptorListener listener) {
            this.acceptorListener = listener;
        }
       
        /**
         * As new MINA {@code IoSession}s come in, set up a
         * {@code SocketConnection} and notify the associated
         * {@code AcceptorListener}. A new {@code CompleteMessageFilter}
         * instance will be attached to the new connection.
         *
         * @param session the newly created {@code IoSession}
         */
        @Override
        public void sessionCreated(IoSession session) throws Exception {
            logger.log(Level.FINE, "accepted session {0}", session);
            CompleteMessageFilter filter = new CompleteMessageFilter();
            ConnectionListener connListener = acceptorListener.newConnection();
            SocketConnection connection =
                new SocketConnection(connListener, filter, session);
            session.setAttachment(connection);
        }
    }
}
TOP

Related Classes of com.sun.sgs.impl.io.SocketAcceptor

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.