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

org.jivesoftware.openfire.nio.ClientConnectionHandler Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2005-2008 Jive Software. All rights reserved.
 *
 * 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.jivesoftware.openfire.nio;

import org.apache.mina.core.session.IdleStatus;
import org.apache.mina.core.session.IoSession;
import org.jivesoftware.openfire.Connection;
import org.jivesoftware.openfire.XMPPServer;
import org.jivesoftware.openfire.handler.IQPingHandler;
import org.jivesoftware.openfire.net.ClientStanzaHandler;
import org.jivesoftware.openfire.net.StanzaHandler;
import org.jivesoftware.openfire.session.ConnectionSettings;
import org.jivesoftware.openfire.spi.ConnectionConfiguration;
import org.jivesoftware.util.JiveGlobals;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.xmpp.packet.IQ;
import org.xmpp.packet.IQ.Type;
import org.xmpp.packet.JID;

/**
 * ConnectionHandler that knows which subclass of {@link StanzaHandler} should
 * be created and how to build and configure a {@link NIOConnection}.
 *
 * @author Gaston Dombiak
 */
public class ClientConnectionHandler extends ConnectionHandler {

    private static final Logger Log = LoggerFactory.getLogger(ClientConnectionHandler.class);

    public ClientConnectionHandler(ConnectionConfiguration configuration) {
        super(configuration);
    }

    @Override
    NIOConnection createNIOConnection(IoSession session) {
        return new NIOConnection(session, new OfflinePacketDeliverer(), configuration );
    }

    @Override
    StanzaHandler createStanzaHandler(NIOConnection connection) {
        return new ClientStanzaHandler(XMPPServer.getInstance().getPacketRouter(), connection);
    }

    @Override
    int getMaxIdleTime() {
        return JiveGlobals.getIntProperty(ConnectionSettings.Client.IDLE_TIMEOUT, 6 * 60 * 1000) / 1000;
    }

    /**
     * In addition to the functionality provided by the parent class, this
     * method will send XMPP ping requests to the remote entity on every first
     * invocation of this method (which will occur after a period of half the
     * allowed connection idle time has passed, without any IO).
     * 
     * XMPP entities must respond with either an IQ result or an IQ error
     * (feature-unavailable) stanza upon receiving the XMPP ping stanza. Both
     * responses will be received by Openfire and will cause the connection idle
     * count to be reset.
     * 
     * Entities that do not respond to the IQ Ping stanzas can be considered
     * dead, and their connection will be closed by the parent class
     * implementation on the second invocation of this method.
     * 
     * Note that whitespace pings that are sent by XMPP entities will also cause
     * the connection idle count to be reset.
     * 
     * @see ConnectionHandler#sessionIdle(IoSession, IdleStatus)
     */
    @Override
    public void sessionIdle(IoSession session, IdleStatus status) throws Exception {
        super.sessionIdle(session, status);
        
        final boolean doPing = JiveGlobals.getBooleanProperty(ConnectionSettings.Client.KEEP_ALIVE_PING, true);
        if (doPing && session.getIdleCount(status) == 1) {
            final ClientStanzaHandler handler = (ClientStanzaHandler) session.getAttribute(HANDLER);
            final JID entity = handler.getAddress();
            
            if (entity != null) {
                // Ping the connection to see if it is alive.
                final IQ pingRequest = new IQ(Type.get);
                pingRequest.setChildElement("ping",
                        IQPingHandler.NAMESPACE);
                pingRequest.setFrom( XMPPServer.getInstance().getServerInfo().getXMPPDomain() );
                pingRequest.setTo(entity); 
                
                // Get the connection for this session
                final Connection connection = (Connection) session.getAttribute(CONNECTION);

                if (Log.isDebugEnabled()) {
                    Log.debug("ConnectionHandler: Pinging connection that has been idle: " + connection);
                }

                connection.deliver(pingRequest);
            }
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy