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

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

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

import org.jgroups.Address;
import org.jgroups.JChannel;
import org.jgroups.View;
import org.jgroups.blocks.cs.BaseServer;
import org.jgroups.blocks.cs.TcpServer;
import org.jgroups.jmx.JmxConfigurator;
import org.jgroups.protocols.raft.ELECTION;
import org.jgroups.protocols.raft.RAFT;
import org.jgroups.protocols.raft.Role;
import org.jgroups.raft.blocks.ReplicatedStateMachine;
import org.jgroups.stack.IpAddress;
import org.jgroups.util.ByteArrayDataInputStream;
import org.jgroups.util.Util;

import java.io.DataInput;
import java.net.InetAddress;
import java.nio.ByteBuffer;
import java.util.Objects;

/**
 * Demos {@link ReplicatedStateMachine}
 * @author Bela Ban
 * @since  0.1
 */
public class ReplicatedStateMachineDemo implements org.jgroups.blocks.cs.Receiver, RAFT.RoleChange {
    protected JChannel                              ch;
    protected ReplicatedStateMachine rsm;
    protected BaseServer                            server; // listens for commands from a remote client

    public enum Command {PUT, GET, REMOVE, SHOW_ALL, DUMP_LOG, SNAPSHOT, GET_VIEW}


    public void start(String props, String name, boolean follower, long timeout,
                      InetAddress bind_addr, int port, boolean listen, boolean nohup) throws Exception {
        ch=new JChannel(props).name(name);
        rsm=new ReplicatedStateMachine(ch).raftId(name).timeout(timeout);
        if(follower)
            disableElections(ch);
        ch.setReceiver(new org.jgroups.Receiver() {
            @Override public void viewAccepted(View view) {
                System.out.println("-- view change: " + view);
            }
        });

        ch.connect("rsm");
        Util.registerChannel(rsm.channel(), "rsm");
        rsm.addRoleChangeListener(this);
        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);
            }
        });
        if(listen)
            start(bind_addr, port);
        if(!nohup)
            loop();
    }

    @Override
    public void receive(Address sender, byte[] buf, int offset, int length) {
        ByteArrayDataInputStream in=new ByteArrayDataInputStream(buf, offset, length);
        try {
            receive(sender, in);
        }
        catch(Exception e) {
            e.printStackTrace();
        }
    }

    @Override
    public void receive(Address sender, ByteBuffer buf) {
        Util.bufferToArray(sender, buf, this);
    }

    @Override
    public void receive(Address sender, DataInput in) throws Exception {
        int ordinal=in.readByte();
        ReplicatedStateMachineDemo.Command cmd=Command.values()[ordinal];

        switch(cmd) {
            case PUT: // key (String) : value(String) -> Object
                String key=Util.objectFromStream(in), value=Util.objectFromStream(in);
                Object retval=put(key, value);
                sendResponse(sender, retval);
                break;
            case GET: // key (String) -> Object
                key=Util.objectFromStream(in);
                retval=get(key);
                sendResponse(sender, retval);
                break;

            case REMOVE:
                key=Util.objectFromStream(in);
                retval=remove(key);
                sendResponse(sender, retval);
                break;

            case SHOW_ALL:
                String result=rsm.toString();
                sendResponse(sender, result);
                break;

            case DUMP_LOG:
                result=dumpLog();
                sendResponse(sender, result);
                break;

            case SNAPSHOT:
                result=(String)snapshot();
                sendResponse(sender, result);
                break;

            case GET_VIEW:
                result=getView();
                sendResponse(sender, result);
                break;
        }

    }

    protected void start(InetAddress bind_addr, int port) throws Exception {
        server=new TcpServer(bind_addr, port).receiver(this);
        server.start();
        JmxConfigurator.register(server, Util.getMBeanServer(), "rsm:name=rsm");
        int local_port=server.localAddress() instanceof IpAddress? ((IpAddress)server.localAddress()).getPort(): 0;
        System.out.printf("\n-- %s listening at %s:%s\n\n", ReplicatedStateMachineDemo.class.getSimpleName(),
                          bind_addr != null? bind_addr : "0.0.0.0",  local_port);
    }


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

    protected 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':
                    System.out.println(dumpLog());
                    break;
                case '6':
                    snapshot();
                    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 Object put(String key, String value) {
        try {
            return rsm.put(Objects.requireNonNull(key, "key must be non-null)"),
                           Objects.requireNonNull(value, "value must be non-null"));
        }
        catch(Throwable t) {
            String ret=String.format("failed setting %s=%s: %s", key, value, t);
            System.err.println(ret);
            return ret;
        }
    }

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

    protected Object remove(String key) {
        try {
            return rsm.remove(key);
        }
        catch(Exception ex) {
            String err=String.format("failed removing %s: %s", key, ex);
            System.out.println(err);
            return err;
        }
    }

    protected Object snapshot() {
        try {
            rsm.snapshot();
            return "snapshot suceeded";
        }
        catch(Exception e) {
            return String.format("snapshot failed: %s", e);
        }
    }

    protected String getView() {
        return String.format("local address: %s\nview: %s", ch.getAddress(), ch.getView());
    }

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

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

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

    protected String dumpLog() {
        return String.format("\nindex (term): command\n---------------------\n%s\n", rsm.dumpLog());
    }

    protected void sendResponse(Address target, Object rsp) throws Exception {
        byte[] buf=Util.objectToByteBuffer(rsp);
        server.send(target, buf, 0, buf.length);
    }

    @Override
    public void roleChanged(Role role) {
        System.out.println("-- changed role to " + role);
    }

    public static void main(String[] args) throws Exception {
        String      props="raft.xml";
        String      name=null;
        boolean     follower=false, listen=false, nohup=false;
        long        timeout=3000;
        InetAddress bind_addr=null;
        int         port=2065;

        for(int i=0; i < args.length; i++) {
            if(args[i].equals("-props")) {
                props=args[++i];
                continue;
            }
            if(args[i].equals("-name")) {
                name=args[++i];
                continue;
            }
            if(args[i].equals("-follower")) {
                follower=true;
                continue;
            }
            if(args[i].equals("-listen")) {
                listen=true;
                continue;
            }
            if(args[i].equals("-nohup")) {
                nohup=true;
                continue;
            }
            if(args[i].equals("-timeout")) {
                timeout=Long.parseLong(args[++i]);
                continue;
            }
            if(args[i].equals("-bind_addr")) {
                bind_addr=InetAddress.getByName(args[++i]);
                continue;
            }
            if(args[i].equals("-port")) {
                port=Integer.parseInt(args[++i]);
                continue;
            }
            System.out.printf("\n%s [-props ] [-name ] [-follower] [-timeout timeout]\n" +
                                "                   [-bind_addr ] [-port ] [-nohup]\n\n",
                              ReplicatedStateMachineDemo.class.getSimpleName());
            return;
        }
        new ReplicatedStateMachineDemo().start(props, name, follower, timeout, bind_addr, port, listen, nohup);
    }


}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy