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

org.jgroups.protocols.DELAY Maven / Gradle / Ivy

package org.jgroups.protocols;

import org.jgroups.Event;
import org.jgroups.annotations.Property;
import org.jgroups.stack.Protocol;
import org.jgroups.util.MessageBatch;
import org.jgroups.util.Util;

import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;


/**
 * Delays incoming/outgoing messages by a random number of milliseconds (range between 0 and n
 * where n is determined by the user) and nanoseconds (constant amount).
 * 

* Incoming messages can be delayed independently from outgoing messages (or not delayed at all). *

* This protocol should be inserted directly above the transport protocol (e.g. UDP). * * @author Bela Ban * @author Sanne Grinovero * @author Matej Cimbora */ public class DELAY extends Protocol { private static final Random randomNumberGenerator = new Random(); @Property(description = "Upper bound of number of milliseconds to delay passing a message up the stack (exclusive)") protected int in_delay; @Property(description = "Upper bound number of milliseconds to delay passing a message down the stack (exclusive)") protected int out_delay; @Property(description = "Number of nanoseconds to delay passing a message up the stack") protected int in_delay_nanos; @Property(description = "Number of nanoseconds to delay passing a message down the stack") protected int out_delay_nanos; @Property(description = "Keep the delay constant. By default delay time randoms between 0 and upper bound") protected boolean constant_delay; protected DelayedMessageHandler delayed_message_handler; protected DelayQueue delayed_messages = new DelayQueue(); public int getInDelay() {return in_delay;} public void setInDelay(int in_delay) {this.in_delay=in_delay;} public int getOutDelay() {return out_delay;} public void setOutDelay(int out_delay) {this.out_delay=out_delay;} public int getInDelayNanos() {return in_delay_nanos;} public void setInDelayNanos(int in_delay_nanos) {this.in_delay_nanos=in_delay_nanos;} public int getOutDelayNanos() {return out_delay_nanos;} public void setOutDelayNanos(int out_delay_nanos) {this.out_delay_nanos=out_delay_nanos;} @Override public void init() throws Exception { super.init(); delayed_message_handler = new DelayedMessageHandler(); delayed_message_handler.start(); } @Override public void destroy() { super.destroy(); if (delayed_message_handler != null) Util.interruptAndWaitToDie(delayed_message_handler); } public Object down(final Event evt) { if (isMessage(evt)) { delayed_messages.add(new DelayedMessage(evt, System.nanoTime())); // don't wait for the result, return immediately return null; } else { return down_prot.down(evt); } } public Object up(final Event evt) { if (isMessage(evt)) sleep(in_delay, in_delay_nanos); return up_prot.up(evt); } public void up(final MessageBatch batch) { sleep(in_delay, in_delay_nanos); up_prot.up(batch); } private static boolean isMessage(final Event evt) { return evt.getType() == Event.MSG; } /** * Compute a random number between 0 and n */ private int computeDelay(final int n) { if (n <= 1) { return 0; } return constant_delay ? n : randomNumberGenerator.nextInt(n); } private void sleep(final int variable_milliseconds_delay, final int nano_delay) { final int millis = computeDelay(variable_milliseconds_delay); if (millis != 0 || nano_delay != 0) { Util.sleep(millis, nano_delay); } } private class DelayedMessage implements Delayed { private final Event event; private final long start; private final long delay_time; public DelayedMessage(Event event, long start) { this.event = event; this.start = start; this.delay_time = TimeUnit.NANOSECONDS.convert(computeDelay(out_delay), TimeUnit.MILLISECONDS) + out_delay_nanos; } @Override public long getDelay(TimeUnit unit) { return unit.convert(start + delay_time - System.nanoTime(), TimeUnit.NANOSECONDS); } @Override public int compareTo(Delayed o) { if (o == this) return 0; // return Long.compare(this.getDelay(TimeUnit.NANOSECONDS), o.getDelay(TimeUnit.NANOSECONDS)); // JDK 7 only long my_delay=this.getDelay(TimeUnit.NANOSECONDS), other_delay=o.getDelay(TimeUnit.NANOSECONDS); return (my_delay < other_delay) ? -1 : ((my_delay == other_delay) ? 0 : 1); } } private class DelayedMessageHandler extends Thread { private List buffer = new ArrayList(); @Override public void run() { for (;;) { try { delayed_messages.drainTo(buffer); for (DelayedMessage evt : buffer) down_prot.down(evt.event); buffer.clear(); } catch (Exception e) { // handling thread should never die } } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy