Package org.jgroups.tests

Source Code of org.jgroups.tests.UnicastContentionTest

package org.jgroups.tests;

import org.jgroups.*;
import org.jgroups.jmx.JmxConfigurator;
import org.jgroups.util.Util;

import java.util.concurrent.CountDownLatch;
import java.util.Vector;
import java.util.Map;
import java.text.NumberFormat;

/**
* Tests contention of locks in UNICAST, by concurrently sending and receiving unicast messages. The contention is
* in the 'connections' hashmap, and results in a lot of retransmissions. Run 2 instances with
* java org.jgroups.tests.UnicastContentionTest -props udp.xml -num_msgs 100 -num_threads 200
* and the UNICAST.num_xmits value will be high
* @author Bela Ban
*/
public class UnicastContentionTest {
    static final String GROUP="UnicastContentionTest-Cluster";

    int num_msgs=10000;
    int size=1000; // bytes
    int num_mbrs=2;
    int num_threads=1;
    int MOD=1000;

    private static NumberFormat f;


    static {
        f=NumberFormat.getNumberInstance();
        f.setGroupingUsed(false);
        f.setMaximumFractionDigits(2);
    }



    private void start(String props, int num_msgs, int size, int num_mbrs, int num_threads, boolean dump_stats) throws Exception {
        this.num_msgs=num_msgs;
        this.size=size;
        this.num_mbrs=num_mbrs;
        this.num_threads=num_threads;
        this.MOD=num_threads * num_msgs / 10;

        MySender[] senders=new MySender[num_threads];
        JChannel ch=new JChannel(props);
        JmxConfigurator.registerChannel(ch, Util.getMBeanServer(), "jgroups", GROUP, true);
        final CountDownLatch latch=new CountDownLatch(1);
        MyReceiver receiver=new MyReceiver(latch);
        ch.setReceiver(receiver);
        ch.connect(GROUP);

        System.out.println("Waiting for " + num_mbrs + " members");
        latch.await();
        View view=ch.getView();
        Address local_addr=ch.getAddress();
        Address dest=pickNextMember(view, local_addr);
        System.out.println("View is " + view + "\n" + num_threads + " threads are sending " + num_msgs + " messages (of " + size + " bytes) to " + dest);

        for(int i=0; i < senders.length; i++)
            senders[i]=new MySender(dest, ch);

        for(MySender sender: senders)
            sender.start();
        for(MySender sender: senders)
            sender.join();

        if(dump_stats) {
            Util.keyPress("enter to dump stats and close channel");
            System.out.println("stats:\n" + printStats(ch.dumpStats("NAKACK")) + "\n" + printStats(ch.dumpStats("FC")) +
                    "\n" + printStats(ch.dumpStats("UNICAST")));
        }
        else
            Util.sleep(2000);
        Util.close(ch);
    }

    @SuppressWarnings("unchecked")
    private static String printStats(Map<String,Object> map) {
        StringBuilder sb=new StringBuilder();
        for(Map.Entry<String,Object> entry: map.entrySet()) {
            sb.append(entry.getKey()).append("\n");
            Map<Object,Object> val=(Map)entry.getValue();
            for(Map.Entry<Object,Object> tmp: val.entrySet()) {
                sb.append(tmp.getKey()).append("=").append(tmp.getValue()).append("\n");
            }
            sb.append("\n");
        }
        return sb.toString();
    }

    private static Address pickNextMember(View view, Address local_addr) {
        Vector<Address> mbrs=view.getMembers();
        for(Address mbr: mbrs) {
            if(!mbr.equals(local_addr))
                return mbr;
        }
        return null;
    }


    private class MySender extends Thread {
        final byte[] buf=new byte[size];
        final Address dest;
        final JChannel ch;

        public MySender(Address dest, JChannel ch) {
            this.dest=dest;
            this.ch=ch;
        }

        public void run() {
            for(int i=0; i < num_msgs; i++) {
                Message msg=new Message(dest, null, buf);
                try {
                    ch.send(msg);
                }
                catch(Exception e) {
                    e.printStackTrace();
                }
            }
        }
    }

    private class MyReceiver extends ReceiverAdapter {
        private final CountDownLatch latch;
        private int msgs=0;
        private long start=0;
        private int expected_msgs=num_msgs * num_threads;

        public MyReceiver(CountDownLatch latch) {
            this.latch=latch;
        }

        /** We receive a message. Doesn't need to be reentrant because only 1 sender sends us unicast messages */
        public void receive(Message msg) {
            if(start == 0)
                start=System.currentTimeMillis();
            msgs++;
            if(msgs % MOD == 0)
                System.out.println("-- " + msgs + " received");
            if(msgs >= expected_msgs) {
                long time=System.currentTimeMillis() - start;
                double msgs_sec=msgs / (time / 1000.0);
                double throughput=msgs_sec * size;
                System.out.println(new StringBuilder("-- received ").append(msgs).append(" messages")
                        .append(" (" + time + " ms, " + f.format(msgs_sec) + " msgs/sec, " +
                        Util.printBytes(throughput) + "/sec)"));
            }
        }

        public void viewAccepted(View new_view) {
            if(new_view.size() >= num_mbrs)
                latch.countDown();
        }
    }


    public static void main(String[] args) throws Exception {
        int num_msgs=10000;
        int size=1000; //bytes
        int num_mbrs=2;
        int num_threads=1;
        String props=null;
        boolean dump_stats=false;

        for(int i=0; i < args.length; i++) {
            if(args[i].equals("-num_msgs")) {
                num_msgs=Integer.parseInt(args[++i]);
                continue;
            }
            if(args[i].equals("-size")) {
                size=Integer.parseInt(args[++i]);
                continue;
            }
            if(args[i].equals("-num_mbrs")) {
                num_mbrs=Integer.parseInt(args[++i]);
                continue;
            }
            if(args[i].equals("-num_threads")) {
                num_threads=Integer.parseInt(args[++i]);
                continue;
            }
            if(args[i].equals("-props")) {
                props=args[++i];
                continue;
            }
            if(args[i].equals("-dump_stats")) {
                dump_stats=true;
                continue;
            }
            help();
            return;
        }

        new UnicastContentionTest().start(props, num_msgs, size, num_mbrs, num_threads, dump_stats);
    }



    private static void help() {
        System.out.println("UnicastStressTest2 [-props properties] [-num_msgs <number of messages to send>]" +
                " [-size bytes] [-num_mbrs members]");
    }
}
TOP

Related Classes of org.jgroups.tests.UnicastContentionTest

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.