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

org.jgroups.raft.demos.ProgrammaticRSM Maven / Gradle / Ivy

There is a newer version: 1.0.14.Final
Show newest version
package org.jgroups.raft.demos;

import org.jgroups.JChannel;
import org.jgroups.Receiver;
import org.jgroups.View;
import org.jgroups.logging.LogFactory;
import org.jgroups.protocols.*;
import org.jgroups.protocols.dns.DNS_PING;
import org.jgroups.protocols.pbcast.GMS;
import org.jgroups.protocols.pbcast.NAKACK2;
import org.jgroups.protocols.pbcast.STABLE;
import org.jgroups.protocols.raft.*;
import org.jgroups.raft.blocks.ReplicatedStateMachine;
import org.jgroups.stack.*;
import org.jgroups.util.Util;

import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.util.*;
import java.util.stream.Stream;

/**
 * Demos {@link ReplicatedStateMachine}. Option -members defines the set of members (supposed to be static across
 * all nodes). Option -name needs to name one of the members.
 * @author Bela Ban
 * @since  1.0.0
 */
public class ProgrammaticRSM {
    protected static final JChannel                              ch;
    protected static final ReplicatedStateMachine rsm;
    protected static final NonReflectiveProbeHandler             h;


    static {
        LogFactory.useJdkLogger(true);
        // prevents setting default values: GraalVM doesn't accept creation of InetAddresses at build time (in the
        // image), so we have to set the default valiues at run time
        Configurator.skipSettingDefaultValues(true);

        boolean use_udp=Boolean.getBoolean("use.udp");

        try {
            ch=create(use_udp);
            Configurator.skipSettingDefaultValues(false);
            h=new NonReflectiveProbeHandler(ch).initialize(ch.getProtocolStack().getProtocols());
            ch.setReceiver(new Receiver() {
                @Override public void viewAccepted(View view) {
                    System.out.println("-- view change: " + view);
                }
            });
            rsm=new ReplicatedStateMachine<>(ch);
        }
        catch(Exception ex) {
            throw new RuntimeException(ex);
        }
    }


    public static void main(String[] args) throws Exception {
        String       name=null;
        boolean      follower=false;
        long         timeout=3000;
        String       bind_addr=null;
        int          bind_port=0;
        List members=null;

        for(int i=0; i < args.length; i++) {
            if(args[i].equals("-name")) {
                name=args[++i];
                continue;
            }
            if(args[i].equals("-follower")) {
                follower=true;
                continue;
            }
            if(args[i].equals("-timeout")) {
                timeout=Long.parseLong(args[++i]);
                continue;
            }
            if(args[i].equals("-bind_addr")) {
                bind_addr=args[++i];
                continue;
            }
            if(args[i].equals("-bind_port")) {
                bind_port=Integer.parseInt(args[++i]);
                continue;
            }
            if(args[i].equals("-members")) {
                members=Util.parseCommaDelimitedStrings(args[++i]);
                continue;
            }
            System.out.println("ReplicatedStateMachine -members members -name name" +
                               "                       [-follower] [-timeout timeout] -tcp true|false]\n" +
                               "                       [-bind_addr addr] [-bind_port port]\n" +
                               " Example: -members A,B,C,D -name C");
            return;
        }

        if(members == null || members.isEmpty())
            throw new IllegalArgumentException("-members must be set");
        if(name == null)
            throw new IllegalArgumentException("-name must be set");

        ch.setName(name);
        ProtocolStack stack=ch.getProtocolStack();
        TP transport=stack.getTransport();
        transport.registerProbeHandler(h);

        InetAddress ba=bind_addr == null? Util.getAddress("site_local", Util.getIpStackType())
          : InetAddress.getByName(bind_addr);
        InetAddress diag_addr=Util.getAddress("224.0.75.75", Util.getIpStackType()),
          mcast_addr=Util.getAddress("228.8.8.8", Util.getIpStackType()),
          mping_mcast=Util.getAddress("230.5.6.7", Util.getIpStackType());
        transport.setBindAddress(ba).setBindPort(bind_port).setDiagnosticsAddr(diag_addr);
        if(transport instanceof UDP)
            ((UDP)transport).setMulticastAddress(mcast_addr);

        Discovery discovery=stack.findProtocol(TCPPING.class);
        if(discovery != null)
            ((TCPPING)discovery).initialHosts(Collections.singletonList(new InetSocketAddress(ba, 7800)));
        discovery=stack.findProtocol(MPING.class);
        if(discovery != null)
            ((MPING)discovery).setMcastAddr(mping_mcast);

        RAFT raft=stack.findProtocol(RAFT.class);
        raft.members(members).raftId(name);

        rsm.raftId(name).timeout(timeout);
        if(follower)
            disableElections(ch);

        try {
            ch.connect("rsm");
            DiagnosticsHandler diag_handler=transport.getDiagnosticsHandler();
            if(diag_handler != null) {
                Set probe_handlers=diag_handler.getProbeHandlers();
                probe_handlers.removeIf(probe_handler -> {
                    String[] keys=probe_handler.supportedKeys();
                    return keys != null && Stream.of(keys).anyMatch(s -> s.startsWith("jmx"));
                });
            }
            transport.registerProbeHandler(h);

            rsm.addNotificationListener(new ReplicatedStateMachine.Notification<>() {
                @Override
                public void put(String key, Object val, Object old_val) {
                    System.out.printf("-- put(%s, %s) -> %s\n", key, val, old_val);
                }

                @Override
                public void remove(String key, Object old_val) {
                    System.out.printf("-- remove(%s) -> %s\n", key, old_val);
                }
            });

            rsm.addRoleChangeListener(role -> System.out.println("-- changed role to " + role));

            loop();
        }
        finally {
            Util.close(ch);
        }
    }

    protected static JChannel create(boolean udp) throws Exception {
        List prots=new ArrayList<>();
        TP transport=udp? new UDP().diagEnableUdp(true)
          : new TCP().diagEnableUdp(false).diagEnableTcp(true).setBindPort(7800);
        transport.setThreadPoolMaxThreads(200).setDiagnosticsEnabled(true);
        prots.add(transport);

        // discovery protocols
        if(udp)
            prots.add(new PING());
        else {
            prots.add(new MPING());
            if(System.getProperty("jgroups.dns.dns_query") != null)
                prots.add(new DNS_PING());
            prots.add(new TCPPING());
        }

        Protocol[] rest={
          new MERGE3().setMinInterval(10000).setMaxInterval(30000),
          new FD_SOCK(),
          new FD_ALL3().setTimeout(60000).setInterval(10000),
          new VERIFY_SUSPECT(),
          new NAKACK2(),
          new UNICAST3(),
          new STABLE(),
          new NO_DUPES(),
          new GMS().setJoinTimeout(2000),
          new UFC(),
          new MFC(),
          new FRAG4(),
          new ELECTION(),
          new RAFT(),
          new REDIRECT(),
          new CLIENT()
        };
        prots.addAll(Arrays.asList(rest));
        return new JChannel(prots);
    }

    protected static void disableElections(JChannel ch) {
        ELECTION election=ch.getProtocolStack().findProtocol(ELECTION.class);
        if(election != null)
            election.noElections(true);
    }

    protected static void loop() {
        boolean looping=true;
        while(looping) {
            int input=Util.keyPress("[1] add [2] get [3] remove [4] show all [5] dump log [6] snapshot [7] put N [x] exit\n" +
                                      "first-applied=" + firstApplied() +
                                      ", last-applied=" + rsm.lastApplied() +
                                      ", commit-index=" + rsm.commitIndex() +
                                      ", log size=" + Util.printBytes(logSize()) + "\n");
            switch(input) {
                case '1':
                    put(read("key"), read("value"));
                    break;
                case '2':
                    get(read("key"));
                    break;
                case '3':
                    remove(read("key"));
                    break;
                case '4':
                    System.out.println(rsm + "\n");
                    break;
                case '5':
                    dumpLog();
                    break;
                case '6':
                    try {
                        rsm.snapshot();
                    }
                    catch(Exception e) {
                        e.printStackTrace();
                    }
                    break;
                case '7':
                    try {
                        int num=Util.readIntFromStdin("num: ");
                        System.out.println();
                        String value="hello world #";
                        int print=num / 10;
                        long start=System.currentTimeMillis();
                        for(int i=1; i <= num; i++) {
                            put("key-" + i, value + i);
                            if(i > 0 && i % print == 0)
                                System.out.println("-- count=" + i);
                        }
                        long diff=System.currentTimeMillis() - start;
                        System.out.println("\n" + num + " puts took " + diff + " ms; " + (num / (diff / 1000.0)) + " ops /sec\n");
                    }
                    catch(Throwable ignored) {

                    }
                    break;
                case 'x':
                    looping=false;
                    break;
            }
        }
    }

    protected static void put(String key, String value) {
        if(key == null || value == null) {
            System.err.printf("Key (%s) or value (%s) is null\n",key,value);
            return;
        }
        try {
            rsm.put(key, value);
        }
        catch(Throwable t) {
            System.err.println("failed setting " + key + "=" + value + ": " + t);
        }
    }

    protected static void get(String key) {
        Object val=rsm.get(key);
        System.out.printf("-- get(%s) -> %s\n", key, val);
    }

    protected static void remove(String key) {
        try {
            rsm.remove(key);
        }
        catch(Exception ex) {
            System.err.println("failed removing " + key + ": " + ex);
        }
    }

    protected static String read(String name) {
        try {
            return Util.readStringFromStdin(name + ": ");
        }
        catch(Exception e) {
            return null;
        }
    }

    protected static int firstApplied() {
        RAFT raft=rsm.channel().getProtocolStack().findProtocol(RAFT.class);
        return raft.log().firstAppended();
    }

    protected static int logSize() {
        return rsm.logSize();
    }

    protected static void dumpLog() {
        System.out.printf("\nindex (term): command\n---------------------\n%s\n",
                          rsm.dumpLog());
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy