/*
 * Copyright (c) 2003, 2012, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code 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 General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package com.sun.jmx.remote.internal;

import java.io.IOException;
import java.io.InterruptedIOException;

import com.sun.jmx.remote.util.ClassLogger;
import com.sun.jmx.remote.util.EnvHelp;

public abstract class ClientCommunicatorAdmin {
    private static volatile long threadNo = 1;

    public ClientCommunicatorAdmin(long period) {
        this.period = period;

        if (period > 0) {
            checker = new Checker();

            Thread t = new Thread(null,
                                  checker,
                                  "JMX client heartbeat " +  (++threadNo),
                                  0,
                                  false);

            t.setDaemon(true);
            t.start();
        } else
            checker = null;
    }

    
Called by a client to inform of getting an IOException.
/** * Called by a client to inform of getting an IOException. */
public void gotIOException (IOException ioe) throws IOException { restart(ioe); }
Called by this class to check a client connection.
/** * Called by this class to check a client connection. */
protected abstract void checkConnection() throws IOException;
Tells a client to re-start again.
/** * Tells a client to re-start again. */
protected abstract void doStart() throws IOException;
Tells a client to stop because failing to call checkConnection.
/** * Tells a client to stop because failing to call checkConnection. */
protected abstract void doStop();
Terminates this object.
/** * Terminates this object. */
public void terminate() { synchronized(lock) { if (state == TERMINATED) { return; } state = TERMINATED; lock.notifyAll(); if (checker != null) checker.stop(); } } private void restart(IOException ioe) throws IOException { // check state synchronized(lock) { if (state == TERMINATED) { throw new IOException("The client has been closed."); } else if (state == FAILED) { // already failed to re-start by another thread throw ioe; } else if (state == RE_CONNECTING) { // restart process has been called by another thread // we need to wait while(state == RE_CONNECTING) { try { lock.wait(); } catch (InterruptedException ire) { // be asked to give up InterruptedIOException iioe = new InterruptedIOException(ire.toString()); EnvHelp.initCause(iioe, ire); throw iioe; } } if (state == TERMINATED) { throw new IOException("The client has been closed."); } else if (state != CONNECTED) { // restarted is failed by another thread throw ioe; } return; } else { state = RE_CONNECTING; lock.notifyAll(); } } // re-starting try { doStart(); synchronized(lock) { if (state == TERMINATED) { throw new IOException("The client has been closed."); } state = CONNECTED; lock.notifyAll(); } return; } catch (Exception e) { logger.warning("restart", "Failed to restart: " + e); logger.debug("restart",e); synchronized(lock) { if (state == TERMINATED) { throw new IOException("The client has been closed."); } state = FAILED; lock.notifyAll(); } try { doStop(); } catch (Exception eee) { // OK. // We know there is a problem. } terminate(); throw ioe; } } // -------------------------------------------------------------- // private varaibles // -------------------------------------------------------------- private class Checker implements Runnable { public void run() { myThread = Thread.currentThread(); while (state != TERMINATED && !myThread.isInterrupted()) { try { Thread.sleep(period); } catch (InterruptedException ire) { // OK. // We will check the state at the following steps } if (state == TERMINATED || myThread.isInterrupted()) { break; } try { checkConnection(); } catch (Exception e) { synchronized(lock) { if (state == TERMINATED || myThread.isInterrupted()) { break; } } e = (Exception)EnvHelp.getCause(e); if (e instanceof IOException && !(e instanceof InterruptedIOException)) { try { gotIOException((IOException)e); } catch (Exception ee) { logger.warning("Checker-run", "Failed to check connection: "+ e); logger.warning("Checker-run", "stopping"); logger.debug("Checker-run",e); break; } } else { logger.warning("Checker-run", "Failed to check the connection: " + e); logger.debug("Checker-run",e); // XXX stop checking? break; } } } if (logger.traceOn()) { logger.trace("Checker-run", "Finished."); } } private void stop() { if (myThread != null && myThread != Thread.currentThread()) { myThread.interrupt(); } } private Thread myThread; } // -------------------------------------------------------------- // private variables // -------------------------------------------------------------- private final Checker checker; private long period; // state private final static int CONNECTED = 0; private final static int RE_CONNECTING = 1; private final static int FAILED = 2; private final static int TERMINATED = 3; private int state = CONNECTED; private final int[] lock = new int[0]; private static final ClassLogger logger = new ClassLogger("javax.management.remote.misc", "ClientCommunicatorAdmin"); }