Package org.apache.activemq.transport

Source Code of org.apache.activemq.transport.InactivityMonitor

/**
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements.  See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.activemq.transport;

import java.io.IOException;
import java.util.concurrent.atomic.AtomicBoolean;

import org.apache.activemq.command.KeepAliveInfo;
import org.apache.activemq.command.WireFormatInfo;
import org.apache.activemq.thread.Scheduler;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

/**
* Used to make sure that commands are arriving periodically from the peer of
* the transport.
*
* @version $Revision$
*/
public class InactivityMonitor extends TransportFilter {

    private static final Log LOG = LogFactory.getLog(InactivityMonitor.class);

    private WireFormatInfo localWireFormatInfo;
    private WireFormatInfo remoteWireFormatInfo;
    private final AtomicBoolean monitorStarted = new AtomicBoolean(false);

    private final AtomicBoolean commandSent = new AtomicBoolean(false);
    private final AtomicBoolean inSend = new AtomicBoolean(false);

    private final AtomicBoolean commandReceived = new AtomicBoolean(true);
    private final AtomicBoolean inReceive = new AtomicBoolean(false);

    private final Runnable readChecker = new Runnable() {
        public void run() {
            readCheck();
        }
    };

    private final Runnable writeChecker = new Runnable() {
        public void run() {
            writeCheck();
        }
    };

    public InactivityMonitor(Transport next) {
        super(next);
    }

    public void stop() throws Exception {
        stopMonitorThreads();
        next.stop();
    }

    final void writeCheck() {
            if (inSend.get()) {
            LOG.trace("A send is in progress");
            return;
        }

        if (!commandSent.get()) {
            LOG.trace("No message sent since last write check, sending a KeepAliveInfo");
            // TODO: use a thread pool for this..
            Thread thread = new Thread("ActiveMQ: Activity Generator: "+next.getRemoteAddress()) {
                public void run() {
                    try {
                        oneway(new KeepAliveInfo());
                    } catch (IOException e) {
                        onException(e);
                    }
                };
            };
            thread.setDaemon(true);
            thread.start();
        } else {
            LOG.trace("Message sent since last write check, resetting flag");
        }

        commandSent.set(false);
    }

    final void readCheck() {
        if (inReceive.get()) {
            LOG.trace("A receive is in progress");
            return;
        }

        if (!commandReceived.get()) {
            LOG.debug("No message received since last read check for " + toString() + "! Throwing InactivityIOException.");

            // TODO: use a thread pool for this..
            Thread thread = new Thread("ActiveMQ: Inactivity Handler: "+next.getRemoteAddress()) {
                public void run() {
                        onException(new InactivityIOException("Channel was inactive for too long: "+next.getRemoteAddress()));
                };
            };
            thread.setDaemon(true);
            thread.start();

        } else {
            LOG.trace("Message received since last read check, resetting flag: ");
        }
        commandReceived.set(false);
    }

    public void onCommand(Object command) {
        inReceive.set(true);
        try {
            if (command.getClass() == WireFormatInfo.class) {
                synchronized (this) {
                    IOException error=null;
                    remoteWireFormatInfo = (WireFormatInfo)command;
                    try {
                        startMonitorThreads();
                    } catch (IOException e) {
                        error = e;
                    }
                    if( error!=null ) {
                        onException(error);
                    }
                }
            }
            synchronized (readChecker) {
                transportListener.onCommand(command);
            }
        } finally {
            commandReceived.set(true);
            inReceive.set(false);
        }
    }

    public void oneway(Object o) throws IOException {
        // Disable inactivity monitoring while processing a command.
        inSend.set(true);
        try {
            if (o.getClass() == WireFormatInfo.class) {
                synchronized (this) {
                    localWireFormatInfo = (WireFormatInfo)o;
                    startMonitorThreads();
                }
            }
            synchronized (writeChecker) {
                next.oneway(o);
            }
        } finally {
            commandSent.set(true);
            inSend.set(false);
        }
    }

    public void onException(IOException error) {
        if (monitorStarted.get()) {
            stopMonitorThreads();
        }
        transportListener.onException(error);
    }

    private synchronized void startMonitorThreads() throws IOException {
        if (monitorStarted.get()) {
            return;
        }
        if (localWireFormatInfo == null) {
            return;
        }
        if (remoteWireFormatInfo == null) {
            return;
        }

        long l = Math.min(localWireFormatInfo.getMaxInactivityDuration(), remoteWireFormatInfo.getMaxInactivityDuration());
        if (l > 0) {
            monitorStarted.set(true);
            Scheduler.executePeriodically(writeChecker, l / 2);
            Scheduler.executePeriodically(readChecker, l);
        }
    }

    /**
     *
     */
    private synchronized void stopMonitorThreads() {
        if (monitorStarted.compareAndSet(true, false)) {
            Scheduler.cancel(readChecker);
            Scheduler.cancel(writeChecker);
        }
    }

}
TOP

Related Classes of org.apache.activemq.transport.InactivityMonitor

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.