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

org.jboss.byteman.sample.helper.PeriodicHelper Maven / Gradle / Ivy

Go to download

The Byteman sample jar contains some example helper classes and auxiliary classes used by the] sample rule scripts contained in the distribution

There is a newer version: 4.0.24
Show newest version
/*
* JBoss, Home of Professional Open Source
* Copyright 2010 Red Hat and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This 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 software 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 software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*
* @authors Andrew Dinn
*/
package org.jboss.byteman.sample.helper;

import org.jboss.byteman.rule.Rule;
import org.jboss.byteman.rule.helper.Helper;

/**
 * A helper class which adds a background thread when the helper class is activated and removes the background
 * thread when it is deactivated. The thread loops, calling a specific trigger method, {@link #periodicTrigger}
 * and then waiting for a fixed time. Rule sets which employ the helper class can attach rules to the periodic
 * trigger method in order to perform operations which should happen at regular intervalsx. This is useful,
 * for example, for rule sets which collect statistical information. A periodically triggered rule can read and
 * then rezero a set of stats counters, allowing it to provide regular interval statistics.
 *
 * The basic way to use this class is to employ attach a rule to method {@link #periodicTrigger()}. The rule will
 * be triggered at 10 second intervals. If the period needs to be altered then a rule attached to method
 * {@link #getPeriod()} can be used to return an alternative value. {@link #getPeriod()} is called once when the
 * periodic trigger thread is created. If the wait interval needs to be recomputed before each wait then a rule
 * attached to method {@link #resetPeriod(long)} can be used to return the desired value. The input argument is
 * the value returned by the call to {@link #getPeriod()}.
 */

public class PeriodicHelper extends Helper
{
    /**
     * the default period which the helper will wait for between calls to periodicTrigger in milliseconds. this
     * can be redefined either by overriding defaultPeriod
     */
    public final static long DEFAULT_PERIOD = 10000L;

    public PeriodicHelper(Rule rule) {
        super(rule);
        shutDown = false;
    }

    /**
     * a method which is called at regular intervals by the periodic helper thread to trigger rule processing.
     * This can be redefined by attaching one or more rules to the method. It is also possible to override
     * this method in a subclass
     */

    protected void periodicTrigger()
    {
        // do nothing -- rules will override this behaviour
    }

    /**
     * a method which is called when the periodic helper thread is started to compute the interval in milliseconds
     * for which the thread should wait between calls to the trigger method which by default returns the default
     * interval of 10 seconds. This can either be overridden or redefined by attaching a rule to the method.
     * @return the wait period
     */

    protected long getPeriod()
    {
        return DEFAULT_PERIOD;
    }

    /**
     * a method which is called when the periodic helper thread is about to wait which by default returns the
     * input value. this can be overridden or redefined by attaching a rule to the method.
     * @param initialPeriod the initial wait time returned by getPeriod when the periodic thread was created
     * @return the wait period
     */

    protected long resetPeriod(long initialPeriod)
    {
        return initialPeriod;
    }

    /**
     * helper activation method which creates a periodic helper thread to perform periodic calls to the trigger
     * method. should only be called when synchronized on PeriodicHelper.class.
     */
    public static void activated()
    {
        if (theHelper == null) {
            theHelper = new PeriodicHelper(null);
            theHelper.start();
        }
    }

    /**
     * helper deactivation method which shuts down the periodic helper thread. will only be called when
     * synchronized on PeriodicHelper.class
     */
    public static void deactivated()
    {
        if (theHelper != null) {
            theHelper.shutdown();
            theHelper = null;
        }
    }

    /**
     * method called in activate to create and run the shutdown thread.  will only be called when synchronized
     * on PeriodicHelper.class
     */
    private void start()
    {
        theHelperThread = new PeriodicHelperThread();
        theHelperThread.start();
    }

    /**
     * method called in deactivate the helper thread. will only be called when synchronized on
     * PeriodicHelper.class
     */
    private void shutdown()
    {
        synchronized (this) {
            shutDown = true;
            this.notify();
        }
        try {
            theHelperThread.join();
        } catch (InterruptedException e) {
            // ignore -- should never happen
        }
        theHelperThread = null;
    }

    /**
     * method called by the periodic helper thread to wait between calls to the trigger method
     */

    private boolean doWait(long periodMilliSecs)
    {
        synchronized(this) {
            if (!shutDown) {
                try {
                    setTriggering(true);
                    periodMilliSecs = resetPeriod(periodMilliSecs);
                    setTriggering(false);
                    wait(periodMilliSecs);
                } catch (InterruptedException e) {
                    // ignore -- should never happen
                }
            }
            return !shutDown;
        }
    }

    /**
     * singleton instance holding the current periodic helper
     */

    private static PeriodicHelper theHelper = null;

    /**
     * handle on the current helper thread
     */

    private static PeriodicHelperThread theHelperThread = null;

    /**
     * flag which enforces shutdown
     */
    private boolean shutDown;

    private class PeriodicHelperThread extends Thread
    {
        private long periodMilliSecs;

        public PeriodicHelperThread()
        {
            super("Periodic Helper Thread");
        }

        public void run()
        {
            periodMilliSecs = getPeriod();
            setTriggering(false);
            while (doWait(periodMilliSecs)) {
                setTriggering(true);
                periodicTrigger();
                setTriggering(false);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy