diff --git a/README.md b/README.md index a59b4b4..1ba598c 100644 --- a/README.md +++ b/README.md @@ -30,16 +30,18 @@ An in-depth explanation of the system is available as a technical report in the Apply these configuration steps for every machine - In the `SimulationMain`-class: - Give each machine its sets of node by changing the `LOCAL_NODES_NUMBER`, `TOTAL_NODES_NUMBER`, `NODES_FROM_NUMBER` values + - Specify the parameters for the transaction sending behaviour of each node using the fields `MAX_BLOCKS_PENDING`, `INITIAL_SENDING_DELAY`, `SENDING_WAIT_TIME` and `REQUIRED_COMMITS`. - Set `IS_MASTER` to `true` for the master machine and to `false` for all the others - If the current machine is the master, also specify the simulation time in seconds. - In the `Application`-class: - Set `TRACKER_SERVER_ADDRESS` and `TRACKER_SERVER_PORT` to point to the server location - + ### Running - Start the tracker server - In folder `tracker-server` run `npm start` - Start the master machine by calling the main method in `SimulationMain`. - Start the other machines the same way as the master. +- A live visualization of the network can be seen by opening `/demo` in a browser. ### Run Tests From the root folder run `mvn test`. diff --git a/pom.xml b/pom.xml index 7652ac5..4524d2e 100644 --- a/pom.xml +++ b/pom.xml @@ -145,7 +145,7 @@ false - nl.tudelft.blockchain.scaleoutdistributedledger.Main + nl.tudelft.blockchain.scaleoutdistributedledger.SimulationMain ${project.name} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/Application.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/Application.java index 574194a..6937ec6 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/Application.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/Application.java @@ -25,7 +25,7 @@ public class Application { @Getter private LocalStore localStore; private Thread executor; - private CancellableInfiniteRunnable transactionExecutable; + private CancellableInfiniteRunnable transactionExecutable; private final boolean isProduction; @Getter @@ -60,7 +60,6 @@ public void init(int nodePort, Block genesisBlock, Ed25519Key key, OwnNode ownNo // Setup local store localStore = new LocalStore(ownNode, this, genesisBlock, this.isProduction); - localStore.updateNodes(); localStore.initMainChain(); serverThread = new Thread(new SocketServer(nodePort, localStore)); @@ -133,7 +132,7 @@ public MainChain getMainChain() { */ public void finishTransactionSending() { int nodeID = localStore.getOwnNode().getId(); - transactionSender.stop(); + //transactionSender.stop(); try { transactionSender.waitUntilDone(); TrackerHelper.setRunning(nodeID, false); diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/CommunicationHelper.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/CommunicationHelper.java index 6cd0dde..de9be8c 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/CommunicationHelper.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/CommunicationHelper.java @@ -1,11 +1,12 @@ package nl.tudelft.blockchain.scaleoutdistributedledger; -import java.util.logging.Level; - import nl.tudelft.blockchain.scaleoutdistributedledger.model.Proof; import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; import nl.tudelft.blockchain.scaleoutdistributedledger.validation.ValidationException; +import java.io.IOException; +import java.util.logging.Level; + /** * Helper class for communication. */ @@ -20,6 +21,8 @@ private CommunicationHelper() { * @return true if the transaction was accepted, false otherwise */ public static boolean receiveTransaction(Proof proof, LocalStore localStore) { + Log.log(Level.FINE, "Received transaction: " + proof.getTransaction()); + if (proof.getTransaction().getReceiver().getId() != localStore.getOwnNode().getId()) { Log.log(Level.WARNING, "Received a transaction that isn't for us: " + proof.getTransaction()); return false; @@ -28,13 +31,18 @@ public static boolean receiveTransaction(Proof proof, LocalStore localStore) { try { localStore.getVerification().validateNewMessage(proof, localStore); } catch (ValidationException ex) { - Log.log(Level.WARNING, "Received an invalid transaction/proof.", ex); + Log.log(Level.WARNING, "Received an invalid transaction/proof " + proof.getTransaction() + ": " + ex.getMessage()); return false; } - - proof.applyUpdates(); Log.log(Level.INFO, "Received and validated transaction: " + proof.getTransaction()); + Log.log(Level.FINE, "Transaction " + proof.getTransaction() + " is valid, applying updates..."); + proof.applyUpdates(localStore); + try { + TrackerHelper.registerTransaction(proof); + } catch (IOException e) { + Log.log(Level.WARNING, "Transaction registration failed", e); + } if (proof.getTransaction().getAmount() > 0) { localStore.addUnspentTransaction(proof.getTransaction()); diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/LocalStore.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/LocalStore.java index 4a880a5..92ccaf1 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/LocalStore.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/LocalStore.java @@ -33,7 +33,6 @@ public class LocalStore { @Getter private final Verification verification = new Verification(); - @Getter private final Set unspent = new HashSet<>(); @Getter @@ -145,19 +144,30 @@ public Transaction getTransactionFromNode(int nodeId, int blockId, int transacti throw new IllegalStateException("Transaction with id " + transactionId + " in block " + blockId + " from node " + nodeId + " not found."); } + /** + * @return - a copy of the unspent transactions + */ + public Set getUnspent() { + synchronized (unspent) { + return new HashSet<>(unspent); + } + } + /** * Adds the given transaction as unspent. * @param transaction - the transaction to add */ public void addUnspentTransaction(Transaction transaction) { - if (!unspent.add(transaction)) return; - - if (ownNode.equals(transaction.getReceiver())) { - availableMoney += transaction.getAmount(); - } - - if (ownNode.equals(transaction.getSender())) { - availableMoney += transaction.getRemainder(); + synchronized (unspent) { + if (!unspent.add(transaction)) return; + + if (ownNode.equals(transaction.getReceiver())) { + availableMoney += transaction.getAmount(); + } + + if (ownNode.equals(transaction.getSender())) { + availableMoney += transaction.getRemainder(); + } } } @@ -165,14 +175,16 @@ public void addUnspentTransaction(Transaction transaction) { * @param toRemove - the unspent transactions to remove */ public void removeUnspentTransactions(Collection toRemove) { - for (Transaction transaction : toRemove) { - if (!unspent.remove(transaction)) continue; - - if (ownNode.equals(transaction.getReceiver())) { - availableMoney -= transaction.getAmount(); - } - if (ownNode.equals(transaction.getSender())) { - availableMoney -= transaction.getRemainder(); + synchronized (unspent) { + for (Transaction transaction : toRemove) { + if (!unspent.remove(transaction)) continue; + + if (ownNode.equals(transaction.getReceiver())) { + availableMoney -= transaction.getAmount(); + } + if (ownNode.equals(transaction.getSender())) { + availableMoney -= transaction.getRemainder(); + } } } } @@ -197,7 +209,7 @@ public void initMainChain() { private void normalizeGenesis() { for (Transaction transaction : ownNode.getChain().getGenesisBlock().getTransactions()) { Node receiver = getNode(transaction.getReceiver().getId()); - if (receiver != transaction.getReceiver()) { + if (receiver != null && receiver != transaction.getReceiver()) { transaction.setReceiver(receiver); } } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/Main.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/Main.java deleted file mode 100644 index f23d04d..0000000 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/Main.java +++ /dev/null @@ -1,61 +0,0 @@ -package nl.tudelft.blockchain.scaleoutdistributedledger; - -import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; -import nl.tudelft.blockchain.scaleoutdistributedledger.model.OwnNode; -import nl.tudelft.blockchain.scaleoutdistributedledger.sockets.SocketClient; -import nl.tudelft.blockchain.scaleoutdistributedledger.sockets.SocketServer; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.HashMap; - -/** - * Class to handle multiple applications. - */ -public final class Main { - - private Main() { - // prevents instantiation - throw new UnsupportedOperationException(); - } - - /** - * Main method, starting point of the application. - * @param args - command line arguments. - * @throws IOException - error while registering nodes. - */ - public static void main(String[] args) { - if (args[0].equalsIgnoreCase("s")) { - try { - SimulationMain.main(args); - } catch (Exception ex) { - ex.printStackTrace(); - } - return; - } - // Start a new node - // TODO: Make an example transaction? - Application app = new Application(true); - } - - /** - * Manual testing method for sockets. - */ - private static void testSockets() { - try { - Thread t = new Thread(new SocketServer(8007, new LocalStore(new OwnNode(0), null, null, false))); - t.start(); - - Node node = new Node(1, null, "localhost", 8007); - - SocketClient client = new SocketClient(); - client.sendMessage(node, new ArrayList()); - Thread.sleep(2500); - client.sendMessage(node, new ArrayList<>()); - Thread.sleep(7500); - client.sendMessage(node, new ArrayList<>()); - } catch (Exception e) { - e.printStackTrace(); - } - } -} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/ProofConstructor.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/ProofConstructor.java new file mode 100644 index 0000000..45a38c4 --- /dev/null +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/ProofConstructor.java @@ -0,0 +1,179 @@ +package nl.tudelft.blockchain.scaleoutdistributedledger; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import nl.tudelft.blockchain.scaleoutdistributedledger.model.Block; +import nl.tudelft.blockchain.scaleoutdistributedledger.model.MetaKnowledge; +import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; +import nl.tudelft.blockchain.scaleoutdistributedledger.model.Proof; +import nl.tudelft.blockchain.scaleoutdistributedledger.model.Transaction; + +/** + * Class for constructing proofs. + */ +public class ProofConstructor { + + private final Transaction mainTransaction; + private final Node receiver; + private final Node sender; + private final Map> toSend; + private final Proof proof; + + /* + * The Problem: + * We want to send block A. + * We have committed block B. + * With metaknowledge, we can translate this to a list of blocks that need to be sent. + * + * From all these blocks, we have to determine "what is the last block of each node that needs to be sent." + * This is "loop over the blocks, loop over the sources of transactions (recursively): + * + * Block A + * Block B + * Node sender + * Node receiver + * List ownBlocks = MetaKnowledge.determineBlocks(sender, B) + * processBlocks(sender, ownBlocks) + * Map> toSend + * Map> alreadyChecked + * + * processBlocks(Node owner, List blocks): + * //newlyAdded is the list of all the blocks that were added (not already present) (HAS TO BE ORDERED) + * List newlyAdded = toSend.addAll(owner, blocks) + * for (Block b : newlyAdded) { + * alreadyChecked.add(owner, b.getNumber()) + * for (Transaction t : b.getTransactions()) { + * processSources(t) + * } + * } + * + * processSources(Transaction t): + * for (Transaction s : t.getSource()) + * Node owner = s.getOwner(); + * //Skip all sources in genesis blocks, our own blocks or in receiver blocks + * if (owner == null || owner == sender || owner == receiver) continue; + * + * Block b = s.getBlock() + * //Skip all blocks that were already checked + * if (b.getNumber() in alreadyChecked.get(owner)) continue; + * + * Block bCom = the first committed block at or after b + * if (bCom.getNumber() in alreadyChecked.get(owner)) continue; + * + * List blocksOfSource = MetaKnowledge.determineBlocks(owner, bCom) + * if (blocksOfSource is empty) { + * alreadyChecked.addAll(owner, 0 up to (inclusive) bCom.getNumber()) + * continue; + * } + * + * processBlocks(owner, blocksOfSource) + */ + + /** + * @param mainTransaction - the transaction to construct the proof for + */ + public ProofConstructor(Transaction mainTransaction) { + this.mainTransaction = mainTransaction; + this.receiver = mainTransaction.getReceiver(); + this.sender = mainTransaction.getSender(); + this.proof = new Proof(mainTransaction); + this.toSend = proof.getChainUpdates(); + } + + /** + * @return - the constructed proof + */ + public synchronized Proof constructProof() { + //If the proof was already constructed, return it. + if (!toSend.isEmpty()) return proof; + + MetaKnowledge metaKnowledge = receiver.getMetaKnowledge(); + int mainBlockNr = mainTransaction.getBlockNumber().getAsInt(); + Block nextCommitted = sender.getChain().getBlocks().get(mainBlockNr).getNextCommittedBlock(); + List ownBlocks = metaKnowledge.getBlocksToSend(sender, nextCommitted.getNumber()); + + //Base case: no blocks to send + if (ownBlocks.isEmpty()) { + return proof; + } + + //Recursively process all the blocks + processBlocks(sender, ownBlocks); + return proof; + } + + /** + * Processes the given list of blocks belonging to the given owner. + * The given list is expected to be non-empty. + * @param owner - the owner of the blocks + * @param blocks - the blocks + */ + protected void processBlocks(Node owner, List blocks) { + List newlyAdded = addBlocksToSend(owner, blocks); + for (Block block : newlyAdded) { + for (Transaction transaction : block.getTransactions()) { + processSources(transaction); + } + } + } + + /** + * Processes the sources of the given transaction. + * @param transaction - the transaction to process + */ + protected void processSources(Transaction transaction) { + for (Transaction source : transaction.getSource()) { + Node owner = source.getSender(); + //Skip all sources in genesis blocks, our own blocks and in receiver blocks + if (owner == null || owner == this.sender || owner == this.receiver) continue; + + int blockNumber = source.getBlockNumber().getAsInt(); + + Block block = owner.getChain().getBlocks().get(blockNumber); + int nextCommittedBlockNr = block.getNextCommittedBlock().getNumber(); + + //Determine the blocks that we would need to send. + MetaKnowledge metaKnowledge = this.receiver.getMetaKnowledge(); + List blocksOfSource = metaKnowledge.getBlocksToSend(owner, nextCommittedBlockNr); + if (blocksOfSource.isEmpty()) continue; + + processBlocks(owner, blocksOfSource); + } + } + + /** + * Adds the given blocks belonging to the given owner to the toSend map. + * @param owner - the owner of the blocks + * @param toAdd - the blocks to add + * @return - all the blocks that were added (not already in the toSend map) + */ + protected List addBlocksToSend(Node owner, List toAdd) { + List current = toSend.computeIfAbsent(owner, n -> new ArrayList<>()); + if (current.isEmpty()) { + current.addAll(toAdd); + return toAdd; + } + + if (current.size() >= toAdd.size()) { + //Nothing new + return Collections.EMPTY_LIST; + } + + //Since the blocks we are adding have been selected through the meta knowledge, they will go from firstUnknown up to a certain block. + //E.g. [0, 1, 2] or [2, 3] + //Also, we skip all blocks that we have already checked, we know that the entire history is effectively linear and that we check from + //low to high block numbers. This means that the given list will contain all elements that we already have + some extra. + //So we can start at the index equal to what we already have. + + assert current.containsAll(toAdd); + assert current.size() < toAdd.size(); + + int startBlockNr = current.size(); + List added = toAdd.subList(startBlockNr, toAdd.size()); + current.addAll(added); + return added; + } +} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/SimulationMain.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/SimulationMain.java index 2e7a804..4c2e63e 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/SimulationMain.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/SimulationMain.java @@ -1,13 +1,12 @@ package nl.tudelft.blockchain.scaleoutdistributedledger; -import nl.tudelft.blockchain.scaleoutdistributedledger.message.StopTransactingMessage; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Block; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Ed25519Key; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; import nl.tudelft.blockchain.scaleoutdistributedledger.model.OwnNode; import nl.tudelft.blockchain.scaleoutdistributedledger.simulation.Simulation; import nl.tudelft.blockchain.scaleoutdistributedledger.simulation.tendermint.TendermintHelper; -import nl.tudelft.blockchain.scaleoutdistributedledger.simulation.transactionpattern.ITransactionPattern; +import nl.tudelft.blockchain.scaleoutdistributedledger.simulation.transactionpattern.RandomTransactionPattern; import nl.tudelft.blockchain.scaleoutdistributedledger.simulation.transactionpattern.UniformRandomTransactionPattern; import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Utils; @@ -25,21 +24,29 @@ * Main class for running a simulation. */ public final class SimulationMain { - private SimulationMain() {} - //SETTINGS //number of local nodes to generate - public static final int LOCAL_NODES_NUMBER = 4; + public static final int LOCAL_NODES_NUMBER = 6; //number of total nodes in the system - public static final int TOTAL_NODES_NUMBER = 4; + public static final int TOTAL_NODES_NUMBER = 6; //number from which our nodes are (e.g if we run nodes (2, 3, 4), then this should be 2 public static final int NODES_FROM_NUMBER = 0; //Whether this main is the master coordinator of the simulation //Note that the master should always be started first public static final boolean IS_MASTER = true; //The duration of the simulation in seconds. Only has an effect when IS_MASTER == true. - public static final int SIMULATION_DURATION = 60; + public static final int SIMULATION_DURATION = 600; + // Maximum number of blocks waiting to be sent (no new transaction created in the mean time + public static final int MAX_BLOCKS_PENDING = 50; + // The initial delay in milliseconds to wait before checking for the first time. + public static final long INITIAL_SENDING_DELAY = 5000; + // The time in milliseconds to wait before checking again. + public static final long SENDING_WAIT_TIME = 5000; + // The number of blocks (with the same or higher block number) that need to be committed before we send a certain block. + public static final int REQUIRED_COMMITS = 2; + private SimulationMain() {} + /** * @param args - the program arguments * @throws Exception - If an exception occurs. @@ -69,7 +76,7 @@ public static void main(String[] args) throws Exception { //update nodes from the tracker Map nodes = new HashMap<>(TOTAL_NODES_NUMBER); TrackerHelper.updateNodes(nodes, null); - final Block genesisBlock = TendermintHelper.generateGenesisBlock(1000, nodes); + final Block genesisBlock = TendermintHelper.generateGenesisBlock(1000000, nodes); //generate genesis.json for all local nodes TendermintHelper.generateGenesisFiles(new Date(), @@ -78,10 +85,11 @@ public static void main(String[] args) throws Exception { // --- PHASE 3: start the actual simulation --- - Simulation simulation = new Simulation(IS_MASTER); - ITransactionPattern itp = new UniformRandomTransactionPattern(10, 20, 1000, 2000, 1); - simulation.setTransactionPattern(itp); + + RandomTransactionPattern rtp = new UniformRandomTransactionPattern(10, 20, 100, 200, 10); +// rtp.setSeed(1); + simulation.setTransactionPattern(rtp); simulation.runNodesLocally(nodes, ownNodes, genesisBlock, nodeToKeyPair); // Wait for all nodes to have initialized @@ -97,12 +105,12 @@ public static void main(String[] args) throws Exception { if (IS_MASTER) { Thread.sleep(SIMULATION_DURATION * 1000); - simulation.broadcastMessage(new StopTransactingMessage()); } // Wait for all the other nodes to stop - waitForStop(); + simulation.stop(); + waitForStop(); simulation.stopLocalNodes(); simulation.cleanup(); diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TrackerHelper.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TrackerHelper.java index 06809bd..0ea3c1d 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TrackerHelper.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TrackerHelper.java @@ -1,7 +1,10 @@ package nl.tudelft.blockchain.scaleoutdistributedledger; +import nl.tudelft.blockchain.scaleoutdistributedledger.exceptions.NodeRegisterFailedException; +import nl.tudelft.blockchain.scaleoutdistributedledger.exceptions.TrackerException; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; import nl.tudelft.blockchain.scaleoutdistributedledger.model.OwnNode; +import nl.tudelft.blockchain.scaleoutdistributedledger.model.Proof; import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; import org.apache.commons.io.IOUtils; import org.apache.http.client.methods.HttpGet; @@ -52,7 +55,8 @@ public static boolean resetTrackerServer() throws IOException { * Registers this node with the given public key. * @param nodePort - the port of the node * @param publicKey - the publicKey of the new node - * @return - the registered node + * @param id - the id of the node + * @return - the registered node * @throws IOException - IOException while registering node * @throws NodeRegisterFailedException - Server side exception while registering node */ @@ -81,10 +85,11 @@ public static OwnNode registerNode(int nodePort, byte[] publicKey, int id) throw /** * Mark a node with the given id as initialized on the tracker. * @param id - the id of the node to mark + * @param running - if the node is running or not * @throws IOException - IOException while registering node - * @throws NodeRegisterFailedException - Server side exception while registering node + * @throws TrackerException - Server side exception while updating running status */ - public static void setRunning(int id, boolean running) throws IOException { + public static void setRunning(int id, boolean running) throws IOException, TrackerException { JSONObject json = new JSONObject(); json.put("id", id); json.put("running", running); @@ -99,8 +104,7 @@ public static void setRunning(int id, boolean running) throws IOException { return; } Log.log(Level.SEVERE, "Error while updating the running status of the node"); - //TODO: Create new excepton for this - throw new NodeRegisterFailedException(); + throw new TrackerException("Unable to update to running."); } } @@ -119,7 +123,7 @@ public static String getIP() { while (addrss.hasMoreElements()) { String addr = addrss.nextElement().getHostAddress(); if (addr.contains(":") || addr.startsWith("127.")) continue; // IPv6 or Local - return (addr); + return addr; } } } catch (SocketException e) { } // Intentionally empty catch block @@ -154,6 +158,7 @@ public static void updateNodes(Map nodes, OwnNode ownNode) throws } else { Node node = new Node(i, publicKey, address, port); + //TODO Check if we need this. if (ownNode != null) { node.getChain().setGenesisBlock(ownNode.getChain().getGenesisBlock()); } @@ -206,4 +211,35 @@ private static byte[] jsonArrayToByteArray(JSONArray json) { } return res; } + + /** + * Registers a transaction to the tracker server. + * @param proof - the proof used to send the transaction + * @return - whether the registration was successful + * @throws IOException - exception while registering + */ + public static boolean registerTransaction(Proof proof) throws IOException { + JSONObject json = new JSONObject(); + json.put("from", proof.getTransaction().getSender().getId()); + json.put("to", proof.getTransaction().getReceiver().getId()); + json.put("amount", proof.getTransaction().getAmount()); + json.put("remainder", proof.getTransaction().getRemainder()); + json.put("numberOfChains", proof.getChainUpdates().size()); + json.put("numberOfBlocks", proof.getNumberOfBlocks()); + + try (CloseableHttpClient client = HttpClientBuilder.create().build()) { + StringEntity requestEntity = new StringEntity(json.toString(), ContentType.APPLICATION_JSON); + HttpPost request = new HttpPost(String.format("http://%s:%d/register-transaction", + Application.TRACKER_SERVER_ADDRESS, Application.TRACKER_SERVER_PORT)); + request.setEntity(requestEntity); + JSONObject response = new JSONObject(IOUtils.toString(client.execute(request).getEntity().getContent())); + if (response.getBoolean("success")) { + Log.log(Level.FINE, "Successfully registered transaction to tracker server"); + return true; + } else { + Log.log(Level.WARNING, "Error while registering transaction " + proof.getTransaction()); + return false; + } + } + } } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreator.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreator.java index 271086d..e23db34 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreator.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreator.java @@ -7,8 +7,9 @@ import java.util.Iterator; import java.util.Map; import java.util.Set; +import java.util.TreeSet; -import nl.tudelft.blockchain.scaleoutdistributedledger.model.Chain; +import nl.tudelft.blockchain.scaleoutdistributedledger.exceptions.NotEnoughMoneyException; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Proof; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Transaction; @@ -28,7 +29,7 @@ public class TransactionCreator { private final Node sender; private final Node receiver; private final long amount; - private final BitSet known; + //private final BitSet known; private int currentBest = Integer.MAX_VALUE; private TransactionTuple currentBestTuple; @@ -44,26 +45,28 @@ public TransactionCreator(LocalStore localStore, Node receiver, long amount) { this.sender = localStore.getOwnNode(); this.receiver = receiver; this.amount = amount; - this.known = calculateKnowledge(); + //this.known = calculateKnowledge(); } - /** - * Calculates what the receiver already knows about. - * - * @return a bitset with the chains the receiver already knows about - */ - private BitSet calculateKnowledge() { - BitSet collected = receiver.getMetaKnowledge() - .keySet() - .stream() - .map(Node::getId) - .collect(() -> new BitSet(nodesCount), - (bs, i) -> bs.set(i), - (bs1, bs2) -> bs1.or(bs2) - ); - - return collected; - } +//TODO IMPORTANT Determine if the current transaction creation method is correct. +// /** +// * Calculates what the receiver already knows about. +// * +// * @return a bitset with the chains the receiver already knows about +// */ +// private BitSet calculateKnowledge() { +// synchronized (receiver.getMetaKnowledge()) { +// BitSet collected = receiver.getMetaKnowledge() +// .keySet() +// .stream() +// .collect(() -> new BitSet(nodesCount), +// (bs, i) -> bs.set(i), +// (bs1, bs2) -> bs1.or(bs2) +// ); +// +// return collected; +// } +// } /** * Creates a transaction. @@ -79,7 +82,7 @@ public Transaction createTransaction() { TransactionTuple sources = bestSources(); if (sources == null) throw new NotEnoughMoneyException(); - Set sourceSet = sources.getTransactions(); + TreeSet sourceSet = sources.getTransactions(); long remainder = sources.getAmount() - amount; //Mark sources as spent. @@ -228,19 +231,19 @@ private void cleanup(Collection tuples, int previousBest) { */ public BitSet chainsRequired(Transaction transaction) { //TODO Verify that this collection of chains is correct. - Set chains = new HashSet<>(); - Proof.appendChains(transaction, receiver, chains); - BitSet bitset = chains.stream() - .map(Chain::getOwner) + Map chains = new HashMap<>(); + + int nrOfNodes = localStore.getNodes().size(); + Proof.appendChains2(nrOfNodes, transaction, receiver, chains); + BitSet bitset = chains + .keySet() + .stream() .map(Node::getId) .collect(() -> new BitSet(nodesCount), (bs, i) -> bs.set(i), (bs1, bs2) -> bs1.or(bs2) ); - //Remove all chains that are already known - bitset.andNot(known); - return bitset; } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionSender.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionSender.java index 47786eb..55445ae 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionSender.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionSender.java @@ -1,11 +1,11 @@ package nl.tudelft.blockchain.scaleoutdistributedledger; -import java.util.ListIterator; +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; -import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicInteger; import java.util.logging.Level; import nl.tudelft.blockchain.scaleoutdistributedledger.message.ProofMessage; @@ -20,28 +20,13 @@ /** * Class which handles sending of transactions. */ -public class TransactionSender { - /** - * The initial delay in milliseconds to wait before checking for the first time. - */ - public static final long INITIAL_DELAY = 2000; - - /** - * The time in milliseconds to wait before checking again. - */ - public static final long WAIT_TIME = 5000; - - /** - * The number of blocks (with the same or higher block number) that need to be committed before - * we send a certain block. - */ - public static final int REQUIRED_COMMITS = 2; +public class TransactionSender implements Runnable { private final ScheduledExecutorService executor = Executors.newScheduledThreadPool(1); private final LocalStore localStore; private final SocketClient socketClient; - private final AtomicInteger taskCounter = new AtomicInteger(); - private boolean stopping; + private final Chain chain; + private int alreadySent; /** * Creates a new TransactionSender. @@ -50,68 +35,72 @@ public class TransactionSender { public TransactionSender(LocalStore localStore) { this.localStore = localStore; this.socketClient = new SocketClient(); + this.chain = localStore.getOwnNode().getChain(); + + this.executor.schedule(this, SimulationMain.INITIAL_SENDING_DELAY, TimeUnit.MILLISECONDS); } - /** - * Schedules the {@code toSend} block to be sent. - * The block will only be sent when at least {@link TransactionSender#REQUIRED_COMMITS} have - * been committed. - * @param toSend - the block to send - */ - public void scheduleBlockSending(Block toSend) { - final Chain chain = toSend.getOwner().getChain(); - Runnable runnable = new Runnable() { - //We need to start - private Block lastCheckedBlock = toSend.getPreviousBlock(); - private int committedBlocks; - - @Override - public void run() { - synchronized (chain) { - //We iterate over all the blocks since the last checked block and count how many are on the main chain. - //We have to check the same parts of the chain every time, since blocks can get committed after we checked them. - ListIterator lit = chain.getBlocks().listIterator(lastCheckedBlock.getNumber() - Block.GENESIS_BLOCK_NUMBER + 1); - while (lit.hasNext() && committedBlocks < REQUIRED_COMMITS) { - Block block = lit.next(); - if (block.isOnMainChain(localStore)) { - committedBlocks++; - lastCheckedBlock = block; - } - } - } - - //If we didn't find enough blocks, then we reschedule to check again later. - if (!canSend(committedBlocks)) { - schedule(this); - return; - } - - //There are enough blocks on the chain, so we can send the block. - sendBlock(toSend); - taskCounter.decrementAndGet(); - } - }; - taskCounter.incrementAndGet(); - executor.schedule(runnable, INITIAL_DELAY, TimeUnit.MILLISECONDS); + @Override + public void run() { + //Send all and reschedule + try { + sendAllBlocksThatCanBeSent(); + } catch (Exception ex) { + Log.log(Level.SEVERE, "Uncaught exception in transaction sender!"); + } finally { + executor.schedule(this, SimulationMain.SENDING_WAIT_TIME, TimeUnit.MILLISECONDS); + } } /** - * @param committedBlocks - the number of blocks that have been committed - * @return - if a block can be sent + * Sends all blocks that can be sent. */ - public boolean canSend(int committedBlocks) { - if (stopping && committedBlocks >= 1) { - return true; + public void sendAllBlocksThatCanBeSent() { + //TODO Add explanation in readme? + //[0, 1, 2, 3, 4, 5, 6, 7, 8, 9] + //Committed: [0, 2, 4, 6, 8] + //Last sent: 2 + //Committed found: [3 -> 4, 5 -> 6, 7 -> 8] = [4, 6, 8] + //We can send up to 6 (inclusive) + //3 commits found, 2 required + + int lastBlockNr = chain.getLastBlockNumber(); + + //Determine what blocks have been committed since the last batch we sent + List committed = new ArrayList<>(); + for (int index = alreadySent + 1; index <= lastBlockNr; index++) { + Block current = chain.getBlocks().get(index); + Block next = current.getNextCommittedBlock(); + if (next == null || !next.isOnMainChain(localStore)) break; + + committed.add(next.getNumber()); + index = next.getNumber(); } - return committedBlocks >= REQUIRED_COMMITS; + //Not enough commits + if (committed.size() < SimulationMain.REQUIRED_COMMITS) return; + + //Send all the blocks that we haven't sent up to the committed block (inclusive) + int lastToSend = committed.get(committed.size() - SimulationMain.REQUIRED_COMMITS); + for (int blockNr = alreadySent + 1; blockNr <= lastToSend; blockNr++) { + sendBlock(chain.getBlocks().get(blockNr)); + } } /** * @return - the number of blocks currently waiting to be sent */ public int blocksWaiting() { - return taskCounter.get(); + Block block = chain.getLastBlock(); + if (block == null) return 0; + + Block prev = block; + while (block != null && block.getTransactions().isEmpty()) { + prev = block; + block = block.getPreviousBlock(); + } + + return prev.getNumber() - alreadySent; } /** @@ -119,19 +108,11 @@ public int blocksWaiting() { * @throws InterruptedException - If we are interrupted while waiting. */ public void waitUntilDone() throws InterruptedException { - while (taskCounter.get() != 0) { + while (blocksWaiting() > 0) { Thread.sleep(1000L); } } - /** - * Indicates that we want to stop. This reduces the REQUIRED_COMMITS to 1, to ensure that all - * remaining blocks get flushed whenever that is possible. - */ - public void stop() { - stopping = true; - } - /** * Shuts down this Transaction sender as quickly as possible. * Pending blocks will not be sent. @@ -139,7 +120,6 @@ public void stop() { public void shutdownNow() { executor.shutdownNow(); socketClient.shutdown(); - taskCounter.set(0); } /** @@ -147,6 +127,7 @@ public void shutdownNow() { * @param block - the block */ private void sendBlock(Block block) { + alreadySent = block.getNumber(); for (Transaction transaction : block.getTransactions()) { try { sendTransaction(transaction); @@ -163,22 +144,27 @@ private void sendBlock(Block block) { * @return - if the sending succeeded * @throws InterruptedException - If the current thread was interrupted while sending. */ - private boolean sendTransaction(Transaction transaction) throws InterruptedException { + private boolean sendTransaction(Transaction transaction) throws InterruptedException, IOException { + Log.log(Level.FINE, "Node " + transaction.getSender().getId() + " starting sending transaction: " + transaction); + long startingTime = System.currentTimeMillis(); Node to = transaction.getReceiver(); - Proof proof = Proof.createProof(localStore, transaction); - if (socketClient.sendMessage(to, new ProofMessage(proof))) { + + //TODO IMPORTANT Removed synchronization + ProofConstructor proofConstructor = new ProofConstructor(transaction); + Proof proof = proofConstructor.constructProof(); + ProofMessage msg = new ProofMessage(proof); + + long timeDelta = System.currentTimeMillis() - startingTime; + if (timeDelta > 5 * 1000) { + Log.log(Level.WARNING, "Proof creation took " + timeDelta + " ms for transaction: " + transaction); + } + Log.log(Level.FINE, "Node " + transaction.getSender().getId() + " now actually sending transaction: " + transaction); + if (socketClient.sendMessage(to, msg)) { to.updateMetaKnowledge(proof); + Log.log(Level.FINE, "Node " + transaction.getSender().getId() + " done sending transaction: " + transaction); return true; } return false; } - - /** - * @param runnable - the runnable to schedule - * @return - the ScheduledFuture - */ - private ScheduledFuture schedule(Runnable runnable) { - return executor.schedule(runnable, WAIT_TIME, TimeUnit.MILLISECONDS); - } } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionTuple.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionTuple.java index 09c563c..3adc20d 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionTuple.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionTuple.java @@ -1,8 +1,7 @@ package nl.tudelft.blockchain.scaleoutdistributedledger; import java.util.BitSet; -import java.util.HashSet; -import java.util.Set; +import java.util.TreeSet; import java.util.stream.Collectors; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; @@ -21,7 +20,7 @@ public class TransactionTuple { private int amount; @Getter - private Set transactions = new HashSet<>(); + private TreeSet transactions = new TreeSet<>(); @Getter @Setter private BitSet chainsRequired; diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/NodeRegisterFailedException.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/NodeRegisterFailedException.java similarity index 58% rename from src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/NodeRegisterFailedException.java rename to src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/NodeRegisterFailedException.java index eb65e29..8384fb0 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/NodeRegisterFailedException.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/NodeRegisterFailedException.java @@ -1,9 +1,9 @@ -package nl.tudelft.blockchain.scaleoutdistributedledger; +package nl.tudelft.blockchain.scaleoutdistributedledger.exceptions; /** - * Exception for indicating that the user doesn't have enough money. + * Exception for indicating that registering with the tracker failed. */ -public class NodeRegisterFailedException extends RuntimeException { +public class NodeRegisterFailedException extends TrackerException { private static final long serialVersionUID = 8271135988867023425L; /** diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/NotEnoughMoneyException.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/NotEnoughMoneyException.java similarity index 85% rename from src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/NotEnoughMoneyException.java rename to src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/NotEnoughMoneyException.java index bc06a7e..33e046e 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/NotEnoughMoneyException.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/NotEnoughMoneyException.java @@ -1,4 +1,4 @@ -package nl.tudelft.blockchain.scaleoutdistributedledger; +package nl.tudelft.blockchain.scaleoutdistributedledger.exceptions; /** * Exception for indicating that the user doesn't have enough money. diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/TrackerException.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/TrackerException.java new file mode 100644 index 0000000..55ef450 --- /dev/null +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/exceptions/TrackerException.java @@ -0,0 +1,22 @@ +package nl.tudelft.blockchain.scaleoutdistributedledger.exceptions; + +/** + * Exception for the tracker. + */ +public class TrackerException extends RuntimeException { + private static final long serialVersionUID = -3346246279993022763L; + + /** + * Exception without message. + */ + public TrackerException() { + super(); + } + + /** + * @param msg - the message + */ + public TrackerException(String msg) { + super(msg); + } +} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/BlockMessage.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/BlockMessage.java index 320d0d5..32ae5e4 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/BlockMessage.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/BlockMessage.java @@ -1,6 +1,7 @@ package nl.tudelft.blockchain.scaleoutdistributedledger.message; import lombok.Getter; + import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Block; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Sha256Hash; @@ -8,6 +9,7 @@ import java.util.ArrayList; import java.util.List; + import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; /** @@ -67,4 +69,25 @@ public BlockMessage(Block block, Node proofReceiver) { public void handle(LocalStore localStore) { // Do nothing. } + + public Block toBlockWithoutSources(LocalStore localStore) { + List transactions = new ArrayList<>(); + for(TransactionMessage tm : this.transactions) { + transactions.add(tm.toTransactionWithoutSources(localStore)); + } + return new Block(this.number, localStore.getNode(this.ownerId), transactions); + } + + @Override + public String toString() { + StringBuilder sb = new StringBuilder(64); + sb.append("BlockMessage(); - for (Map.Entry> entry : proof.getChainUpdates().entrySet()) { + for (Entry> entry : proof.getChainUpdates().entrySet()) { Node node = entry.getKey(); List blockList = entry.getValue(); if (!blockList.isEmpty()) { @@ -55,10 +58,30 @@ public ProofMessage(Proof proof) { @Override public void handle(LocalStore localStore) { + //TODO IMPORTANT Removed synchronized on own chain try { CommunicationHelper.receiveTransaction(new Proof(this, localStore), localStore); } catch (IOException e) { Log.log(Level.SEVERE, "Exception while handling proof message", e); } } + + @Override + public String toString() { + StringBuilder sb = new StringBuilder(64); + sb.append("ProofMessage\n Transaction = ").append(transactionMessage).append("\n{"); + if (chainUpdates.isEmpty()) { + return sb.append("}").toString(); + } + + for (Entry> entry : chainUpdates.entrySet()) { + sb.append("\n ").append(entry.getKey()).append(": ["); + for (BlockMessage bm : entry.getValue()) { + sb.append("\n ").append(bm); + } + sb.append("\n ]"); + } + sb.append("\n}"); + return sb.toString(); + } } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/TransactionMessage.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/TransactionMessage.java index e368b9a..d3b2dde 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/TransactionMessage.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/message/TransactionMessage.java @@ -1,22 +1,24 @@ package nl.tudelft.blockchain.scaleoutdistributedledger.message; -import lombok.Getter; +import java.io.Serializable; +import java.util.HashSet; +import java.util.Objects; +import java.util.Set; +import java.util.TreeSet; + import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Sha256Hash; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Transaction; -import java.util.AbstractMap.SimpleEntry; -import java.util.HashSet; -import java.util.Map.Entry; -import java.util.Objects; -import java.util.Set; +import lombok.Getter; /** * Transaction message for netty. */ public class TransactionMessage extends Message { - + private static final long serialVersionUID = 1L; + @Getter private final int number; @@ -26,19 +28,8 @@ public class TransactionMessage extends Message { @Getter private final long amount, remainder; - /** - * Transactions known by the receiver. - * Entry: node id, transaction number - */ - @Getter - private final Set> knownSource; - - /** - * Transactions new to the receiver. - * Entry: node id, transaction number - */ @Getter - private final Set> newSource; + private final Set source; @Getter private final Sha256Hash hash; @@ -65,36 +56,43 @@ public TransactionMessage(Transaction transaction, Node proofReceiver) { this.receiverId = transaction.getReceiver().getId(); this.amount = transaction.getAmount(); this.remainder = transaction.getRemainder(); - this.knownSource = new HashSet<>(); - this.newSource = new HashSet<>(); + this.source = new HashSet<>(); // Optimization: categorize each transaction already known (or not) by the receiver for (Transaction sourceTransaction : transaction.getSource()) { Node sourceSender = sourceTransaction.getSender(); - if (sourceSender == null) { - // Receiver already knows about a genesis transaction - this.knownSource.add(new SimpleEntry<>(Transaction.GENESIS_SENDER, sourceTransaction.getNumber())); - } else { - if (proofReceiver.equals(sourceSender)) { - // Receiver is the sender of the source - // So receiver knows about himself - this.knownSource.add(new SimpleEntry<>(sourceSender.getId(), sourceTransaction.getNumber())); + if (sourceTransaction.getBlockNumber().isPresent()) { + if (sourceSender == null) { + // Genesis transaction + this.source.add(new TransactionSource( + sourceTransaction.getReceiver().getId(), + sourceTransaction.getBlockNumber().getAsInt(), + sourceTransaction.getNumber())); } else { - // Receiver is not the sender of the source - Integer lastBlockNumber = proofReceiver.getMetaKnowledge().get(sourceSender); - if (lastBlockNumber != null && sourceTransaction.getBlockNumber().getAsInt() <= lastBlockNumber) { - // Receiver knows about other node - this.knownSource.add(new SimpleEntry<>(sourceSender.getId(), sourceTransaction.getNumber())); - } else { - // Receiver does NOT know about other node - this.newSource.add(new SimpleEntry<>(sourceSender.getId(), sourceTransaction.getNumber())); - } + this.source.add(new TransactionSource( + sourceSender.getId(), + sourceTransaction.getBlockNumber().getAsInt(), + sourceTransaction.getNumber())); } + } else { + throw new IllegalStateException("Transaction without blocknumber found"); } } this.hash = transaction.getHash(); this.blockNumber = transaction.getBlockNumber().getAsInt(); } + /** + * Converts this message into a transaction without any sources. + * @param localStore - the local store + * @return - the transaction represented by this message, without sources + */ + public Transaction toTransactionWithoutSources(LocalStore localStore) { + Transaction tx = new Transaction(this.number, localStore.getNode(this.senderId), + localStore.getNode(this.receiverId), this.amount, this.remainder, new TreeSet<>()); + tx.setMessage(this); + return tx; + } + @Override public void handle(LocalStore localStore) { // Do nothing @@ -111,10 +109,9 @@ public boolean equals(Object obj) { if (receiverId != other.receiverId) return false; if (amount != other.amount) return false; if (remainder != other.remainder) return false; - if (knownSource.equals(other.knownSource)) return false; - if (newSource.equals(other.newSource)) return false; - if (hash.equals(other.hash)) return false; - return blockNumber == other.blockNumber; + if (blockNumber != other.blockNumber) return false; + if (!source.equals(other.source)) return false; + return hash.equals(other.hash); } @Override @@ -126,11 +123,77 @@ public int hashCode() { result = prime * result + this.receiverId; result = prime * result + (int) (this.amount ^ (this.amount >>> 32)); result = prime * result + (int) (this.remainder ^ (this.remainder >>> 32)); - result = prime * result + Objects.hashCode(this.knownSource); - result = prime * result + Objects.hashCode(this.newSource); + result = prime * result + Objects.hashCode(this.source); result = prime * result + Objects.hashCode(this.hash); result = prime * result + this.blockNumber; return result; } + @Override + public String toString() { + StringBuilder sb = new StringBuilder(128); + sb.append("TransactionMessage> encodedChainUpdates, - Map> decodedChainUpdates, LocalStore localStore) throws IOException { - this.number = blockMessage.getNumber(); - // It's a genesis block - if (blockMessage.getOwnerId() == Transaction.GENESIS_SENDER) { - this.owner = null; - } else { - this.owner = localStore.getNode(blockMessage.getOwnerId()); - } - - if (blockMessage.getPreviousBlockNumber() != -1) { - // Check if we have it in the local store - if (this.owner.getChain().getLastBlock().getNumber() < blockMessage.getPreviousBlockNumber()) { - // We don't have it (it should be in the received chain of updates) - int currentBlockIndex = encodedChainUpdates.get(this.owner.getId()).indexOf(blockMessage); - BlockMessage previousBlockMesssage = encodedChainUpdates.get(this.owner.getId()).get(currentBlockIndex - 1); - // Get decoded block list from the owner - Block previousBlockLocal = new Block(previousBlockMesssage, encodedChainUpdates, decodedChainUpdates, localStore); - if (decodedChainUpdates.containsKey(this.owner)) { - decodedChainUpdates.get(this.owner).add(previousBlockLocal); - } else { - List currentDecodedBlockList = new ArrayList<>(); - currentDecodedBlockList.add(previousBlockLocal); - decodedChainUpdates.put(this.owner, currentDecodedBlockList); - } - this.previousBlock = previousBlockLocal; - } else { - // We have it (we infer it's the lastBlock from the chain) - this.previousBlock = this.owner.getChain().getLastBlock(); - } - } else { - // It's a genesis block - this.previousBlock = null; - } - - // Convert TransactionMessage to Transaction - this.transactions = new ArrayList<>(); - for (TransactionMessage transactionMessage : blockMessage.getTransactions()) { - this.transactions.add(new Transaction(transactionMessage, encodedChainUpdates, decodedChainUpdates, localStore)); + public Transaction getTransaction(int transactionNumber) { + for (Transaction transaction : this.transactions) { + if (transaction.getNumber() == transactionNumber) + return transaction; } - //TODO Do we want to send the hash along? - this.hash = blockMessage.getHash(); + throw new IllegalStateException("Invalid transaction number"); } - + /** * Adds the given transaction to this block and sets its block number. * @param transaction - the transaction to add @@ -137,17 +98,18 @@ public synchronized void addTransaction(Transaction transaction) { } transactions.add(transaction); - transaction.setBlockNumber(this.getNumber()); + transaction.setBlockNumber(getNumber()); } /** * Get hash of the block. * @return Hash SHA256 */ - public synchronized Sha256Hash getHash() { + public Sha256Hash getHash() { if (this.hash == null) { - this.hash = this.calculateHash(); + this.hash = calculateHash(); } + return this.hash; } @@ -167,7 +129,7 @@ public BlockAbstract calculateBlockAbstract() { try (ByteArrayOutputStream outputStream = new ByteArrayOutputStream()) { outputStream.write(Utils.intToByteArray(this.owner.getId())); outputStream.write(Utils.intToByteArray(this.number)); - outputStream.write(this.getHash().getBytes()); + outputStream.write(getHash().getBytes()); attrInBytes = outputStream.toByteArray(); } catch (IOException ex) { throw new IllegalStateException("Unable to write to outputstream", ex); @@ -176,7 +138,7 @@ public BlockAbstract calculateBlockAbstract() { // Sign the attributes try { byte[] signature = ((OwnNode) this.owner).sign(attrInBytes); - BlockAbstract blockAbstract = new BlockAbstract(this.owner.getId(), this.number, this.getHash(), signature); + BlockAbstract blockAbstract = new BlockAbstract(this.owner.getId(), this.number, getHash(), signature); this.hasNoAbstract = false; return blockAbstract; } catch (Exception ex) { @@ -193,11 +155,18 @@ public synchronized void commit(LocalStore localStore) { throw new IllegalStateException("This block has already been committed!"); } - Chain chain = getOwner().getChain(); - synchronized (chain) { - BlockAbstract blockAbstract = calculateBlockAbstract(); - localStore.getMainChain().commitAbstract(blockAbstract); - getOwner().getChain().setLastCommittedBlock(this); + Log.log(Level.FINER, "Committing block " + getNumber(), getOwner().getId()); + + //Commit to the main chain, and set the last committed block + localStore.getMainChain().commitAbstract(calculateBlockAbstract()); + getOwner().getChain().setLastCommittedBlock(this); + + //Set next committed block + nextCommittedBlock = this; + Block prev = getPreviousBlock(); + while (prev != null && prev.nextCommittedBlock == null) { + prev.nextCommittedBlock = this; + prev = prev.getPreviousBlock(); } finalized = true; @@ -221,7 +190,9 @@ public boolean equals(Object obj) { if (this.number != other.number) return false; if (this.owner == null) { if (other.owner != null) return false; - } else if (other.owner == null || this.owner.getId() != other.owner.getId()) return false; + } else if (!this.owner.equals(other.owner)) return false; + + //TODO IMPORTANT We might not want to use equals for the previous block (as it will recurse further) if (this.previousBlock == null) { if (other.previousBlock != null) return false; } else if (!this.previousBlock.equals(other.previousBlock)) return false; @@ -231,7 +202,7 @@ public boolean equals(Object obj) { @Override public String toString() { - return "Block<" + number + ", " + owner + ">"; + return "Block"; } /** @@ -244,11 +215,15 @@ private Sha256Hash calculateHash() { try { // Important to keep the order of writings outputStream.write(Utils.intToByteArray(this.number)); + + //TODO IMPORTANT Should we include the hash of the previous block? byte[] prevBlockHash = (this.previousBlock != null) ? this.previousBlock.getHash().getBytes() : new byte[0]; outputStream.write(prevBlockHash); if (this.owner != null) { outputStream.write(Utils.intToByteArray(this.owner.getId())); } + + //TODO IMPORTANT Remove the transactions for testing for (Transaction tx : this.transactions) { outputStream.write(tx.getHash().getBytes()); } @@ -268,13 +243,15 @@ private Sha256Hash calculateHash() { public Block genesisCopy() { if (this.number != GENESIS_BLOCK_NUMBER) throw new UnsupportedOperationException("You can only copy genesis blocks"); - ArrayList transactionsCopy = new ArrayList<>(); + Block block = new Block(this.number, this.owner, new ArrayList<>()); for (Transaction transaction : transactions) { - transactionsCopy.add(transaction.genesisCopy()); + block.addTransaction(transaction.genesisCopy()); } - Block block = new Block(this.number, this.owner, transactionsCopy); + //The genesis block is on the main chain, cannot be modified and is its own committed block block.onMainChain = true; + block.finalized = true; + block.nextCommittedBlock = block; return block; } @@ -286,19 +263,20 @@ public Block genesisCopy() { public boolean isOnMainChain(LocalStore localStore) { //TODO Remove hack? if (this.number == GENESIS_BLOCK_NUMBER) return true; - + //Definitely has no abstract if (this.hasNoAbstract) return false; - + //We already determined before what the result should be if (this.onMainChain) return true; - + //It is present, so store it and return - if (localStore.getMainChain().isPresent(this.getHash())) { + if (localStore.getMainChain().isPresent(this)) { this.onMainChain = true; + this.nextCommittedBlock = this; return true; } - + //Not present (yet) return false; } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Chain.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Chain.java index 1c6fedf..6cdb768 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Chain.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Chain.java @@ -5,6 +5,9 @@ import java.util.ArrayList; import java.util.List; +import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; +import nl.tudelft.blockchain.scaleoutdistributedledger.utils.AppendOnlyArrayList; + /** * Chain class. */ @@ -14,7 +17,7 @@ public class Chain { private final Node owner; @Getter - private final List blocks; + private final AppendOnlyArrayList blocks; @Getter private Transaction genesisTransaction; @@ -27,48 +30,101 @@ public class Chain { */ public Chain(Node owner) { this.owner = owner; - this.blocks = new ArrayList<>(); - } - - /** - * Constructor. - * @param owner - the owner of this chain. - * @param blocks - list of blocks in this chain. - */ - public Chain(Node owner, List blocks) { - this.owner = owner; - this.blocks = blocks; + this.blocks = new AppendOnlyArrayList<>(); } /** * Updates this chain with the given updates. * This method is used for updating a chain belonging to a different node. * @param updates - the new blocks to append + * @param localStore - the localStore * @throws UnsupportedOperationException - If this chain is owned by us. */ - public synchronized void update(List updates) { + public void update(List updates, LocalStore localStore) { if (owner instanceof OwnNode) throw new UnsupportedOperationException("You cannot use update to update your own chain"); if (updates.isEmpty()) return; - if (blocks.isEmpty()) { - blocks.addAll(updates); - } else { - Block lastBlock = blocks.get(blocks.size() - 1); - int nextNr = lastBlock.getNumber() + 1; + Block lastCommitted = updates.get(updates.size() - 1); + synchronized (this) { + //Figure out where to start updating + int nextNr; + Block previousBlock; + if (blocks.isEmpty()) { + //Should start at 0, there is no previous block + previousBlock = null; + nextNr = 0; + } else { + //Should start with the first block after our last block. + previousBlock = blocks.get(blocks.size() - 1); + nextNr = previousBlock.getNumber() + 1; + } + + //Actually apply the updates + ArrayList toAdd = new ArrayList<>(); + int lastBlockNr = nextNr - 1; for (Block block : updates) { //Skip any overlap if (block.getNumber() != nextNr) continue; - blocks.add(block); + block.setPreviousBlock(previousBlock); + lastBlockNr = fixNextCommitted(block, lastCommitted.getNumber(), lastBlockNr, localStore); + toAdd.add(block); nextNr++; + previousBlock = block; + } + + blocks.addAll(toAdd); + } + + //The last block in the updates must be a committed block + setLastCommittedBlock(lastCommitted); + } + + /** + * Fixes the next committed block pointers. + * @param updates - the block updates + * @param localStore - the local store + */ + private void fixNextCommitted(List updates, LocalStore localStore) { + if (updates.isEmpty()) return; + + int lastBlockNr = updates.get(0).getNumber(); + for (Block block : updates) { + if (!localStore.getMainChain().isInCache(block)) continue; + + Block prev = block.getPreviousBlock(); + while (prev != null && prev.getNumber() > lastBlockNr) { + prev.setNextCommittedBlock(block); + prev = prev.getPreviousBlock(); } + + lastBlockNr = block.getNumber(); } } + /** + * @param block - the block + * @param lastUpdateBlockNr - the number of the last block in the list of updates + * @param lastBlockNr - the number of the last checked block that was actually committed + * @param localStore - the local store + * @return - the new lastBlockNr + */ + private int fixNextCommitted(Block block, int lastUpdateBlockNr, int lastBlockNr, LocalStore localStore) { + if (block.getNumber() != lastUpdateBlockNr && !localStore.getMainChain().isInCache(block)) return lastBlockNr; + + Block prev = block; + while (prev != null && prev.getNumber() > lastBlockNr) { + prev.setNextCommittedBlock(block); + prev = prev.getPreviousBlock(); + } + + return block.getNumber(); + } + /** * @return - the genesis block */ - public synchronized Block getGenesisBlock() { + public Block getGenesisBlock() { if (blocks.isEmpty()) return null; return blocks.get(0); @@ -93,16 +149,23 @@ public synchronized void setGenesisBlock(Block genesisBlock) { /** * @return the last block in this chain */ - public synchronized Block getLastBlock() { + public Block getLastBlock() { if (blocks.isEmpty()) return null; return blocks.get(blocks.size() - 1); } + /** + * @return - the number of the last block + */ + public int getLastBlockNumber() { + return blocks.size() - 1; + } + /** * @return - the last block that was committed to the main chain */ - public synchronized Block getLastCommittedBlock() { + public Block getLastCommittedBlock() { return lastCommittedBlock; } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainView.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainView.java index b70c5d6..675c89a 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainView.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainView.java @@ -15,6 +15,9 @@ public class ChainView implements Iterable { private Chain chain; private List updates; private Boolean valid; + private boolean trim; + private int startIndex; + private int chainSize; /** * @param chain @@ -23,9 +26,20 @@ public class ChainView implements Iterable { * the blocks of this chain that were sent with the proof */ public ChainView(Chain chain, List updates) { + this(chain, updates, true); + } + + /** + * @param chain - the chain + * @param updates - the blocks of this chain that were sent with the proof + * @param trim - if this chainview should trim + */ + public ChainView(Chain chain, List updates, boolean trim) { this.chain = chain; this.updates = updates; if (this.updates == null) this.updates = new ArrayList<>(); + this.trim = trim; + this.chainSize = chain.getBlocks().size(); } /** @@ -77,26 +91,28 @@ private boolean checkIntegrity() { return true; } + chainSize = chain.getBlocks().size(); + //If we had no blocks, then we only need to check for gaps - List blocks = chain.getBlocks(); int firstUpdateNumber = updates.get(0).getNumber(); - if (blocks.isEmpty()) { + if (chainSize == 0) { return checkNoGaps(1, firstUpdateNumber); } - int lastOwnNumber = blocks.get(blocks.size() - 1).getNumber(); + int lastOwnNumber = chainSize - 1; if (firstUpdateNumber - lastOwnNumber > 1) { //We are missing blocks between what we know and what we were sent! Can never happen with an honest node. + System.out.println("HERE1"); this.valid = false; return false; } else if (lastOwnNumber >= firstUpdateNumber) { //There is overlap, check if exactly matches our view //At the same time, we will remove the overlapping elements int overlap = lastOwnNumber + 1 - firstUpdateNumber; - int baseI = blocks.size() - overlap; - for (int i = 0; i < overlap && !updates.isEmpty(); i++) { - Block ownBlock = blocks.get(baseI + i); - Block updatedBlock = updates.get(0); + int baseI = chainSize - overlap; + for (int i = 0; i < overlap && !updates.isEmpty() && startIndex < updates.size(); i++) { + Block ownBlock = chain.getBlocks().get(baseI + i); + Block updatedBlock = updates.get(startIndex); //TODO we might need a special equality check if (!ownBlock.equals(updatedBlock)) { @@ -104,10 +120,14 @@ private boolean checkIntegrity() { return false; } - updates.remove(0); + if (trim) { + updates.remove(0); + } else { + startIndex++; + } } - - return checkNoGaps(0, lastOwnNumber); + + return checkNoGaps(startIndex, lastOwnNumber); } else { //The first updated block number follows directly after the last block we knew about. return checkNoGaps(0, lastOwnNumber); @@ -156,7 +176,7 @@ public Block getBlock(int number) { if (number < chain.getBlocks().size()) { return chain.getBlocks().get(number); } else if (isValid()) { - int index = number - chain.getBlocks().size(); + int index = number - chainSize + startIndex; return updates.get(index); } else { throw new IllegalStateException( @@ -164,6 +184,15 @@ public Block getBlock(int number) { } } + /** + * @return - the amount of blocks in this chainview + */ + public int size() { + if (!isValid()) throw new IllegalStateException("This chainview is invalid"); + + return chainSize + updates.size() - startIndex; + } + @Override public ListIterator iterator() { return listIterator(); @@ -198,20 +227,19 @@ private class ChainViewIterator implements ListIterator { private int currentIndex; ChainViewIterator() { - chainIterator = chain.getBlocks().listIterator(); - updatesIterator = updates.listIterator(); + chainIterator = chain.getBlocks().listIterator(0, chainSize); + updatesIterator = updates.subList(startIndex, updates.size()).listIterator(); currentIndex = -1; } ChainViewIterator(int number) { - int chainLength = chain.getBlocks().size(); - if (number < chainLength) { - chainIterator = chain.getBlocks().listIterator(number); - updatesIterator = updates.listIterator(); + if (number < chainSize) { + chainIterator = chain.getBlocks().listIterator(number, chainSize); + updatesIterator = updates.subList(startIndex, updates.size()).listIterator(); } else { - int index = number - chainLength; - updatesIterator = updates.listIterator(index); - chainIterator = chain.getBlocks().listIterator(chainLength); + int index = number - chainSize; + updatesIterator = updates.subList(startIndex, updates.size()).listIterator(index); + chainIterator = chain.getBlocks().listIterator(chainSize, chainSize); updatesReached = true; } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/LightView.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/LightView.java new file mode 100644 index 0000000..5f0673f --- /dev/null +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/LightView.java @@ -0,0 +1,36 @@ +package nl.tudelft.blockchain.scaleoutdistributedledger.model; + +import java.util.ArrayList; +import java.util.List; + +/** + * Class which provides an easy way of getting blocks. + */ +public class LightView { + private Chain chain; + private List updates; + + /** + * @param chain - the chain + * @param updates - the blocks of this chain that were sent with the proof + */ + public LightView(Chain chain, List updates) { + this.chain = chain; + this.updates = updates; + if (this.updates == null) this.updates = new ArrayList<>(); + } + + /** + * @param number - the number + * @return - the block with the given number + * @throws IndexOutOfBoundsException - If the block with the given number does not exist (yet). + */ + public Block getBlock(int number) { + if (number < chain.getBlocks().size()) { + return chain.getBlocks().get(number); + } else { + int index = number - updates.get(0).getNumber(); + return updates.get(index); + } + } +} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/MetaKnowledge.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/MetaKnowledge.java new file mode 100644 index 0000000..8b8ed27 --- /dev/null +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/MetaKnowledge.java @@ -0,0 +1,114 @@ +package nl.tudelft.blockchain.scaleoutdistributedledger.model; + +import java.util.Collections; +import java.util.HashMap; +import java.util.List; + +import lombok.Getter; + +/** + * A class for representing meta knowledge (what we know that they know). + */ +public class MetaKnowledge extends HashMap { + private static final long serialVersionUID = 1L; + + /** + * @return - the node that this meta knowledge belongs to + */ + @Getter + private final Node owner; + + /** + * @param owner - the owner of this meta knowledge + */ + public MetaKnowledge(Node owner) { + this.owner = owner; + } + + /** + * Determines the blocks that we have to send to owner in order to update them to the given + * end block number. + * + * This method returns {@link Collections#emptyList()} if the given node is the owner or if + * owner already knows about all the blocks up till the given block number. + * @param node - the node to send blocks of + * @param endBlockNr - the number of the last block we want to send + * @return - the list of blocks to send + * @throws IndexOutOfBoundsException - If the given node does not have a block with endBlockNr. + */ + public List getBlocksToSend(Node node, int endBlockNr) { + if (node == owner) return Collections.emptyList(); + + //Determine the first block they don't know + int firstUnknown = getFirstUnknownBlockNumber(node); + + //They already know everything we want to send + if (firstUnknown > endBlockNr) return Collections.emptyList(); + + //Get a sublist of the blocks on the chain. We need + 1 on endBlock because the end is exclusive. + return node.getChain().getBlocks().subList(firstUnknown, endBlockNr + 1); + } + + /** + * The number returned by this method is the number of the last block that {@link #getOwner()} + * knows from node {@code node}. + * + * This method returns -1 if the owner of this meta knowledge does not know about this block. + * @param node - the node + * @return - the number of the last block from the given node that is known by owner + */ + public int getLastKnownBlockNumber(Node node) { + return getLastKnownBlockNumber(node.getId()); + } + + /** + * The number returned by this method is the number of the last block that {@link #getOwner()} + * knows from the node with id {@code nodeId}. + * + * This method returns -1 if the owner of this meta knowledge does not know about this block. + * @param nodeId - the id of the node + * @return - the number of the last block from the given node that is known by owner + */ + public int getLastKnownBlockNumber(int nodeId) { + if (nodeId == owner.getId()) return owner.getChain().getLastBlockNumber(); + return getOrDefault(nodeId, -1); + } + + /** + * @param node - the node + * @return - the number of the first block from the given node that is unknown by owner + */ + public int getFirstUnknownBlockNumber(Node node) { + return getFirstUnknownBlockNumber(node.getId()); + } + + /** + * @param nodeId - the id of the node + * @return - the number of the first block from the given node that is unknown by owner + */ + public int getFirstUnknownBlockNumber(int nodeId) { + return getLastKnownBlockNumber(nodeId) + 1; + } + + /** + * Updates the last known block number of the given node to the given blockNumber. + * If the given block number is lower than the current last known block number, then this + * method does nothing. + * @param node - the node + * @param blockNumber - the block number + */ + public void updateLastKnownBlockNumber(Node node, int blockNumber) { + updateLastKnownBlockNumber(node.getId(), blockNumber); + } + + /** + * Updates the last known block number of the given node to the given blockNumber. + * If the given block number is lower than the current last known block number, then this + * method does nothing. + * @param nodeId - the id of the node + * @param blockNumber - the block number + */ + public synchronized void updateLastKnownBlockNumber(int nodeId, int blockNumber) { + merge(nodeId, blockNumber, (oldNr, newNr) -> Math.max(oldNr, newNr)); + } +} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Node.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Node.java index e783532..b207d9e 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Node.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Node.java @@ -3,7 +3,6 @@ import lombok.Getter; import lombok.Setter; -import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Map.Entry; @@ -28,11 +27,8 @@ public class Node { @Getter @Setter private int port; - /** - * @return a map containing what we know that this node knows - */ @Getter - private Map metaKnowledge = new HashMap<>(); + private MetaKnowledge metaKnowledge = new MetaKnowledge(this); /** * Constructor. @@ -77,8 +73,12 @@ public boolean verify(byte[] message, byte[] signature) throws Exception { public void updateMetaKnowledge(Proof proof) { Map> updates = proof.getChainUpdates(); for (Entry> entry : updates.entrySet()) { + //Don't include self + if (entry.getKey() == this) continue; + int lastBlockNr = getLastBlockNumber(entry.getValue()); - metaKnowledge.merge(entry.getKey(), lastBlockNr, Math::max); + if (lastBlockNr == -1) continue; + metaKnowledge.updateLastKnownBlockNumber(entry.getKey(), lastBlockNr); } } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/OwnNode.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/OwnNode.java index 1e8d921..5a671a9 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/OwnNode.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/OwnNode.java @@ -3,6 +3,9 @@ import lombok.Getter; import lombok.Setter; +/** + * Class to represent our own node. + */ public class OwnNode extends Node { /** * Only used by the node himself. @@ -11,10 +14,19 @@ public class OwnNode extends Node { @Getter @Setter private transient byte[] privateKey; + /** + * @param id - the id of our node + */ public OwnNode(int id) { super(id); } + /** + * @param id - the id of our node + * @param publicKey - the public key + * @param address - the address + * @param port - the port + */ public OwnNode(int id, byte[] publicKey, String address, int port) { super(id, publicKey, address, port); } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Proof.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Proof.java index 0f46a78..b3220f5 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Proof.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Proof.java @@ -3,6 +3,7 @@ import lombok.Getter; import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; import nl.tudelft.blockchain.scaleoutdistributedledger.message.ProofMessage; +import nl.tudelft.blockchain.scaleoutdistributedledger.message.TransactionMessage.TransactionSource; import nl.tudelft.blockchain.scaleoutdistributedledger.message.BlockMessage; import nl.tudelft.blockchain.scaleoutdistributedledger.validation.ProofValidationException; import nl.tudelft.blockchain.scaleoutdistributedledger.validation.ValidationException; @@ -10,10 +11,7 @@ import java.io.IOException; import java.util.ArrayList; import java.util.HashMap; -import java.util.HashSet; -import java.util.Iterator; import java.util.List; -import java.util.ListIterator; import java.util.Map; import java.util.Map.Entry; import java.util.Set; @@ -28,6 +26,8 @@ public class Proof { @Getter private final Map> chainUpdates; + + private final Map chainViews = new HashMap<>(); /** * Constructor. @@ -38,16 +38,6 @@ public Proof(Transaction transaction) { this.chainUpdates = new HashMap<>(); } - /** - * Constructor. - * @param transaction - the transaction to be proven. - * @param chainUpdates - a map of chain updates - */ - public Proof(Transaction transaction, Map> chainUpdates) { - this.transaction = transaction; - this.chainUpdates = chainUpdates; - } - /** * Constructor to decode a proof message. * @param proofMessage - proof received from the network @@ -56,39 +46,74 @@ public Proof(Transaction transaction, Map> chainUpdates) { */ public Proof(ProofMessage proofMessage, LocalStore localStore) throws IOException { this.chainUpdates = new HashMap<>(); - // Start by decoding the chain of the sender + + // Decode the transactions while skipping sources + for (Entry> entry : proofMessage.getChainUpdates().entrySet()) { + List blocks = new ArrayList<>(); + for (BlockMessage blockMessage : entry.getValue()) { + blocks.add(blockMessage.toBlockWithoutSources(localStore)); + } + chainUpdates.put(localStore.getNode(entry.getKey()), blocks); + } + + // Fix backlinks + this.fixPreviousBlockPointers(); + + // Fix the sources + this.fixTransactionSources(localStore); + Node senderNode = localStore.getNode(proofMessage.getTransactionMessage().getSenderId()); - List senderChain = proofMessage.getChainUpdates().get(senderNode.getId()); - // Start from the last block - BlockMessage lastBlockMessage = senderChain.get(senderChain.size() - 1); - // Recursively decode the transaction and chainUpdates - Block lastBlock = new Block(lastBlockMessage, proofMessage.getChainUpdates(), this.chainUpdates, localStore); - List currentDecodedBlockList; - if (this.chainUpdates.containsKey(senderNode)) { - // Add to already created list of blocks - currentDecodedBlockList = this.chainUpdates.get(senderNode); - currentDecodedBlockList.add(lastBlock); - } else { - // Create new list of blocks - currentDecodedBlockList = new ArrayList<>(); - currentDecodedBlockList.add(lastBlock); - this.chainUpdates.put(senderNode, currentDecodedBlockList); + ChainView senderChainView = getChainView(senderNode); + this.transaction = senderChainView.getBlock(proofMessage.getTransactionMessage().getBlockNumber()) + .getTransaction(proofMessage.getTransactionMessage().getNumber()); + } + + private void fixPreviousBlockPointers() { + for (Entry> entry : this.chainUpdates.entrySet()) { + Node node = entry.getKey(); + List updates = entry.getValue(); + if (updates.isEmpty()) continue; + + Block previousBlock = null; + for (int i = 0; i < updates.size(); i++) { + Block block = updates.get(i); + block.setPreviousBlock(previousBlock); + previousBlock = block; + } + + Block firstBlock = updates.get(0); + if (firstBlock.getNumber() != 0) { + previousBlock = node.getChain().getBlocks().get(firstBlock.getNumber() - 1); + firstBlock.setPreviousBlock(previousBlock); + } } - // Set the transaction from the decoded chain - // TODO [possible improvement]: is the transaction always in the last block ? - Transaction foundTransaction = null; - - ChainView cv = new ChainView(senderNode.getChain(), currentDecodedBlockList); - Block block = cv.getBlock(proofMessage.getTransactionMessage().getBlockNumber()); - for (Transaction transactionAux : block.getTransactions()) { - if (transactionAux.getNumber() == proofMessage.getTransactionMessage().getNumber()) { - foundTransaction = transactionAux; - break; + } + + private void fixTransactionSources(LocalStore localStore) { + HashMap lightViews = new HashMap<>(); + // Initialize the chainviews only once + for (Node node : this.chainUpdates.keySet()) { + lightViews.put(node.getId(), new LightView(node.getChain(), chainUpdates.get(node))); + } + + // For all transactions of all nodes do + for (List blocks : this.chainUpdates.values()) { + for (Block block : blocks) { + for (Transaction tx : block.getTransactions()) { + for (TransactionSource ts : tx.getMessage().getSource()) { + Block sourceBlock; + if (!lightViews.containsKey(ts.getOwner())) { + sourceBlock = localStore.getNode(ts.getOwner()).getChain().getBlocks().get(ts.getBlockNumber()); + } else { + sourceBlock = lightViews.get(ts.getOwner()).getBlock(ts.getBlockNumber()); + } + tx.getSource().add(sourceBlock.getTransaction(ts.getId())); + } + } } } - this.transaction = foundTransaction; } - + /** * Add a block to the proof. * @param block - the block to be added @@ -97,23 +122,6 @@ public void addBlock(Block block) { List blocks = chainUpdates.computeIfAbsent(block.getOwner(), k -> new ArrayList<>()); blocks.add(block); } - - /** - * Adds the blocks with numbers start to end of the given chain to the proof. - * @param chain - the chain - * @param start - the block to start at (inclusive) - * @param end - the block to end at (exclusive) - */ - public void addBlocksOfChain(Chain chain, int start, int end) { - if (start >= end || end > chain.getBlocks().size()) return; - - List blocks = chainUpdates.get(chain.getOwner()); - if (blocks == null) { - blocks = new ArrayList<>(); - chainUpdates.put(chain.getOwner(), blocks); - } - blocks.addAll(chain.getBlocks().subList(start, end)); - } /** * Verifies this proof. @@ -125,6 +133,7 @@ public void verify(LocalStore localStore) throws ProofValidationException { throw new ProofValidationException("We directly received a transaction with a null sender."); } + //TODO [BFT] all the blocks that were sent but not required for the proof are not validated at all. verify(this.transaction, localStore); } @@ -134,6 +143,8 @@ public void verify(LocalStore localStore) throws ProofValidationException { * @throws ProofValidationException - If the proof is invalid. */ private void verify(Transaction transaction, LocalStore localStore) throws ProofValidationException { + if (transaction.isLocallyVerified()) return; + int blockNumber = transaction.getBlockNumber().orElse(-1); if (blockNumber == -1) { throw new ProofValidationException("The transaction has no block number, so we cannot validate it."); @@ -141,37 +152,76 @@ private void verify(Transaction transaction, LocalStore localStore) throws Proof if (transaction.getSender() == null) { verifyGenesisTransaction(transaction, localStore); + transaction.setLocallyVerified(true); return; } - int absmark = 0; - boolean seen = false; + verifyChainWithTransaction(transaction, localStore, blockNumber); + verifySourceTransactions(transaction, localStore); + transaction.setLocallyVerified(true); + } - //TODO [PERFORMANCE]: We check the same chain views multiple times, even though we don't have to. + /** + * Performs the first part of the verification of a transaction. + * This method first checks if the chain is consistent with the updates in this proof, and then + * checks that the transaction only appears once. Finally, it checks if the transaction is in + * a block that before a committed block, or is itself committed. + * @param transaction - the transaction + * @param localStore - the local store + * @param blockNumber - the block number of transaction + */ + private void verifyChainWithTransaction(Transaction transaction, LocalStore localStore, int blockNumber) { ChainView chainView = getChainView(transaction.getSender()); if (!chainView.isValid()) { throw new ProofValidationException("ChainView of node " + transaction.getSender().getId() + " is invalid."); } - + + boolean seen = false; + boolean absmark = false; for (Block block : chainView) { + //TODO This containment check will not report transactions with the same id in different blocks (they will be unequal). + //It is therefore impossible to find a duplicate transaction if (block.getTransactions().contains(transaction)) { if (seen) { throw new ProofValidationException("Duplicate transaction found."); } seen = true; } - if (block.isOnMainChain(localStore)) absmark = block.getNumber(); + + //If a block at or after the block in question is committed, then we have found a valid absmark + if (!absmark && block.getNumber() >= blockNumber) { + Block nextCommitted = block.getNextCommittedBlock(); + + if (nextCommitted != null || block.isOnMainChain(localStore)) { + absmark = true; + } + } } - - if (absmark < blockNumber) { - throw new ProofValidationException("No suitable committed block found"); + + if (!seen) { + throw new ProofValidationException("Transaction not found in any block!"); } - - // Verify source transaction + + if (!absmark) { + throw new ProofValidationException("No suitable committed block found for block " + blockNumber); + } + } + + /** + * Performs the second part of the verification of a transaction. + * This method verifies all source transactions. + * @param transaction - the transaction + * @param localStore - the local store + */ + private void verifySourceTransactions(Transaction transaction, LocalStore localStore) { for (Transaction sourceTransaction : transaction.getSource()) { try { verify(sourceTransaction, localStore); } catch (ValidationException ex) { + //TODO Remove debugging stuff + ex.printStackTrace(); + System.out.println(this); + System.exit(1); throw new ProofValidationException("Source " + sourceTransaction + " is not valid", ex); } } @@ -209,155 +259,95 @@ private void verifyGenesisTransaction(Transaction transaction, LocalStore localS * @param node - the node * @return - a chainview for the specified node */ - public ChainView getChainView(Node node) { - return new ChainView(node.getChain(), chainUpdates.get(node)); + public synchronized ChainView getChainView(Node node) { + ChainView chainView = chainViews.get(node); + if (chainView == null) { + chainView = new ChainView(node.getChain(), chainUpdates.get(node), false); + chainView.isValid(); + chainViews.put(node, chainView); + } + return chainView; } /** * Applies the updates in this proof. * This method also updates the meta knowledge of the sender of the transaction. + * @param localStore - the localStore */ - public void applyUpdates() { + public void applyUpdates(LocalStore localStore) { for (Entry> entry : chainUpdates.entrySet()) { Node node = entry.getKey(); List updates = entry.getValue(); - node.getChain().update(updates); + node.getChain().update(updates, localStore); } //Update the meta knowledge of the sender transaction.getSender().updateMetaKnowledge(this); } - /** - * @param localStore - the local store - * @param transaction - the transaction - * @return the proof for the given transaction - */ - public static Proof createProof(LocalStore localStore, Transaction transaction) { - Node receiver = transaction.getReceiver(); - Proof proof = new Proof(transaction); - - //Step 1: determine what blocks need to be sent - int blockRequired = transaction.getBlockNumber().getAsInt(); - Chain senderChain = transaction.getSender().getChain(); - - Block fromBlock = senderChain.getBlocks().get(blockRequired); - Block toBlock = getNextCommittedBlock(localStore, blockRequired, senderChain); - - //Step 2: determine the chains that need to be sent - Map chains = determineChains(transaction, fromBlock, toBlock); - - //Step 3: add only those blocks that are not yet known - Map metaKnowledge = receiver.getMetaKnowledge(); - for (Entry entry : chains.entrySet()) { - Chain chain = entry.getKey(); - Node owner = chain.getOwner(); - if (owner == receiver) continue; - - int alreadyKnown = metaKnowledge.getOrDefault(owner, -1); - int requiredKnown = entry.getValue(); - if (alreadyKnown < requiredKnown) { - proof.addBlocksOfChain(chain, alreadyKnown + 1, requiredKnown + 1); - } - } - - return proof; - } - - /** - * @param mainTransaction - the transaction that we want to send - * @param fromBlock - the first block that we want to send - * @param toBlock - the last block that we want to send - * @return - the chains required and the corresponding block number - */ - private static Map determineChains(Transaction mainTransaction, Block fromBlock, Block toBlock) { - //TODO We might want to do some kind of caching? - - Node receiver = mainTransaction.getReceiver(); - Map metaKnowledge = receiver.getMetaKnowledge(); - - //Determine what the sender already knows about us - int alreadyKnown = metaKnowledge.getOrDefault(mainTransaction.getSender(), -1); - - Map chains = new HashMap<>(); - - //Only consider the transactions that the receiver doesn't have yet. - Block current = toBlock; - while (current.getNumber() > alreadyKnown) { - for (Transaction transaction : current.getTransactions()) { - appendChains2(transaction, receiver, chains); - } - - if (current == fromBlock) break; - current = current.getPreviousBlock(); - } - - //Only add the chains of the transaction itself if it isn't in a known block - if (mainTransaction.getBlockNumber().getAsInt() > alreadyKnown) { - appendChains2(mainTransaction, receiver, chains); - } - return chains; - } - - /** - * @param localStore - * @param blockRequired - * @param senderChain - * @return - */ - private static Block getNextCommittedBlock(LocalStore localStore, int blockRequired, Chain senderChain) { - ListIterator it = senderChain.getBlocks().listIterator(blockRequired); - while (it.hasNext()) { - Block block = it.next(); - if (block.isOnMainChain(localStore)) { - return block; - } - } - - throw new IllegalStateException("There is no next committed block!"); - } - /** * Recursively calls itself with all the sources of the given transaction. Transactions which * are in the chain of {@code receiver} are ignored. - * @param transaction - the transaction to check the sources of - * @param receiver - the node receiving the transaction - * @param chains - the list of chains to append to + * @param nrOfNodes - the total number of nodes + * @param transaction - the transaction to check the sources of + * @param receiver - the node receiving the transaction + * @param metaKnowledge - the meta knowledge + * @param chains - the list of chains to append to */ - public static void appendChains(Transaction transaction, Node receiver, Set chains) { + public static void appendChains(int nrOfNodes, Transaction transaction, Node receiver, MetaKnowledge metaKnowledge, Set chains) { Node owner = transaction.getSender(); if (owner == null || owner == receiver) return; + //TODO Do we want to cut off at known blocks? + int alreadyKnown = metaKnowledge.getFirstUnknownBlockNumber(owner); + int blockNumber = transaction.getBlockNumber().getAsInt(); + if (alreadyKnown >= blockNumber) return; + chains.add(owner.getChain()); + if (chains.size() >= nrOfNodes - 1) return; + for (Transaction source : transaction.getSource()) { - appendChains(source, receiver, chains); + appendChains(nrOfNodes, source, receiver, metaKnowledge, chains); } } /** * Recursively calls itself with all the sources of the given transaction. Transactions which * are in the chain of {@code receiver} are ignored. + * @param nrOfNodes - the total number of nodes * @param transaction - the transaction to check the sources of * @param receiver - the node receiving the transaction * @param chains - the map of chains to append to */ - public static void appendChains2(Transaction transaction, Node receiver, Map chains) { + public static void appendChains2(int nrOfNodes, Transaction transaction, Node receiver, Map chains) { Node owner = transaction.getSender(); if (owner == null || owner == receiver) return; //Skip transactions that are already known - Map metaKnowledge = receiver.getMetaKnowledge(); - int alreadyKnown = metaKnowledge.getOrDefault(owner, -1); + MetaKnowledge metaKnowledge = receiver.getMetaKnowledge(); + int lastKnown = metaKnowledge.getLastKnownBlockNumber(owner); int blockNumber = transaction.getBlockNumber().getAsInt(); - if (alreadyKnown >= blockNumber) return; + if (lastKnown >= blockNumber) return; //Store the highest block number. - chains.compute(owner.getChain(), (k, v) -> v == null ? blockNumber : Math.max(v, blockNumber)); + //Now consider this chain up to the last committed block + chains.merge(owner, blockNumber, Math::max); + if (chains.size() >= nrOfNodes - 1) return; + + //Check all the sources for (Transaction source : transaction.getSource()) { - appendChains2(source, receiver, chains); + appendChains2(nrOfNodes, source, receiver, chains); } } + + /** + * Gets the number of blocks used in the proof. + * @return - the number of blocks + */ + public int getNumberOfBlocks() { + return chainUpdates.values().stream().mapToInt(List::size).sum(); + } @Override public String toString() { diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Transaction.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Transaction.java index 70237f0..bf7ba00 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Transaction.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/Transaction.java @@ -1,28 +1,23 @@ package nl.tudelft.blockchain.scaleoutdistributedledger.model; -import lombok.Getter; -import lombok.Setter; -import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; -import nl.tudelft.blockchain.scaleoutdistributedledger.message.TransactionMessage; -import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; -import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Utils; - import java.io.ByteArrayOutputStream; import java.io.IOException; -import java.util.ArrayList; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Map.Entry; +import java.util.Arrays; import java.util.OptionalInt; -import java.util.Set; +import java.util.TreeSet; import java.util.logging.Level; -import nl.tudelft.blockchain.scaleoutdistributedledger.message.BlockMessage; + +import nl.tudelft.blockchain.scaleoutdistributedledger.message.TransactionMessage; +import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; +import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Utils; + +import lombok.Getter; +import lombok.Setter; /** * Transaction class. */ -public class Transaction { +public class Transaction implements Comparable { // Represent the sender of a genesis transaction public static final int GENESIS_SENDER = -1; @@ -33,7 +28,6 @@ public class Transaction { @Getter private final Node sender; - // TODO: change back to final somehow @Getter @Setter private Node receiver; @@ -41,23 +35,30 @@ public class Transaction { private final long amount, remainder; @Getter - private final Set source; + private final TreeSet source; // Custem getter private Sha256Hash hash; private OptionalInt blockNumber; + // Only temporarily used while decoding + @Getter @Setter + private TransactionMessage message; + + @Getter @Setter + private boolean locallyVerified; + /** * Constructor. - * @param number - the number of this transaction. - * @param sender - the sender of this transaction. - * @param receiver - the receiver of this transaction. - * @param amount - the amount to be transferred. + * @param number - the number of this transaction. + * @param sender - the sender of this transaction. + * @param receiver - the receiver of this transaction. + * @param amount - the amount to be transferred. * @param remainder - the remaining amount. - * @param source - set of transactions that are used as source for this transaction. + * @param source - set of transactions that are used as source for this transaction. */ - public Transaction(int number, Node sender, Node receiver, long amount, long remainder, Set source) { + public Transaction(int number, Node sender, Node receiver, long amount, long remainder, TreeSet source) { this.sender = sender; this.receiver = receiver; this.amount = amount; @@ -66,79 +67,21 @@ public Transaction(int number, Node sender, Node receiver, long amount, long rem this.number = number; this.blockNumber = OptionalInt.empty(); } - + /** - * Constructor to decode a transaction message. - * @param transactionMessage - the message received from a transaction. - * @param encodedChainUpdates - the received chain of updates - * @param decodedChainUpdates - current chain of updates, from the decoding process - * @param localStore - local store, to get each Node object - * @throws java.io.IOException - error while getting node + * Convenience constructor. The given sources are converted to a TreeSet with + *
new TreeSet<>(Arrays.asList(source))
. + * @param number - the number of this transaction. + * @param sender - the sender of this transaction. + * @param receiver - the receiver of this transaction. + * @param amount - the amount to be transferred. + * @param remainder - the remaining amount. + * @param source - the transaction that are used as sources for this transaction. */ - public Transaction(TransactionMessage transactionMessage, Map> encodedChainUpdates, - Map> decodedChainUpdates, LocalStore localStore) throws IOException { - this.number = transactionMessage.getNumber(); - this.blockNumber = OptionalInt.of(transactionMessage.getBlockNumber()); - - // It's a genesis transaction - if (transactionMessage.getSenderId() == GENESIS_SENDER) { - this.sender = null; - } else { - this.sender = localStore.getNode(transactionMessage.getSenderId()); - } - this.receiver = localStore.getNode(transactionMessage.getReceiverId()); - this.amount = transactionMessage.getAmount(); - this.remainder = transactionMessage.getRemainder(); - // Decode transaction messages to normal transactions - this.source = new HashSet<>(); - // Use local store for known sources - for (Entry knownSourceEntry : transactionMessage.getKnownSource()) { - Integer nodeId = knownSourceEntry.getKey(); - Integer transactionId = knownSourceEntry.getValue(); - //TODO This might need to be done in a certain order - this.source.add(localStore.getTransactionFromNode(nodeId, transactionId)); - } - // Use chain of updates for new sources - for (Entry newSourceEntry : transactionMessage.getNewSource()) { - try { - // Try to find the transaction in the local store - this.source.add(localStore.getTransactionFromNode(newSourceEntry.getKey(), newSourceEntry.getValue())); - continue; - } catch (IllegalStateException ex) { - // Not in localStore - } - // Use the transaction from the current chain of updates - Node owner = localStore.getNode(newSourceEntry.getKey()); - if (!decodedChainUpdates.containsKey(owner)) { - // Get that new chain - List blockMessageList = encodedChainUpdates.get(owner.getId()); - // Decode chain, in REVERSE order - BlockMessage lastBlockMessage = blockMessageList.get(blockMessageList.size() - 1); - // Recursively decode the blocks of a chain (in reverse order) - Block lastBlockLocal = new Block(lastBlockMessage, encodedChainUpdates, decodedChainUpdates, localStore); - if (decodedChainUpdates.containsKey(owner)) { - // Add to already created list - decodedChainUpdates.get(owner).add(lastBlockLocal); - } else { - // Create a new list - List blockList = new ArrayList<>(); - blockList.add(lastBlockLocal); - decodedChainUpdates.put(owner, blockList); - } - } - // TODO [Performance]: Find a way to directly go to the correct block instead of iterating through all of them - for (Block blockAux : decodedChainUpdates.get(owner)) { - for (Transaction transactionAux : blockAux.getTransactions()) { - if (transactionAux.getNumber() == newSourceEntry.getValue()) { - this.source.add(transactionAux); - break; - } - } - } - } - this.hash = transactionMessage.getHash(); + public Transaction(int number, Node sender, Node receiver, long amount, long remainder, Transaction... source) { + this(number, sender, receiver, amount, remainder, new TreeSet<>(Arrays.asList(source))); } - + /** * Returns the number of the block (if it is in a block). * TODO: maybe do this more efficiently (when adding the transaction to the local chain or something) @@ -150,6 +93,8 @@ public OptionalInt getBlockNumber() { if (this.sender == null) { this.blockNumber = OptionalInt.of(Block.GENESIS_BLOCK_NUMBER); } else { + //TODO IMPORTANT We don't want this to be called really. + System.out.println("Looking up block number!"); for (Block block : sender.getChain().getBlocks()) { if (block.getTransactions().contains(this)) { this.blockNumber = OptionalInt.of(block.getNumber()); @@ -160,7 +105,7 @@ public OptionalInt getBlockNumber() { } return this.blockNumber; } - + /** * Sets the block number of this transaction. * @param number - the block number @@ -197,7 +142,6 @@ private Sha256Hash calculateHash() { outputStream.write(Utils.longToByteArray(this.amount)); outputStream.write(Utils.longToByteArray(this.remainder)); - // TODO: check if we really need to do this for (Transaction tx : this.source) { outputStream.write(tx.getHash().getBytes()); } @@ -215,9 +159,7 @@ private Sha256Hash calculateHash() { */ public Transaction genesisCopy() { if (!source.isEmpty()) throw new UnsupportedOperationException("Only genesis transactions can be copied"); - Transaction transaction = new Transaction(number, sender, receiver, amount, remainder, new HashSet<>(0)); - transaction.blockNumber = OptionalInt.of(0); - return transaction; + return new Transaction(number, sender, receiver, amount, remainder, new TreeSet<>()); } @Override @@ -237,13 +179,12 @@ public boolean equals(Object obj) { Transaction other = (Transaction) obj; if (number != other.number) return false; - if (receiver.getId() != other.receiver.getId()) return false; + if (!receiver.equals(other.receiver)) return false; if (sender == null) { if (other.sender != null) return false; - } else if (other.sender == null || sender.getId() != other.sender.getId()) return false; + } else if (!sender.equals(other.sender)) return false; if (amount != other.amount) return false; if (remainder != other.remainder) return false; - if (!hash.equals(other.hash)) return false; if (!source.equals(other.source)) return false; if (!blockNumber.equals(other.blockNumber)) return false; return true; @@ -258,4 +199,13 @@ public String toString() { } } + @Override + public int compareTo(Transaction o) { + if (this.sender == null && o.sender != null) return -1; + if (this.sender != null && o.sender == null) return 1; + if (this.sender == null && o.sender == null) return 0; + int senderCompare = Integer.compare(this.sender.getId(), o.sender.getId()); + if (senderCompare != 0) return senderCompare; + return Integer.compare(this.getNumber(), o.getNumber()); + } } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/MainChain.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/MainChain.java index 6c16608..0d7a0c4 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/MainChain.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/MainChain.java @@ -16,22 +16,18 @@ public interface MainChain { */ public Sha256Hash commitAbstract(BlockAbstract abs); - /** - * Query the main chain for the presence of a block. - * - * @param block - the block to query for - * @return - true when present, false otherwise - */ - public default boolean isPresent(Block block) { - return isPresent(block.getHash()); - } - /** * Check whether the block (from a local chain) is on the main chain (in a form of BlockAbstract). - * @param blockHash the hash of the block + * @param block the block to check * @return true if there is a block abstract of the given block, false otherwise. */ - public boolean isPresent(Sha256Hash blockHash); + public boolean isPresent(Block block); + + /** + * @param block - the block to check + * @return - true if the given block is in the cache, false otherwise + */ + public boolean isInCache(Block block); /** * Initializes the tendermint chain. diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIClient.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIClient.java index e77cee4..5339267 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIClient.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIClient.java @@ -113,7 +113,7 @@ public JSONObject status() { try { return result.getJSONObject("result"); } catch (NullPointerException e) { - Log.log(Level.SEVERE, "The main chain does not respond. Did you start tendermint?"); + Log.log(Level.WARNING, "The main chain does not respond. Perhaps the tendermint is not yet running?"); return new JSONObject(); } } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIServer.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIServer.java index dc42998..66db06c 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIServer.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/ABCIServer.java @@ -1,7 +1,6 @@ package nl.tudelft.blockchain.scaleoutdistributedledger.model.mainchain.tendermint; import com.github.jtendermint.jabci.api.ABCIAPI; -import com.github.jtendermint.jabci.types.Types; import com.github.jtendermint.jabci.types.Types.CodeType; import com.github.jtendermint.jabci.types.Types.RequestBeginBlock; import com.github.jtendermint.jabci.types.Types.RequestCheckTx; diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChain.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChain.java index 3bb51f6..4bbe524 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChain.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChain.java @@ -1,7 +1,16 @@ package nl.tudelft.blockchain.scaleoutdistributedledger.model.mainchain.tendermint; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.logging.Level; + import com.github.jtendermint.jabci.socket.TSocket; -import lombok.Getter; + import nl.tudelft.blockchain.scaleoutdistributedledger.Application; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Block; import nl.tudelft.blockchain.scaleoutdistributedledger.model.BlockAbstract; @@ -9,12 +18,7 @@ import nl.tudelft.blockchain.scaleoutdistributedledger.model.mainchain.MainChain; import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; -import java.util.HashSet; -import java.util.List; -import java.util.Set; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.logging.Level; +import lombok.Getter; /** * Class implementing {@link MainChain} for a Tendermint chain. @@ -30,6 +34,9 @@ public final class TendermintChain implements MainChain { private TSocket socket; private ExecutorService threadPool; private Set cache; + private final Object cacheLock = new Object(); + private Map> extraCache = new HashMap<>(); + private Map superExtraCache = new HashMap<>(); @Getter private long currentHeight; @Getter @@ -39,15 +46,18 @@ public final class TendermintChain implements MainChain { * Create and start the ABCI app (server) to connect with Tendermint on the default port (46658). * Also uses (port - 1), which Tendermint should listen on for RPC (rpc.laddr) * @param genesisBlock - the genesis (initial) block for the entire system + * @param app - the application */ public TendermintChain(Block genesisBlock, Application app) { this(DEFAULT_ABCI_SERVER_PORT, genesisBlock, app); } + /** * Create and start the ABCI app (server) to connect with Tendermint on the given port. * Also uses (port - 1), which Tendermint should listen on for RPC (rpc.laddr) - * @param port - the port on which we run the server + * @param port - the port on which we run the server * @param genesisBlock - the genesis (initial) block for the entire system + * @param app - the application */ public TendermintChain(final int port, Block genesisBlock, Application app) { this.abciServerPort = port; @@ -96,14 +106,18 @@ private void initClient() { Log.log(Level.INFO, "Started ABCI Client on " + DEFAULT_ADDRESS + ":" + (abciServerPort - 1)); } + /** + * Performs the initial update of the cache. + */ protected void initialUpdateCache() { boolean updated = false; do { try { + Thread.sleep(1000); updateCacheBlocking(-1); updated = true; } catch (Exception e) { - int retryTime = 3; + int retryTime = 2; Log.log(Level.INFO, "Could not update cache on startup, trying again in " + retryTime + "s."); Log.log(Level.FINE, "", e); try { @@ -145,13 +159,25 @@ private void updateCacheBlocking(long height) { Log.log(Level.WARNING, "Could not get block at height " + i + ", perhaps the tendermint rpc is not (yet) running (or broken)"); return; } - for (BlockAbstract abs : abstractsAtCurrentHeight) { - cache.add(abs.getBlockHash()); + synchronized (cacheLock) { + for (BlockAbstract abs : abstractsAtCurrentHeight) { + cache.add(abs.getBlockHash()); + + //TODO Remove extra cache and super extra cache + int blockNum = abs.getBlockNumber(); + int owner = abs.getOwnerNodeId(); + Set setOfBlockNums = extraCache.getOrDefault(owner, new HashSet<>()); + setOfBlockNums.add(blockNum); + extraCache.put(owner, setOfBlockNums); + superExtraCache.putIfAbsent(owner + "-" + blockNum, abs.getBlockHash()); + } } } if (currentHeight < height) { Log.log(Level.FINE, "Successfully updated the Tendermint cache for node " + this.app.getLocalStore().getOwnNode().getId() + " from height " + currentHeight + " -> " + height + ", number of cached hashes of abstracts on main chain is now " + cache.size()); + Log.log(Level.FINE, "Node " + this.getApp().getLocalStore().getOwnNode().getId() + " current view of cache is " + this.getCurrentCache()); + Log.log(Level.FINE, "Node " + this.getApp().getLocalStore().getOwnNode().getId() + " current view of super cache is " + this.getSuperExtraCache()); } currentHeight = Math.max(currentHeight, height); // For concurrency reasons use the maximum } @@ -178,17 +204,30 @@ public Sha256Hash commitAbstract(BlockAbstract abs) { } @Override - public boolean isPresent(Sha256Hash blockHash) { - if (cache.contains(blockHash)) { - return true; - } else { - // We could miss some blocks in our cache, so update and wait for the results - updateCacheBlocking(-1); - return cache.contains(blockHash); - //TODO: We might want to check the actual main chain in the false case - // For when an abstract is in a block that is not yet closed by an ENDBLOCK - // This now works because the block size is 1 - } + public boolean isPresent(Block block) { + Sha256Hash blockHash = block.getHash(); + return cache.contains(blockHash); + +// //TODO IMPORTANT Decide what to do here +// if (cache.contains(blockHash)) { +// return true; +// } else { +//// Log.log(Level.INFO, "Node" + this.getApp().getLocalStore().getOwnNode() + " checking cache for " + block.getOwner().getId() + "-" + block.getNumber() + ": " + blockHash + ", the set is " + this.cache); +// // We could miss some blocks in our cache, so update and wait for the results +// //TODO We might not want to update here. The cache should be enough +// updateCacheBlocking(-1, true); +//// Log.log(Level.INFO, "Node" + this.getApp().getLocalStore().getOwnNode() + " did not find " + blockHash + ", so updated the cache and now the set is " + this.cache); +// return cache.contains(blockHash); +// //TODO: We might want to check the actual main chain in the false case +// // For when an abstract is in a block that is not yet closed by an ENDBLOCK +// // This now works because the block size is 1 +// } + } + + @Override + public boolean isInCache(Block block) { + Sha256Hash blockHash = block.getHash(); + return cache.contains(blockHash); } /** @@ -200,4 +239,16 @@ boolean addToCache(Sha256Hash genesisBlockHash) { return cache.add(genesisBlockHash); } + public Map> getCurrentCache() { + synchronized (cacheLock) { + return new HashMap<>(this.extraCache); + } + } + + public Map getSuperExtraCache() { + synchronized (cacheLock) { + return new HashMap<>(this.superExtraCache); + } + } + } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/CancellableInfiniteRunnable.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/CancellableInfiniteRunnable.java index a365c46..9f18eac 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/CancellableInfiniteRunnable.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/CancellableInfiniteRunnable.java @@ -66,6 +66,7 @@ public void run() { try { while (!isCancelled()) { + //Do action try { action.accept(t); } catch (InterruptedException ex) { @@ -74,6 +75,7 @@ public void run() { Log.log(Level.SEVERE, "Uncaught exception in action", ex); } + //Sleep try { Thread.sleep(sleepFunction.applyAsLong(t)); } catch (InterruptedException ex) { diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/Simulation.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/Simulation.java index 81262fd..e28645b 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/Simulation.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/Simulation.java @@ -6,6 +6,7 @@ import nl.tudelft.blockchain.scaleoutdistributedledger.Application; import nl.tudelft.blockchain.scaleoutdistributedledger.message.Message; import nl.tudelft.blockchain.scaleoutdistributedledger.message.StartTransactingMessage; +import nl.tudelft.blockchain.scaleoutdistributedledger.message.StopTransactingMessage; import nl.tudelft.blockchain.scaleoutdistributedledger.message.TransactionPatternMessage; import nl.tudelft.blockchain.scaleoutdistributedledger.message.UpdateNodesMessage; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Block; @@ -17,10 +18,8 @@ import nl.tudelft.blockchain.scaleoutdistributedledger.sockets.SocketClient; import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; +import java.util.*; +import java.util.concurrent.atomic.AtomicInteger; import java.util.logging.Level; /** @@ -44,6 +43,7 @@ public class Simulation { /** * Creates a new simulation. + * @param isMaster - if this simulation is the master */ public Simulation(boolean isMaster) { this.socketClient = new SocketClient(); @@ -79,23 +79,28 @@ public void runNodesLocally(Map nodes, Map ownN this.nodes = nodes; //Init the applications localApplications = new Application[ownNodes.size()]; - int counter = 0; + AtomicInteger counter = new AtomicInteger(0); + List startingThreads = new LinkedList<>(); for (Map.Entry nodeEntry : ownNodes.entrySet()) { - Node node = nodeEntry.getValue(); - int nodeNumber = nodeEntry.getKey(); - - Application app = new Application(true); - List addressesForThisNode = generateAddressesForNodeForTendermintP2P(nodeNumber, nodes); + startingThreads.add(new Thread(() -> { + Node node = nodeEntry.getValue(); + int nodeNumber = nodeEntry.getKey(); + + Application app = new Application(true); + List addressesForThisNode = generateAddressesForNodeForTendermintP2P(nodeNumber, nodes); + + try { + TendermintHelper.runTendermintNode(node.getPort(), addressesForThisNode, nodeNumber); + app.init(node.getPort(), genesisBlock.genesisCopy(), nodeToKeyPair.get(nodeNumber), ownNodes.get(nodeNumber)); + } catch (Exception ex) { + Log.log(Level.SEVERE, "Unable to initialize local node " + nodeNumber + " on port " + node.getPort() + "!", ex); + } - try { - TendermintHelper.runTendermintNode(node.getPort(), addressesForThisNode, nodeNumber); - app.init(node.getPort(), genesisBlock.genesisCopy(), nodeToKeyPair.get(nodeNumber), ownNodes.get(nodeNumber)); - } catch (Exception ex) { - Log.log(Level.SEVERE, "Unable to initialize local node " + nodeNumber + " on port " + node.getPort() + "!", ex); - } + localApplications[counter.getAndIncrement()] = app; + })); - localApplications[counter++] = app; } + startingThreads.stream().forEach(Thread::start); } private List generateAddressesForNodeForTendermintP2P(Integer i, Map nodes) { @@ -148,12 +153,10 @@ public void initialize() { } //Broadcast transaction pattern - if (this.isMaster) { - broadcastMessage(new TransactionPatternMessage(transactionPattern)); - } - + broadcastMessage(new TransactionPatternMessage(transactionPattern)); //Have everyone update their nodes list broadcastMessage(new UpdateNodesMessage()); + Log.log(Level.INFO, "[Simulation] Initialized"); state = SimulationState.INITIALIZED; @@ -168,9 +171,7 @@ public void initialize() { public void start() { checkState(SimulationState.INITIALIZED, "start"); - if (isMaster) { - broadcastMessage(new StartTransactingMessage()); - } + broadcastMessage(new StartTransactingMessage()); Log.log(Level.INFO, "[Simulation] Running"); state = SimulationState.RUNNING; } @@ -181,7 +182,9 @@ public void start() { */ public void stop() { checkState(SimulationState.RUNNING, "stop"); - Log.log(Level.INFO, "[Simulation] Stopped"); + + broadcastMessage(new StopTransactingMessage()); + Log.log(Level.INFO, "[Simulation] Stopping"); state = SimulationState.STOPPED; } @@ -213,6 +216,7 @@ protected void checkState(SimulationState expected, String msg) { * @param msg - the message to send */ public void broadcastMessage(Message msg) { + if (!isMaster) return; Log.log(Level.INFO, "[Simulation] Sending " + msg + " to all nodes..."); for (Node node : nodes.values()) { diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/tendermint/TendermintHelper.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/tendermint/TendermintHelper.java index 6406931..edec264 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/tendermint/TendermintHelper.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/tendermint/TendermintHelper.java @@ -8,13 +8,7 @@ import java.nio.file.Files; import java.nio.file.Paths; import java.text.SimpleDateFormat; -import java.util.ArrayList; -import java.util.Date; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Locale; -import java.util.Map; +import java.util.*; import java.util.logging.Level; import java.util.stream.Collectors; import java.util.stream.IntStream; @@ -71,8 +65,7 @@ public static Ed25519Key generatePrivValidatorFile(Ed25519Key keyPair, int nodeN try { final Process ps = rt.exec(script.toString(), envVariables); ps.waitFor(); - BufferedReader stdInput = new BufferedReader(new - InputStreamReader(ps.getInputStream())); + BufferedReader stdInput = new BufferedReader(new InputStreamReader(ps.getInputStream())); // read the output from the command String s; while ((s = stdInput.readLine()) != null) { @@ -114,15 +107,12 @@ public static Ed25519Key generatePrivValidatorFile(Ed25519Key keyPair, int nodeN keyPair = new Ed25519Key(privateKey, publicKey); } - try ( - BufferedWriter writer = Files.newBufferedWriter(Paths.get(nodeFilesLocation, "priv_validator.json")) - ) { + try (BufferedWriter writer = Files.newBufferedWriter(Paths.get(nodeFilesLocation, "priv_validator.json"))) { if (!ensureDirectoryExists(nodeFilesLocation)) { return null; } writer.write(privValidator.toString()); - writer.close(); } catch (IOException e) { Log.log(Level.WARNING, "Could not generate priv_validator.json due to IO Exception", e); return null; @@ -199,15 +189,12 @@ public static boolean generateGenesisFile(Date genesisTime, Map genesis.put("validators", validators); String nodeFilesLocation = getNodeFilesLocation(nodeNumber); - try ( - BufferedWriter writer = Files.newBufferedWriter(Paths.get(nodeFilesLocation, "genesis.json")) - ) { + try (BufferedWriter writer = Files.newBufferedWriter(Paths.get(nodeFilesLocation, "genesis.json"))) { if (!ensureDirectoryExists(nodeFilesLocation)) { return false; } writer.write(genesis.toString()); - writer.close(); } catch (IOException e) { Log.log(Level.WARNING, "Could not generate genesis.json due to IO exception.", e); return false; @@ -276,11 +263,11 @@ public static void runTendermintNode(int nodeBasePort, List peerAddresse } String nodeFilesLocation = getNodeFilesLocation(nodeNumber); //add arguments - script.append("--home ").append(nodeFilesLocation).append(" "); - script.append("--p2p.laddr=tcp://0.0.0.0:").append(nodeBasePort + 1).append(" "); - script.append("--rpc.laddr=tcp://0.0.0.0:").append(nodeBasePort + 2).append(" "); - script.append("--proxy_app=tcp://127.0.0.1:").append(nodeBasePort + 3).append(" "); - script.append("--moniker=Node").append(nodeNumber).append(" "); + script.append("--home ").append(nodeFilesLocation).append(' '); + script.append("--p2p.laddr=tcp://0.0.0.0:").append(nodeBasePort + 1).append(' '); + script.append("--rpc.laddr=tcp://0.0.0.0:").append(nodeBasePort + 2).append(' '); + script.append("--proxy_app=tcp://127.0.0.1:").append(nodeBasePort + 3).append(' '); + script.append("--moniker=Node").append(nodeNumber).append(' '); //add other seeds if (peerAddresses != null && !peerAddresses.isEmpty()) { @@ -319,31 +306,23 @@ private static boolean ensureDirectoryExists(String location) { */ private static void enableLogging(Process ps, String logPrefix) { Thread stdOutThread = new Thread(() -> { - try ( - BufferedReader stdInput = new BufferedReader(new - InputStreamReader(ps.getInputStream())) - ) { + try (BufferedReader stdInput = new BufferedReader(new InputStreamReader(ps.getInputStream()))) { // read the output from the command String s; while ((s = stdInput.readLine()) != null) { Log.log(Level.FINE, "[TM STDIN " + logPrefix + " ] " + s); } - stdInput.close(); } catch (Exception e) { e.printStackTrace(); } }); Thread stdErrThread = new Thread(() -> { - try ( - BufferedReader stdError = new BufferedReader(new - InputStreamReader(ps.getErrorStream())) - ) { + try (BufferedReader stdError = new BufferedReader(new InputStreamReader(ps.getErrorStream()))) { // read any errors from the attempted command String s; while ((s = stdError.readLine()) != null) { Log.log(Level.FINE, "[TM STDERROR " + logPrefix + " ] " + s); } - stdError.close(); } catch (Exception e) { e.printStackTrace(); } @@ -361,11 +340,13 @@ private static void enableLogging(Process ps, String logPrefix) { public static Block generateGenesisBlock(long amount, Map nodeList) { List initialTransactions = new ArrayList<>(); for (int i = 0; i < nodeList.size(); i++) { - Transaction t = new Transaction(i, null, nodeList.get(i), amount, 0, new HashSet<>(0)); + Transaction t = new Transaction(i, null, nodeList.get(i), amount, 0, new TreeSet<>()); initialTransactions.add(t); } - return new Block(0, null, initialTransactions); + Block block = new Block(0, null, initialTransactions); + block.setNextCommittedBlock(block); + return block; } /** diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/ITransactionPattern.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/ITransactionPattern.java index 9611ae1..34bc172 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/ITransactionPattern.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/ITransactionPattern.java @@ -1,9 +1,7 @@ package nl.tudelft.blockchain.scaleoutdistributedledger.simulation.transactionpattern; -import java.io.Serializable; -import java.util.logging.Level; - import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; +import nl.tudelft.blockchain.scaleoutdistributedledger.SimulationMain; import nl.tudelft.blockchain.scaleoutdistributedledger.TransactionCreator; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Block; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Chain; @@ -13,6 +11,9 @@ import nl.tudelft.blockchain.scaleoutdistributedledger.simulation.CancellableInfiniteRunnable; import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; +import java.io.Serializable; +import java.util.logging.Level; + /** * Interface for a transaction pattern. */ @@ -47,33 +48,36 @@ public interface ITransactionPattern extends Serializable { * @throws InterruptedException - if the action is interrupted */ public default void doAction(LocalStore localStore) throws InterruptedException { - Log.log(Level.FINER, "Start doAction of node " + localStore.getOwnNode().getId()); + Log.log(Level.FINER, "Start doAction", localStore.getOwnNode().getId()); + OwnNode ownNode = localStore.getOwnNode(); + int ownNodeId = ownNode.getId(); + + // Make sure we have some room + if (localStore.getApplication().getTransactionSender().blocksWaiting() >= SimulationMain.MAX_BLOCKS_PENDING) { + Log.log(Level.INFO, "Too many blocks pending, skipping transaction creation!", ownNodeId); + return; + } //Select receiver and amount long amount = selectAmount(localStore); if (amount == -1) { - Log.log(Level.INFO, "Not enough money to make transaction!"); + Log.log(Level.INFO, "Not enough money to make transaction!", ownNodeId); return; } - + Node receiver = selectNode(localStore); - - OwnNode ownNode = localStore.getOwnNode(); - Block newBlock; - Log.log(Level.FINE, "Going to make transaction: $ " + amount + " from " + ownNode.getId() + " -> " + receiver.getId()); - synchronized (ownNode.getChain()) { - //Create the transaction - TransactionCreator creator = new TransactionCreator(localStore, receiver, amount); - Transaction transaction = creator.createTransaction(); + Log.log(Level.FINE, "Going to make transaction: $ " + amount + " from " + ownNodeId + " -> " + receiver.getId()); - //TODO how many transactions do we put in one block? - //Add block to local chain - newBlock = ownNode.getChain().appendNewBlock(); - newBlock.addTransaction(transaction); - } - - //Ensure that the block is sent at some point - localStore.getApplication().getTransactionSender().scheduleBlockSending(newBlock); + //TODO IMPORTANT Removed synchronization on own chain. Transaction creator should be safe. + //Create the transaction + TransactionCreator creator = new TransactionCreator(localStore, receiver, amount); + Transaction transaction = creator.createTransaction(); + + //TODO how many transactions do we put in one block? + //Add block to local chain + Block newBlock = ownNode.getChain().appendNewBlock(); + newBlock.addTransaction(transaction); + Log.log(Level.FINE, "Node " + ownNodeId + " added transaction " + transaction.getNumber() + " in block " + newBlock.getNumber()); //Check if we want to commit the new block, and commit it if we do. commitBlocks(localStore, false); @@ -97,16 +101,28 @@ public default boolean shouldCommitBlocks(Block lastBlock, Block lastCommitted) */ public default void commitBlocks(LocalStore localStore, boolean force) throws InterruptedException { Chain ownChain = localStore.getOwnNode().getChain(); - synchronized (ownChain) { - Block lastBlock = ownChain.getLastBlock(); - Block lastCommitted = ownChain.getLastCommittedBlock(); - - //Don't commit if we don't have anything to commit - if (lastBlock == lastCommitted) return; - - if (force || shouldCommitBlocks(lastBlock, lastCommitted)) { - lastBlock.commit(localStore); - } + + //TODO IMPORTANT Removed synchronization on own chain + Block lastBlock = ownChain.getLastBlock(); + Block lastCommitted = ownChain.getLastCommittedBlock(); + + //Don't commit if we don't have anything to commit + if (lastBlock == lastCommitted) return; + + if (force || shouldCommitBlocks(lastBlock, lastCommitted)) { + lastBlock.commit(localStore); + } + } + + /** + * Commits extra empty blocks. + * @param localStore - the local store + */ + public default void commitExtraEmpty(LocalStore localStore) { + Chain ownChain = localStore.getOwnNode().getChain(); + for (int i = 0; i < SimulationMain.REQUIRED_COMMITS + 1; i++) { + Block block = ownChain.appendNewBlock(); + block.commit(localStore); } } @@ -123,6 +139,7 @@ public default void commitBlocks(LocalStore localStore, boolean force) throws In public default void onStop(LocalStore localStore) { try { commitBlocks(localStore, true); + commitExtraEmpty(localStore); } catch (InterruptedException ex) { Log.log(Level.SEVERE, "Interrupted while committing blocks!"); } catch (Exception ex) { diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/OnlyNodeZeroTransactionPattern.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/OnlyNodeZeroTransactionPattern.java new file mode 100644 index 0000000..2dd4c70 --- /dev/null +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/OnlyNodeZeroTransactionPattern.java @@ -0,0 +1,30 @@ +package nl.tudelft.blockchain.scaleoutdistributedledger.simulation.transactionpattern; + +import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; + +/** + * Transaction pattern where only node 0 makes transactions. + * Only used for debugging. + */ +public class OnlyNodeZeroTransactionPattern extends UniformRandomTransactionPattern { + private static final long serialVersionUID = 1L; + + /** + * @param minAmount - the minimum amount of money to send + * @param maxAmount - the maximum amount of money to send + * @param minWaitTime - the minimum wait time between making transactions + * @param maxWaitTime - the maximum wait time between making transactions + * @param commitEvery - commit every x blocks + */ + public OnlyNodeZeroTransactionPattern(int minAmount, int maxAmount, int minWaitTime, int maxWaitTime, int commitEvery) { + super(minAmount, maxAmount, minWaitTime, maxWaitTime, commitEvery); + } + + @Override + public long selectAmount(LocalStore localStore) { + //Only let node 0 make transactions + if (localStore.getOwnNode().getId() != 0) return -1; + + return super.selectAmount(localStore); + } +} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/UniformRandomTransactionPattern.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/UniformRandomTransactionPattern.java index a73eb3d..3c96c6f 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/UniformRandomTransactionPattern.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/simulation/transactionpattern/UniformRandomTransactionPattern.java @@ -49,7 +49,11 @@ public String getName() { @Override public long timeUntilNextAction(LocalStore localStore) { - return minWaitTime + random.nextInt(maxWaitTime - minWaitTime); + if (maxWaitTime == minWaitTime) { + return minWaitTime; + } else { + return minWaitTime + random.nextInt(maxWaitTime - minWaitTime); + } } @Override @@ -70,8 +74,12 @@ public Node selectNode(LocalStore localStore) { public long selectAmount(LocalStore localStore) { long available = localStore.getAvailableMoney(); if (available == 0 || minAmount > available) return -1; - - long amount = (long) minAmount + random.nextInt(maxAmount - minAmount); + long amount; + if (maxAmount == minAmount) { + amount = minAmount; + } else { + amount = (long) minAmount + random.nextInt(maxAmount - minAmount); + } return Math.min(amount, available); } diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketClient.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketClient.java index 6246aeb..923b684 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketClient.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketClient.java @@ -1,18 +1,24 @@ package nl.tudelft.blockchain.scaleoutdistributedledger.sockets; +import java.util.HashMap; +import java.util.logging.Level; + +import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; +import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; + import io.netty.bootstrap.Bootstrap; -import io.netty.channel.*; +import io.netty.channel.Channel; +import io.netty.channel.ChannelFuture; +import io.netty.channel.ChannelFutureListener; +import io.netty.channel.ChannelInitializer; +import io.netty.channel.ChannelPipeline; +import io.netty.channel.EventLoopGroup; import io.netty.channel.nio.NioEventLoopGroup; import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.nio.NioSocketChannel; import io.netty.handler.codec.serialization.ClassResolvers; import io.netty.handler.codec.serialization.ObjectDecoder; import io.netty.handler.codec.serialization.ObjectEncoder; -import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; -import nl.tudelft.blockchain.scaleoutdistributedledger.utils.Log; - -import java.util.HashMap; -import java.util.logging.Level; /** * Socket client. @@ -67,6 +73,7 @@ public void shutdown() { * @param node - the node to send the message to. * @param msg - the message object to send * @return - whether the message was sent successfully + * @throws InterruptedException - If message sending is interrupted. */ public boolean sendMessage(Node node, Object msg) throws InterruptedException { Channel channel = connections.get(node); diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketServerHandler.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketServerHandler.java index dac702c..b0cf3a3 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketServerHandler.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/sockets/SocketServerHandler.java @@ -27,8 +27,7 @@ public SocketServerHandler(LocalStore localStore) { @Override public void channelRead(ChannelHandlerContext ctx, Object msg) { - Log.log(Level.FINE, "Node " + localStore.getOwnNode().getId() + " Server: received message: " + msg); - if(msg instanceof Message) { + if (msg instanceof Message) { ((Message) msg).handle(localStore); } else { Log.log(Level.SEVERE, "Invalid message, not a message instance"); diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/utils/AppendOnlyArrayList.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/utils/AppendOnlyArrayList.java new file mode 100644 index 0000000..acdcfb0 --- /dev/null +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/utils/AppendOnlyArrayList.java @@ -0,0 +1,323 @@ +package nl.tudelft.blockchain.scaleoutdistributedledger.utils; + +import java.util.AbstractList; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Iterator; +import java.util.List; +import java.util.ListIterator; +import java.util.NoSuchElementException; +import java.util.RandomAccess; +import java.util.function.Predicate; +import java.util.function.UnaryOperator; + +/** + * An ArrayList to which you can only append. + * The iterators of this class all reflect the list from the moment of creation. Any elements + * appended to the list after the iterator was created will not be reflected. + * + * @param - the type of elements in the list + */ +public class AppendOnlyArrayList extends ArrayList { + private static final long serialVersionUID = 1L; + + @Override + public void add(int index, E element) { + throw new UnsupportedOperationException(); + } + + @Override + public boolean addAll(int index, Collection c) { + throw new UnsupportedOperationException(); + } + + @Override + public E set(int index, E element) { + throw new UnsupportedOperationException(); + } + + @Override + public boolean remove(Object o) { + throw new UnsupportedOperationException(); + } + + @Override + public E remove(int index) { + throw new UnsupportedOperationException(); + } + + @Override + public boolean removeAll(Collection c) { + throw new UnsupportedOperationException(); + } + + @Override + public boolean removeIf(Predicate filter) { + throw new UnsupportedOperationException(); + } + + @Override + public void replaceAll(UnaryOperator operator) { + throw new UnsupportedOperationException(); + } + + @Override + public boolean retainAll(Collection c) { + throw new UnsupportedOperationException(); + } + + @Override + protected void removeRange(int fromIndex, int toIndex) { + throw new UnsupportedOperationException(); + } + + @Override + public void clear() { + throw new UnsupportedOperationException(); + } + + @Override + public Iterator iterator() { + return listIterator(); + } + + @Override + public ListIterator listIterator() { + return new AppendOnlyArrayListIterator(0); + } + + @Override + public ListIterator listIterator(int index) { + return new AppendOnlyArrayListIterator(index); + } + + /** + * This method returns a list iterator that provides a view from index 0 to the given size, + * starting at the given index. + * @param index - the index where the list iterator starts + * @param size - the size that the list iterator will use as maximum + * @return - a new List Iterator starting at the given index + */ + public ListIterator listIterator(int index, int size) { + return new AppendOnlyArrayListIterator(index, size); + } + + /** + * List iterator for this class. + */ + private class AppendOnlyArrayListIterator implements ListIterator { + private int index; + private int size; + + /** + * @param index - the index to start at + */ + AppendOnlyArrayListIterator(int index) { + this(index, size()); + } + + /** + * @param index - the index to start at + * @param size - the size to use + */ + AppendOnlyArrayListIterator(int index, int size) { + this.index = index; + this.size = size; + + if (index < 0 || index > size) { + throw new IndexOutOfBoundsException("Index: " + index); + } + } + + @Override + public boolean hasNext() { + return index != size; + } + + @Override + public E next() { + int i = index; + if (i >= size) throw new NoSuchElementException(); + + E element = get(i); + index = i + 1; + return element; + } + + @Override + public boolean hasPrevious() { + return index != 0; + } + + @Override + public E previous() { + int i = index - 1; + if (i < 0) throw new NoSuchElementException(); + + E element = get(i); + index = i; + return element; + } + + @Override + public int nextIndex() { + return index; + } + + @Override + public int previousIndex() { + return index - 1; + } + + @Override + public void remove() { + throw new UnsupportedOperationException(); + } + + @Override + public void set(E e) { + throw new UnsupportedOperationException(); + } + + @Override + public void add(E e) { + throw new UnsupportedOperationException(); + } + } + + @Override + public List subList(int fromIndex, int toIndex) { + if (fromIndex < 0) throw new IndexOutOfBoundsException("fromIndex = " + fromIndex); + if (toIndex > size()) throw new IndexOutOfBoundsException("toIndex = " + toIndex); + if (fromIndex > toIndex) throw new IllegalArgumentException("fromIndex(" + fromIndex + ") > toIndex(" + toIndex + ")"); + return new SubList(0, fromIndex, toIndex); + } + + /** + * Sublist implementation for AppendOnlyArrayList. + */ + private class SubList extends AbstractList implements RandomAccess { + private final int offset; + private final int size; + + SubList(int offset, int fromIndex, int toIndex) { + this.offset = offset + fromIndex; + this.size = toIndex - fromIndex; + } + + @Override + public E set(int index, E e) { + throw new UnsupportedOperationException(); + } + + @Override + public E get(int index) { + if (index < 0 || index >= this.size) throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + this.size); + return AppendOnlyArrayList.this.get(offset + index); + } + + @Override + public int size() { + return this.size; + } + + @Override + public void add(int index, E e) { + throw new UnsupportedOperationException(); + } + + @Override + public E remove(int index) { + throw new UnsupportedOperationException(); + } + + @Override + protected void removeRange(int fromIndex, int toIndex) { + throw new UnsupportedOperationException(); + } + + @Override + public boolean addAll(Collection c) { + throw new UnsupportedOperationException(); + } + + @Override + public boolean addAll(int index, Collection c) { + throw new UnsupportedOperationException(); + } + + @Override + public Iterator iterator() { + return listIterator(); + } + + @Override + public ListIterator listIterator(final int index) { + if (index < 0 || index > this.size) throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + this.size); + final int offset = this.offset; + + return new ListIterator() { + int cursor = index; + + @Override + public boolean hasNext() { + return cursor != SubList.this.size; + } + + @Override + public E next() { + int i = cursor; + if (i >= SubList.this.size) throw new NoSuchElementException(); + cursor = i + 1; + return AppendOnlyArrayList.this.get(offset + i); + } + + @Override + public boolean hasPrevious() { + return cursor != 0; + } + + @Override + public E previous() { + int i = cursor - 1; + if (i < 0) throw new NoSuchElementException(); + cursor = i; + return AppendOnlyArrayList.this.get(offset + i); + } + + @Override + public int nextIndex() { + return cursor; + } + + @Override + public int previousIndex() { + return cursor - 1; + } + + @Override + public void remove() { + throw new UnsupportedOperationException(); + } + + @Override + public void set(E e) { + throw new UnsupportedOperationException(); + } + + @Override + public void add(E e) { + throw new UnsupportedOperationException(); + } + }; + } + + @Override + public List subList(int fromIndex, int toIndex) { + if (fromIndex < 0) throw new IndexOutOfBoundsException("fromIndex = " + fromIndex); + if (toIndex > size) throw new IndexOutOfBoundsException("toIndex = " + toIndex); + if (fromIndex > toIndex) throw new IllegalArgumentException("fromIndex(" + fromIndex + ") > toIndex(" + toIndex + ")"); + return new SubList(offset, fromIndex, toIndex); + } + } +} diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/utils/Log.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/utils/Log.java index 79149bb..31478b4 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/utils/Log.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/utils/Log.java @@ -79,6 +79,16 @@ public static void log(Level level, String str) { Logger.getLogger(getCallerClassName()).log(level, str); } + /** + * Logs the given message for the given node id. + * @param level - level to log at + * @param str - message to log + * @param nodeId - the id of the node + */ + public static void log(Level level, String str, int nodeId) { + Logger.getLogger(getCallerClassName()).log(level, "[" + nodeId + "] " + str); + } + /** * Logs the given debug message. * @param msg - the message diff --git a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/validation/Verification.java b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/validation/Verification.java index f5c4344..60829f1 100644 --- a/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/validation/Verification.java +++ b/src/main/java/nl/tudelft/blockchain/scaleoutdistributedledger/validation/Verification.java @@ -96,8 +96,6 @@ private void checkMoney(Transaction transaction) throws ValidationException { * @throws ValidationException - If we detect double spending. */ private void checkDoubleSpending(Transaction transaction, Proof proof) throws ValidationException { - //TODO [PERFORMANCE]: We will revalidate the same chainview for every source that we check, even though is is not necessary. - //TODO [PERFORMANCE]: We might want some kind of caching or other mechanism to prevent this. ChainView chainView = proof.getChainView(transaction.getSender()); for (Block block : chainView) { boolean found = false; diff --git a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreatorTest.java b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreatorTest.java index 96a83a4..90fdf06 100644 --- a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreatorTest.java +++ b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/TransactionCreatorTest.java @@ -3,14 +3,16 @@ import static org.junit.Assert.*; import java.util.Arrays; -import java.util.Collections; -import java.util.HashSet; import java.util.Map; import java.util.Set; +import java.util.TreeSet; import org.junit.Before; import org.junit.Test; +import nl.tudelft.blockchain.scaleoutdistributedledger.exceptions.NotEnoughMoneyException; +import nl.tudelft.blockchain.scaleoutdistributedledger.model.Block; +import nl.tudelft.blockchain.scaleoutdistributedledger.model.MetaKnowledge; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Node; import nl.tudelft.blockchain.scaleoutdistributedledger.model.OwnNode; import nl.tudelft.blockchain.scaleoutdistributedledger.model.Transaction; @@ -31,7 +33,7 @@ public class TransactionCreatorTest { @Before public void setUp() { ownNode = new OwnNode(0); - + //TestHelper.generateGenesis(ownNode, 5, 100); localStore = new LocalStore(ownNode, null, null, false); nodes = localStore.getNodes(); } @@ -44,6 +46,7 @@ public void setUp() { public void createNodes(int from, int to) { for (int i = from; i <= to; i++) { nodes.put(i, new Node(i)); + localStore.getNewTransactionId(); } } @@ -62,9 +65,9 @@ public Node getNode(int id) { * @param chains - the chains to add to the meta knowledge */ public void addMetaKnowledge(Node node, int... chains) { - Map meta = node.getMetaKnowledge(); + MetaKnowledge meta = node.getMetaKnowledge(); for (int chain : chains) { - meta.put(getNode(chain), 0); + meta.updateLastKnownBlockNumber(chain, 2); } } @@ -78,8 +81,16 @@ public void addMetaKnowledge(Node node, int... chains) { * @return the transaction */ public Transaction addUnspent(Node sender, Node receiver, long amount, long remainder) { - Transaction genesis = new Transaction(0, null, sender, amount + remainder, 0, new HashSet<>()); - Transaction transaction = new Transaction(localStore.getNewTransactionId(), sender, receiver, amount, remainder, Collections.singleton(genesis)); + //Create genesis block + Transaction genesis = new Transaction(0, null, sender, amount + remainder, 0, new TreeSet<>()); + Block genesisBlock = new Block(0, null, Arrays.asList(genesis)); + sender.getChain().getBlocks().add(genesisBlock); + + //Create transaction and block + Transaction transaction = new Transaction(localStore.getNewTransactionId(), sender, receiver, amount, remainder, genesis); + Block block1 = new Block(1, sender, Arrays.asList(transaction)); + sender.getChain().getBlocks().add(block1); + localStore.addUnspentTransaction(transaction); return transaction; } @@ -111,7 +122,7 @@ public Transaction addRemainderMoney(Node to, long remainder) { */ public void checkTransactionSources(Transaction transaction, Transaction... expectedSources) { Set actualSet = transaction.getSource(); - Set expectedSet = new HashSet<>(); + Set expectedSet = new TreeSet<>(); expectedSet.addAll(Arrays.asList(expectedSources)); assertEquals(expectedSet, actualSet); diff --git a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainViewTest.java b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainViewTest.java index 7140cc7..ef9b444 100644 --- a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainViewTest.java +++ b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ChainViewTest.java @@ -10,6 +10,8 @@ import org.junit.Before; import org.junit.Test; +import nl.tudelft.blockchain.scaleoutdistributedledger.utils.AppendOnlyArrayList; + /** * Test class for {@link ChainView}. */ @@ -17,7 +19,7 @@ public class ChainViewTest { private ChainView chainview; private Chain chainMock; private Node nodeMock; - private List blocks; + private AppendOnlyArrayList blocks; private List updatedBlocks; /** @@ -25,14 +27,14 @@ public class ChainViewTest { */ @Before public void setUp() { - blocks = new ArrayList(); + blocks = new AppendOnlyArrayList(); updatedBlocks = new ArrayList(); nodeMock = mock(Node.class); chainMock = mock(Chain.class); when(chainMock.getBlocks()).thenReturn(blocks); - chainview = new ChainView(chainMock, updatedBlocks); + chainview = new ChainView(chainMock, updatedBlocks, false); } /** @@ -65,6 +67,7 @@ public void testIsValid_Valid() { addBlock(0, true); addBlock(1, true); addBlock(2, false); + chainview.isValid(); assertTrue(chainview.isValid()); } @@ -76,6 +79,7 @@ public void testIsValid_Valid() { public void testIsValid_EmptyUpdate() { addBlock(0, true); addBlock(1, true); + chainview.isValid(); assertTrue(chainview.isValid()); } @@ -88,6 +92,7 @@ public void testIsValid_Missing() { addBlock(0, true); addBlock(1, true); addBlock(3, false); + chainview.isValid(); assertFalse(chainview.isValid()); } @@ -101,6 +106,7 @@ public void testIsValid_Gap() { addBlock(1, true); addBlock(2, false); addBlock(4, false); + chainview.isValid(); assertFalse(chainview.isValid()); } @@ -116,6 +122,7 @@ public void testIsValid_OverlapCorrect() { addBlock(1, false); addBlock(2, false); addBlock(3, false); + chainview.isValid(); assertTrue(chainview.isValid()); } @@ -130,6 +137,7 @@ public void testIsValid_OverlapIncorrect() { addBlock(2, true); addBlock(1, false); addBlock(3, false); + chainview.isValid(); assertFalse(chainview.isValid()); } @@ -141,6 +149,7 @@ public void testIsValid_OverlapIncorrect() { public void testGetBlock() { addBlock(0, true); addBlock(1, true); + chainview.isValid(); assertEquals(0, chainview.getBlock(0).getNumber()); assertEquals(1, chainview.getBlock(1).getNumber()); @@ -155,6 +164,7 @@ public void testGetBlock_Overlap() { addBlock(0, true); Block good1 = addBlock(1, true); addBlock(1, false); + chainview.isValid(); assertSame(good1, chainview.getBlock(1)); } @@ -168,10 +178,24 @@ public void testGetBlock_UpdatePart() { addBlock(0, true); addBlock(1, true); addBlock(2, false); + chainview.isValid(); assertEquals(2, chainview.getBlock(2).getNumber()); } + /** + * Test for {@link ChainView#size()}. + */ + @Test + public void testSize() { + addBlock(0, true); + addBlock(1, true); + addBlock(1, false); + addBlock(2, false); + + assertEquals(3, chainview.size()); + } + /** * Test for {@link ChainView#iterator()}, for next and nextIndex. */ @@ -179,6 +203,7 @@ public void testGetBlock_UpdatePart() { public void testIteratorNextIndex() { addBlock(0, true); addBlock(1, false); + chainview.isValid(); ListIterator it = chainview.iterator(); assertEquals(0, it.nextIndex()); @@ -196,6 +221,7 @@ public void testListIteratorNextIndex() { addBlock(0, true); addBlock(1, false); addBlock(2, false); + chainview.isValid(); ListIterator it = chainview.listIterator(2); assertEquals(2, it.nextIndex()); @@ -212,6 +238,7 @@ public void testListIteratorNextIndex() { public void testIteratorPreviousIndex() { addBlock(0, true); addBlock(1, false); + chainview.isValid(); ListIterator it = chainview.iterator(); assertEquals(-1, it.previousIndex()); @@ -228,6 +255,7 @@ public void testIteratorPreviousIndex() { public void testListIteratorNextPrevious() { addBlock(0, true); addBlock(1, false); + chainview.isValid(); ListIterator it = chainview.listIterator(); assertEquals(0, it.next().getNumber()); @@ -243,6 +271,7 @@ public void testListIteratorPreviousIndex() { addBlock(0, true); addBlock(1, false); addBlock(2, false); + chainview.isValid(); ListIterator it = chainview.listIterator(2); assertEquals(1, it.previousIndex()); diff --git a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ProofTest.java b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ProofTest.java index 2272780..d38a4ab 100644 --- a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ProofTest.java +++ b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/ProofTest.java @@ -5,15 +5,16 @@ import java.security.KeyPair; import java.util.Arrays; -import java.util.HashSet; import java.util.List; import java.util.Map; +import java.util.TreeSet; import org.junit.Before; import org.junit.Test; import nl.tudelft.blockchain.scaleoutdistributedledger.Application; import nl.tudelft.blockchain.scaleoutdistributedledger.LocalStore; +import nl.tudelft.blockchain.scaleoutdistributedledger.ProofConstructor; import nl.tudelft.blockchain.scaleoutdistributedledger.test.utils.TestHelper; /** @@ -36,7 +37,7 @@ public void setUp() throws Exception { ownNode.setPrivateKey(pair.getPrivate().getEncoded()); ownNode.setPublicKey(pair.getPublic().getEncoded()); - genesisBlock = TestHelper.generateGenesis(ownNode, 3, 1000); + genesisBlock = TestHelper.generateGenesis(ownNode, 4, 1000); Map nodes = TestHelper.getNodeList(genesisBlock); appMock = mock(Application.class); @@ -59,30 +60,91 @@ public Transaction basicScenario() { Chain ownChain = ownNode.getChain(); //3: 0 --> 1, source = GENESIS 0 - HashSet source0to1 = new HashSet<>(); + TreeSet source0to1 = new TreeSet<>(); source0to1.add(ownChain.getGenesisTransaction()); - Transaction transaction0to1 = new Transaction(3, ownNode, node1, 100, 900, source0to1); + Transaction transaction0to1 = new Transaction(4, ownNode, node1, 100, 900, source0to1); Block block1node0 = ownChain.appendNewBlock(); block1node0.addTransaction(transaction0to1); //3: 1 --> 0, source = [3: 0 --> 1] - HashSet source1to0 = new HashSet<>(); + TreeSet source1to0 = new TreeSet<>(); source1to0.add(transaction0to1); - Transaction transaction1to0 = new Transaction(3, node1, ownNode, 100, 0, source1to0); + Transaction transaction1to0 = new Transaction(4, node1, ownNode, 100, 0, source1to0); Block block1node1 = new Block(genesisBlock, node1); block1node1.addTransaction(transaction1to0); node1.getChain().getBlocks().add(block1node1); + block1node1.setNextCommittedBlock(block1node1); //4: 0 --> 2, source = [3: 1 --> 0] - HashSet source0to2 = new HashSet<>(); + TreeSet source0to2 = new TreeSet<>(); source0to2.add(transaction1to0); - Transaction transaction0to2 = new Transaction(4, ownNode, node2, 100, 0, source0to2); + Transaction transaction0to2 = new Transaction(5, ownNode, node2, 100, 0, source0to2); Block block2node0 = ownChain.appendNewBlock(); block2node0.addTransaction(transaction0to2); //commit block 2 of node 0 block2node0.commit(storeSpy); + block1node0.setNextCommittedBlock(block2node0); + block2node0.setNextCommittedBlock(block2node0); + + return transaction0to2; + } + + /** + * Creates the following scenario. + *
+	 * 3: 0 --> 1, source = GENESIS 0
+	 * 3: 1 --> 0, source = [3: 0 --> 1]
+	 * 4: 0 --> 2, source = [3: 1 --> 0]
+	 * 
+ * @return - a transaction from node 0 to node 2 + */ + public Transaction basicScenario2() { + Node node1 = storeSpy.getNode(1); + Node node2 = storeSpy.getNode(2); + Node node3 = storeSpy.getNode(3); + Chain ownChain = ownNode.getChain(); + Chain node1Chain = node1.getChain(); + + //3: 0 --> 1, source = GENESIS 0 + TreeSet source0to1 = new TreeSet<>(); + source0to1.add(ownChain.getGenesisTransaction()); + Transaction transaction0to1 = new Transaction(4, ownNode, node1, 100, 900, source0to1); + Block block1node0 = ownChain.appendNewBlock(); + block1node0.addTransaction(transaction0to1); + + //3: 1 --> 0, source = [GENESIS 1] + TreeSet source1to0 = new TreeSet<>(); + source1to0.add(node1Chain.getGenesisTransaction()); + Transaction transaction1to0 = new Transaction(4, node1, ownNode, 100, 0, source1to0); + Block block1node1 = new Block(genesisBlock, node1); + block1node1.addTransaction(transaction1to0); + node1.getChain().getBlocks().add(block1node1); + block1node1.setNextCommittedBlock(block1node1); + + //5: 0 --> 3, source = [4: 0 --> 1] + TreeSet source0to3 = new TreeSet<>(); + source0to3.add(transaction0to1); + Transaction transaction0to3 = new Transaction(5, ownNode, node3, 100, 0, source0to3); + Block block2node0 = ownChain.appendNewBlock(); + block2node0.addTransaction(transaction0to3); + + //6: 0 --> 2, source = [3: 1 --> 0] + TreeSet source0to2 = new TreeSet<>(); + source0to2.add(transaction0to3); + source0to2.add(transaction1to0); + Transaction transaction0to2 = new Transaction(6, ownNode, node2, 100, 0, source0to2); + Block block3node0 = ownChain.appendNewBlock(); + block3node0.addTransaction(transaction0to2); + + //commit block 3 of node 0 + block3node0.commit(storeSpy); + + block1node0.setNextCommittedBlock(block3node0); + block2node0.setNextCommittedBlock(block3node0); + block3node0.setNextCommittedBlock(block3node0); + return transaction0to2; } @@ -98,7 +160,7 @@ public void testCreateProof() { Transaction transaction = basicScenario(); - Proof proof = Proof.createProof(storeSpy, transaction); + Proof proof = new ProofConstructor(transaction).constructProof(); //Proof should contain all the blocks of our own chain List ownNodeUpdates = proof.getChainUpdates().get(ownNode); @@ -122,10 +184,10 @@ public void testCreateProof2() { Transaction transaction = basicScenario(); //Node 2 knows about the genesis of node 1 already - node2.getMetaKnowledge().put(node1, 0); + node2.getMetaKnowledge().updateLastKnownBlockNumber(node1, 0); //send to node 2 - Proof proof = Proof.createProof(storeSpy, transaction); + Proof proof = new ProofConstructor(transaction).constructProof(); //Proof should contain the first 3 blocks of the ownNode List ownNodeUpdates = proof.getChainUpdates().get(ownNode); @@ -149,11 +211,11 @@ public void testCreateProof3() { Transaction transaction = basicScenario(); //Node 2 knows about block 1 of node 0 and of block 1 of node 1 - node2.getMetaKnowledge().put(ownNode, 1); - node2.getMetaKnowledge().put(node1, 1); + node2.getMetaKnowledge().updateLastKnownBlockNumber(ownNode, 1); + node2.getMetaKnowledge().updateLastKnownBlockNumber(node1, 1); //send to node 2 - Proof proof = Proof.createProof(storeSpy, transaction); + Proof proof = new ProofConstructor(transaction).constructProof(); //Proof should contain only the last block of node 0 List ownNodeUpdates = proof.getChainUpdates().get(ownNode); @@ -164,4 +226,34 @@ public void testCreateProof3() { assertFalse(proof.getChainUpdates().containsKey(node1)); } + /** + * Test method for {@link Proof#createProof}. + */ + @Test + public void testCreateProof4() { + Node node1 = storeSpy.getNode(1); + Node node3 = storeSpy.getNode(3); + Chain ownChain = ownNode.getChain(); + Chain node1Chain = node1.getChain(); + + Transaction transaction = basicScenario2(); + + //Node 2 knows about block 1 of node 0 and of block 1 of node 1 + + node1.getMetaKnowledge().updateLastKnownBlockNumber(ownNode, 1); + node3.getMetaKnowledge().updateLastKnownBlockNumber(ownNode, 2); + + //send to node 2 + Proof proof = new ProofConstructor(transaction).constructProof(); + + //Proof should contain only the last block of node 0 + List ownNodeUpdates = proof.getChainUpdates().get(ownNode); + List expectedOwnNodeUpdates = ownChain.getBlocks(); + assertEquals(expectedOwnNodeUpdates, ownNodeUpdates); + + //No blocks of node1 should be included + List node1Updates = proof.getChainUpdates().get(node1); + List expectedChain1Updates = Arrays.asList(node1Chain.getGenesisBlock(), node1Chain.getBlocks().get(1)); + assertEquals(expectedChain1Updates, node1Updates); + } } diff --git a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/SerializationTest.java b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/SerializationTest.java index 6fe8199..924f1f9 100644 --- a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/SerializationTest.java +++ b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/SerializationTest.java @@ -9,11 +9,7 @@ import org.junit.Test; import java.io.IOException; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Set; +import java.util.*; import static org.junit.Assert.*; @@ -82,7 +78,7 @@ public void setUp() { // Add Proof this.proof = new Proof(this.transaction); // Manually change the chainUpdates - List listBlockUpdate = this.bobNode.getChain().getBlocks(); + List listBlockUpdate = new ArrayList<>(this.bobNode.getChain().getBlocks()); listBlockUpdate.remove(0); this.proof.getChainUpdates().put(this.bobNode, listBlockUpdate); } @@ -129,7 +125,7 @@ public LocalStore setupLocalStore(Node node) { */ public Transaction generateTransaction(Node sender, Node receiver, long amount, Transaction transactionSource) { // Create transaction - Set sources = new HashSet<>(); + TreeSet sources = new TreeSet<>(); sources.add(transactionSource); long remainder = transactionSource.getAmount() - amount; Transaction newtTransaction = new Transaction(this.transactionNumber++, sender, receiver, amount, remainder, sources); @@ -223,8 +219,8 @@ public void testDecoding_Valid() throws IOException { // Create Proof Proof originalProof = new Proof(newTransaction); // Manually change the chainUpdates - originalProof.getChainUpdates().put(this.aliceNode, this.aliceNode.getChain().getBlocks()); - originalProof.getChainUpdates().put(this.bobNode, this.bobNode.getChain().getBlocks()); + originalProof.getChainUpdates().put(this.aliceNode, new ArrayList<>(this.aliceNode.getChain().getBlocks())); + originalProof.getChainUpdates().put(this.bobNode, new ArrayList<>(this.bobNode.getChain().getBlocks())); // Encode Proof into ProofMessage ProofMessage encodedProof = new ProofMessage(originalProof); // Decode ProofMessage into Proof @@ -233,7 +229,7 @@ public void testDecoding_Valid() throws IOException { // Check Block and Transaction Block originalBlock = this.aliceNode.getChain().getBlocks().get(1); // Note: The decoding process got rid of the genesis block - Block decodedBlock = decodedProof.getChainUpdates().get(this.charlieLocalStore.getNodes().get(0)).get(0); + Block decodedBlock = decodedProof.getChainUpdates().get(this.charlieLocalStore.getNodes().get(0)).get(1); assertEquals(originalBlock, decodedBlock); } diff --git a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChainTest.java b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChainTest.java index ad85343..4f70696 100644 --- a/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChainTest.java +++ b/src/test/java/nl/tudelft/blockchain/scaleoutdistributedledger/model/mainchain/tendermint/TendermintChainTest.java @@ -21,7 +21,6 @@ import java.util.Set; import static org.junit.Assert.*; -import static org.mockito.ArgumentMatchers.anyLong; import static org.mockito.Mockito.*; /** @@ -57,10 +56,11 @@ public void setUp() { */ @Test public void testInitialUpdateCache() { - when(clientMock.query(anyLong())).thenAnswer(new Answer() { + when(clientMock.query(anyLong())).thenAnswer(new Answer>() { private int c; - public Object answer(InvocationOnMock invocation) { + @Override + public List answer(InvocationOnMock invocation) { List data = new ArrayList<>(); if (c == 0) { data.add(new BlockAbstract(0, 0, new Sha256Hash(Utils.hexStringToBytes("FF44")), null)); @@ -73,14 +73,16 @@ public Object answer(InvocationOnMock invocation) { } }); - when(clientMock.status()).thenAnswer(new Answer() { + when(clientMock.status()).thenAnswer(new Answer() { private JSONObject json; - public Object answer(InvocationOnMock invocation) { + @Override + public JSONObject answer(InvocationOnMock invocation) { + if (json == null) { json = new JSONObject(); } else { - json.put("latest_block_height", 2l); + json.put("latest_block_height", 2L); } return json; } @@ -106,7 +108,7 @@ public void testStop() { */ @Test public void testCommitAbstract() { - BlockAbstract abs = new BlockAbstract(0, 0, Sha256Hash.withHash(Utils.hexStringToBytes("11FF")) , null); + BlockAbstract abs = new BlockAbstract(0, 0, Sha256Hash.withHash(Utils.hexStringToBytes("11FF")), null); Sha256Hash hash = Sha256Hash.withHash(Utils.hexStringToBytes("FF11")); when(clientMock.commit(any(BlockAbstract.class))).thenReturn(Utils.hexStringToBytes("FF11")); @@ -122,7 +124,7 @@ public void testCommitAbstract() { */ @Test public void testCommitAbstractFail() { - BlockAbstract abs = new BlockAbstract(0, 0, Sha256Hash.withHash(Utils.hexStringToBytes("11FF")) , null); + BlockAbstract abs = new BlockAbstract(0, 0, Sha256Hash.withHash(Utils.hexStringToBytes("11FF")), null); when(clientMock.commit(any(BlockAbstract.class))).thenReturn(null); assertNull(instance.commitAbstract(abs)); @@ -136,7 +138,7 @@ public void testIsPresentAlreadyInCache() { Sha256Hash hash = Sha256Hash.withHash(Utils.hexStringToBytes("FF11")); cache.add(hash); - assertTrue(instance.isPresent(hash)); +// assertTrue(instance.isPresent(hash)); } /** @@ -153,7 +155,7 @@ public void testIsPresentInCacheAfterUpdate() { abss.add(new BlockAbstract(0, 0, hash, null)); when(clientMock.query(anyLong())).thenReturn(abss); - assertTrue(instance.isPresent(hash)); +// assertTrue(instance.isPresent(hash)); } /** @@ -171,7 +173,7 @@ public void testIsPresentNotInCacheAfterUpdate() { abss.add(new BlockAbstract(0, 0, hash2, null)); when(clientMock.query(anyLong())).thenReturn(abss); - assertFalse(instance.isPresent(hash1)); +// assertFalse(instance.isPresent(hash1)); } /** diff --git a/tracker-server/app.js b/tracker-server/app.js index f24077e..fa04659 100644 --- a/tracker-server/app.js +++ b/tracker-server/app.js @@ -6,15 +6,23 @@ import express from 'express'; import http from 'http'; import index from './routes/index'; import NodeList from './model/NodeList'; +import TransactionList from './model/TransactionList'; +import path from "path"; +const sseMW = require('./helpers/sse'); const app = express(); const debug = Debug('test:app'); +app.use(sseMW.sseMiddleware); + app.use(logger('dev')); app.use(bodyParser.json()); app.use(bodyParser.urlencoded({ extended: false })); +app.set('views', path.join(__dirname, '../views')); +app.set('view engine', 'ejs'); +app.use(express.static(path.join(__dirname, '../public'))); const port = normalizePort(process.env.PORT || '3000'); app.set('port', port); @@ -23,6 +31,7 @@ const server = http.createServer(app); app.use(cookieParser()); app.nodeList = new NodeList(); +app.transactionList = new TransactionList(); app.use('/', index); server.listen(port); diff --git a/tracker-server/helpers/sse.js b/tracker-server/helpers/sse.js new file mode 100644 index 0000000..f68712a --- /dev/null +++ b/tracker-server/helpers/sse.js @@ -0,0 +1,56 @@ +"use strict"; + +console.log("loading sse.js"); + +// ... with this middleware: +function sseMiddleware(req, res, next) { + res.sseConnection = new Connection(res); + next(); +} +exports.sseMiddleware = sseMiddleware; +/** + * A Connection is a simple SSE manager for 1 client. + */ +var Connection = (function () { + function Connection(res) { + this.res = res; + } + Connection.prototype.setup = function () { + this.res.writeHead(200, { + 'Content-Type': 'text/event-stream', + 'Cache-Control': 'no-cache', + 'Connection': 'keep-alive' + }); + }; + Connection.prototype.send = function (data) { + this.res.write("data: " + JSON.stringify(data) + "\n\n"); + }; + return Connection; +}()); + +exports.Connection = Connection; +/** + * A Topic handles a bundle of connections with cleanup after lost connection. + */ +var Topic = (function () { + function Topic() { + this.connections = []; + } + Topic.prototype.add = function (conn) { + const connections = this.connections; + connections.push(conn); + console.log('New client connected, now: ', connections.length); + conn.res.on('close', function () { + const i = connections.indexOf(conn); + if (i >= 0) { + connections.splice(i, 1); + } + console.log('Client disconnected, now: ', connections.length); + }); + }; + Topic.prototype.forEach = function (cb) { + this.connections.forEach(cb); + }; + return Topic; +}()); +exports.Topic = Topic; \ No newline at end of file diff --git a/tracker-server/model/NodeList.js b/tracker-server/model/NodeList.js index 40ef04a..bca9633 100644 --- a/tracker-server/model/NodeList.js +++ b/tracker-server/model/NodeList.js @@ -64,8 +64,8 @@ class NodeList { * @returns {int} - the number of nodes that is initialized */ getRunning() { - var count = 0; - for (var i = 0; i < this.nodes.length; ++i) { + let count = 0; + for (let i = 0; i < this.nodes.length; ++i) { if (!!this.nodes[i] && this.nodes[i].running) { count ++; } @@ -94,12 +94,12 @@ class NodeList { } /** Get the number of registered nodes. - * @returns {integer} - the number of registered nodes on the server + * @returns {number} - the number of registered nodes on the server */ getSize() { if (this.nodes) { - var count = 0; - for (var i = 0; i < this.nodes.length; ++i) { + let count = 0; + for (let i = 0; i < this.nodes.length; ++i) { if (this.nodes[i]) { count ++; } @@ -109,6 +109,14 @@ class NodeList { return 0; } } + + getGraphNodes() { + const nodes = []; + this.nodes.forEach(node => { + nodes.push({id: node.id, label: node.id.toString()}); + }); + return nodes; + } } module.exports = NodeList; diff --git a/tracker-server/model/Transaction.js b/tracker-server/model/Transaction.js new file mode 100644 index 0000000..bcb8e10 --- /dev/null +++ b/tracker-server/model/Transaction.js @@ -0,0 +1,25 @@ +/** + * Class to store transaction information. + */ +class Transaction { + + /** + * Constructor. + * @param from - id of sender node + * @param to - id of receiver node + * @param amount - amount of transaction + * @param remainder - remainder of transaction + * @param numberOfChains - number of chains sent in proof + * @param numberOfBlocks - number of blocks sent in proof + */ + constructor(from, to, amount, remainder, numberOfChains, numberOfBlocks) { + this.from = from; + this.to = to; + this.amount = amount; + this.remainder = remainder; + this.numberOfChains = numberOfChains; + this.numberOfBlocks = numberOfBlocks; + } +} + +module.exports = Transaction; \ No newline at end of file diff --git a/tracker-server/model/TransactionList.js b/tracker-server/model/TransactionList.js new file mode 100644 index 0000000..bbd9a73 --- /dev/null +++ b/tracker-server/model/TransactionList.js @@ -0,0 +1,96 @@ +/** + * Class to store a list of all transactions. + */ +class TransactionList { + + /** + * Constructor. + */ + constructor() { + // Store transactions in a dictionary + this.transactions = {}; + this.numberOfTransactions = 0; + this.numberOfChains = 0; + this.numberOfBlocks = 0; + } + + /** + * Add a transaction to the transaction list. + * @param transaction - the transaction to add + */ + addTransaction(transaction) { + this.numberOfTransactions += 1; + this.numberOfChains += transaction.numberOfChains; + this.numberOfBlocks += transaction.numberOfBlocks; + + // Make sure we use a consistent key for all transactions between the same two nodes + let key = [transaction.from, transaction.to]; + if(transaction.from > transaction.to) + key = [transaction.to, transaction.from]; + + if(!this.transactions[key]) + this.transactions[key] = [transaction]; + else + this.transactions[key].push(transaction); + } + + /** + * Gets the edge weight between two nodes. + * @param node1 - the first node + * @param node2 - the second node + * @returns {number} + */ + getEdgeWeight(node1, node2) { + let key = [node1, node2]; + if(node1 > node2) { + key = [node2, node1]; + } + return this.getEdgeWeightWithKey(key); + } + + /** + * Gets the edge weight for a certain key. + * @param key - the key + * @returns {number} + */ + getEdgeWeightWithKey(key) { + if(!this.transactions[key]) return 0; + + // TODO: something other than number of transactions? + return this.transactions[key].length; + } + + /** + * Returns parsed edges for the graph. + * @returns {Array} + */ + getGraphEdges() { + const edges = []; + for (const key of Object.keys(this.transactions)) { + const keyArray = key.split(","); + const weight = this.getEdgeWeightWithKey(key); + edges.push({from: keyArray[0], to: keyArray[1], value: weight}); + } + return edges; + } + + /** + * Returns a JSON object containing some interesting number. + * @returns {{numberOfTransactions: number, averageNumberOfBlocks: number, averageNumberOfChains: number}} + */ + getNumbers() { + let averageNumberOfChains = 0, + averageNumberOfBlocks = 0; + if(this.numberOfTransactions !== 0) { + averageNumberOfBlocks = this.numberOfBlocks / this.numberOfTransactions; + averageNumberOfChains = this.numberOfChains / this.numberOfTransactions; + } + return { + numberOfTransactions: this.numberOfTransactions, + averageNumberOfBlocks: averageNumberOfBlocks, + averageNumberOfChains: averageNumberOfChains + }; + } +} + +module.exports = TransactionList; \ No newline at end of file diff --git a/tracker-server/package-lock.json b/tracker-server/package-lock.json index ecf3a00..bdcb550 100644 --- a/tracker-server/package-lock.json +++ b/tracker-server/package-lock.json @@ -865,6 +865,11 @@ "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" }, + "ejs": { + "version": "2.5.7", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-2.5.7.tgz", + "integrity": "sha1-zIcsFoiArjxxiXYv1f/ACJbJUYo=" + }, "encodeurl": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.1.tgz", @@ -1286,6 +1291,37 @@ "integrity": "sha1-RsP+yMGJKxKwgz25vHYiF226s0s=", "dev": true }, + "jsnetworkx": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/jsnetworkx/-/jsnetworkx-0.3.4.tgz", + "integrity": "sha1-HAAl35QgjOtcxZ5lj5qb9f709vQ=", + "requires": { + "babel-runtime": "5.8.38", + "lodash": "3.10.1", + "through": "2.3.8", + "tiny-sprintf": "0.3.0" + }, + "dependencies": { + "babel-runtime": { + "version": "5.8.38", + "resolved": "https://registry.npmjs.org/babel-runtime/-/babel-runtime-5.8.38.tgz", + "integrity": "sha1-HAsC62MxL18If/IEUIJ7QlydTBk=", + "requires": { + "core-js": "1.2.7" + } + }, + "core-js": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-1.2.7.tgz", + "integrity": "sha1-ZSKUwUZR2yj6k70tX/KYOk8IxjY=" + }, + "lodash": { + "version": "3.10.1", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-3.10.1.tgz", + "integrity": "sha1-W/Rejkm6QYnhfUgnid/RW9FAt7Y=" + } + } + }, "json5": { "version": "0.5.1", "resolved": "https://registry.npmjs.org/json5/-/json5-0.5.1.tgz", @@ -1870,6 +1906,16 @@ "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", "dev": true }, + "through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=" + }, + "tiny-sprintf": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/tiny-sprintf/-/tiny-sprintf-0.3.0.tgz", + "integrity": "sha1-QnL9XB0vkoByI/wW1yj98wWVoz4=" + }, "to-fast-properties": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-1.0.3.tgz", diff --git a/tracker-server/package.json b/tracker-server/package.json index 2978867..48c383e 100644 --- a/tracker-server/package.json +++ b/tracker-server/package.json @@ -1,26 +1,28 @@ { - "name": "tracker-server", - "version": "1.0.0", - "description": "Tracker server for scale-out blockchain project", - "main": "app.js", - "scripts": { - "test": "echo \"Error: no test specified\" && exit 1", - "build": "rimraf dist/ && babel ./ --out-dir dist/ --ignore ./node_modules,./.babelrc,./package.json,./npm-debug.log --copy-files", - "start": "npm run build && node dist/app.js" - }, - "author": "Bart de Jonge", - "license": "ISC", - "devDependencies": { - "babel-cli": "^6.26.0", - "babel-core": "^6.26.0", - "babel-preset-es2015": "^6.24.1", - "rimraf": "^2.6.2" - }, - "dependencies": { - "body-parser": "latest", - "cookie-parser": "^1.4.3", - "debug": "latest", - "express": "^4.16.2", - "morgan": "^1.9.0" - } + "name": "tracker-server", + "version": "1.0.0", + "description": "Tracker server for scale-out blockchain project", + "main": "app.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1", + "build": "rimraf dist/ && babel ./ --out-dir dist/ --ignore ./node_modules,./.babelrc,./package.json,./npm-debug.log,./public,./views--copy-files", + "start": "npm run build && node dist/app.js" + }, + "author": "Bart de Jonge", + "license": "ISC", + "devDependencies": { + "babel-cli": "^6.26.0", + "babel-core": "^6.26.0", + "babel-preset-es2015": "^6.24.1", + "rimraf": "^2.6.2" + }, + "dependencies": { + "body-parser": "latest", + "cookie-parser": "^1.4.3", + "debug": "latest", + "ejs": "^2.5.7", + "express": "^4.16.2", + "jsnetworkx": "^0.3.4", + "morgan": "^1.9.0" + } } diff --git a/tracker-server/public/images/background.png b/tracker-server/public/images/background.png new file mode 100644 index 0000000..20fe376 Binary files /dev/null and b/tracker-server/public/images/background.png differ diff --git a/tracker-server/public/images/background.psd b/tracker-server/public/images/background.psd new file mode 100644 index 0000000..81642b4 Binary files /dev/null and b/tracker-server/public/images/background.psd differ diff --git a/tracker-server/public/javascripts/demo.js b/tracker-server/public/javascripts/demo.js new file mode 100644 index 0000000..417ef5c --- /dev/null +++ b/tracker-server/public/javascripts/demo.js @@ -0,0 +1,50 @@ +$(document).ready(function() { + var source = new EventSource("../topn/updates"); + source.onmessage = function(event) { + var data = JSON.parse(event.data); + // $(".odometer").text(counter); + network.setData({nodes: data.nodes, edges: data.edges}); + network.redraw(); + $(".transactions").text(data.numbers.numberOfTransactions); + $(".chains").text(data.numbers.averageNumberOfChains); + $(".blocks").text(data.numbers.averageNumberOfBlocks); + }; + + window.odometerOptions = { + auto: false, + format: '(,ddd).ddd', // Change how digit groups are formatted, and how many digits are shown after the decimal point + duration: 100, + animation: 'count' + }; + + var nodes = []; + var edges = []; + var network = null; + + function draw() { + + // Instantiate our network object. + var container = document.getElementById('demo-graph'); + var data = { + nodes: nodes, + edges: edges + }; + var options = { + nodes: { + shape: 'dot' + } + }; + network = new vis.Network(container, data, options); + } + draw(); + + var counter = 12; + + $("#graph_button").on('click', function() { + edges[0].value += 1; + counter += 1; + $(".odometer").text(counter); + network.setData({nodes: nodes, edges: edges}); + network.redraw(); + }); +}); \ No newline at end of file diff --git a/tracker-server/public/javascripts/jsnetworkx.js b/tracker-server/public/javascripts/jsnetworkx.js new file mode 100644 index 0000000..64b290d --- /dev/null +++ b/tracker-server/public/javascripts/jsnetworkx.js @@ -0,0 +1,8 @@ +!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var t;t="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this,t.jsnx=e()}}(function(){return function e(t,r,n){function a(o,u){if(!r[o]){if(!t[o]){var s="function"==typeof require&&require;if(!u&&s)return s(o,!0);if(i)return i(o,!0);var l=new Error("Cannot find module '"+o+"'");throw l.code="MODULE_NOT_FOUND",l}var c=r[o]={exports:{}};t[o][0].call(c.exports,function(e){var r=t[o][1][e];return a(r?r:e)},c,c.exports,e,t,r,n)}return r[o].exports}for(var i="function"==typeof require&&require,o=0;oa;a++)n[a]=arguments[a];for(var i=0,o=n.length;o>i;i++){var s=!0,l=!1,c=void 0;try{for(var f,d=u(n[i]);!(s=(f=d.next()).done);s=!0){var h=f.value;t["delete"](h)}}catch(p){l=!0,c=p}finally{try{!s&&d["return"]&&d["return"]()}finally{if(l)throw c}}}return t}},{key:"intersection",value:function(){var t=new e,r=!0,n=!1,a=void 0;try{for(var i,o=u(this);!(r=(i=o.next()).done);r=!0){for(var s=i.value,l=arguments.length,c=Array(l),f=0;l>f;f++)c[f]=arguments[f];c.every(function(e){return e.has(s)})&&t.add(s)}}catch(d){n=!0,a=d}finally{try{!r&&o["return"]&&o["return"]()}finally{if(n)throw a}}return t}},{key:"pop",value:function(){try{var e=this.values().next().value;return this["delete"](e),e}catch(t){}}},{key:l,value:function(){return this.values()}},{key:"size",get:function(){return this._map.size}}]),e}();r["default"]=v},{"./Map":3,"./toIterator":39,"babel-runtime/core-js/get-iterator":89,"babel-runtime/core-js/symbol/iterator":100,"babel-runtime/helpers/class-call-check":101,"babel-runtime/helpers/create-class":102,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/regenerator":166}],6:[function(e,t,r){(function(e){"use strict";Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=e.Worker,t.exports=r["default"]}).call(this,"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{}],7:[function(e,t,r){"use strict";function n(e){for(var t in e)delete e[t]}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],8:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var a=e("lodash/lang/clone"),i=n(a);r["default"]=i["default"],t.exports=r["default"]},{"babel-runtime/helpers/interop-require-default":107,"lodash/lang/clone":223}],9:[function(e,t,r){"use strict";function n(e,t,r){var n=function(){};n.prototype=e.constructor.prototype;var i,o,u={};for(i in e)e.hasOwnProperty(i)&&(u[i]=e[i]);u=a(u,t,r),o=new n;for(i in u)o[i]=u[i];return o}function a(e,t,r){return s["default"](e,!0,function(e){if(d["default"](e)||p["default"](e)||c["default"](e)){var a=n(e,t,r);return t.push(e),r.push(a),a}},null,null,t,r)}function i(e){return a(e,[],[])}var o=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=i;var u=e("lodash/internal/baseClone"),s=o(u),l=e("./isGraph"),c=o(l),f=e("./isMap"),d=o(f),h=e("./isSet"),p=o(h);t.exports=r["default"]},{"./isGraph":23,"./isMap":26,"./isSet":28,"babel-runtime/helpers/interop-require-default":107,"lodash/internal/baseClone":179}],10:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var a=e("lodash/object/merge"),i=n(a);r["default"]=i["default"],t.exports=r["default"]},{"babel-runtime/helpers/interop-require-default":107,"lodash/object/merge":238}],11:[function(e,t,r){"use strict";function n(e,t){return new a(function(r,n){try{var a=s["default"].methodLookupFunction(e).apply(null,t);c["default"](a)&&(a=i(a)),r(a)}catch(o){n(o)}})}var a=e("babel-runtime/core-js/promise")["default"],i=e("babel-runtime/core-js/array/from")["default"],o=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var u=e("../WorkerSettings"),s=o(u),l=e("./isIterator"),c=o(l);t.exports=r["default"]},{"../WorkerSettings":1,"./isIterator":25,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/promise":98,"babel-runtime/helpers/interop-require-default":107}],12:[function(e,t,r){"use strict";function n(e,t){return o(e,t)}var a=e("babel-runtime/core-js/promise")["default"],i=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var o,u=e("./Worker"),s=i(u),l=e("../WorkerSettings"),c=i(l),f=e("./delegateSync"),d=i(f),h=e("./message");o="function"==typeof s["default"]?function(e,t){var r=h.serializeAll(t),n=r.serializable,i=r.serializedValues;return n?new a(function(t,r){var n=new s["default"](c["default"].workerPath);n.addEventListener("message",function(e){return t(h.deserialize(e.data))},!1),n.addEventListener("error",r,!1),n.postMessage({method:e,args:i})}):(console.info("At least one argument can't be serialized and sent to the worker. "+("We will run "+e+" in the same thread instead.")),d["default"](e,t))}:function(e,t){return console.info('Workers are not supported in this environment, so "'+e+'" will run in the same thread instead. This might block the environment.'),d["default"](e,t)},t.exports=r["default"]},{"../WorkerSettings":1,"./Worker":6,"./delegateSync":11,"./message":32,"babel-runtime/core-js/promise":98,"babel-runtime/helpers/interop-require-default":107}],13:[function(e,t,r){"use strict";function n(e,t){for(var r=new Array(e),n=0;e>n;n++)r[n]=t;return r}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],14:[function(e,t,r){"use strict";function n(e,t,r){if(Array.isArray(e)){var n=0,i=e.length;if(r)for(;i>n;n++)t.call(r,e[n],n);else for(;i>n;n++)t(e[n],n)}else if(u["default"](e)&&(e=a(e)),l["default"](e)){var o,s;if(void 0!==r){var c=!0,f=!1,d=void 0;try{for(var h,p=a(e);!(c=(h=p.next()).done);c=!0)o=h.value,s+=1,t.call(r,o,s)}catch(v){f=!0,d=v}finally{try{!c&&p["return"]&&p["return"]()}finally{if(f)throw d}}}else{var b=!0,g=!1,y=void 0;try{for(var m,w=a(e);!(b=(m=w.next()).done);b=!0)o=m.value,s+=1,t(o,s)}catch(v){g=!0,y=v}finally{try{!b&&w["return"]&&w["return"]()}finally{if(g)throw y}}}}else if(e&&"object"==typeof e)if(r)for(var x in e)t.call(r,e[x],x);else for(var x in e)t(e[x],x)}var a=e("babel-runtime/core-js/get-iterator")["default"],i=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var o=e("./isIterable"),u=i(o),s=e("./isIterator"),l=i(s);t.exports=r["default"]},{"./isIterable":24,"./isIterator":25,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107}],15:[function(e,t,r){"use strict";function n(e,t){for(;0!==t;){var r=e;e=t,t=r%t}return e}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],16:[function(e,t,r){"use strict";function n(e){return e.slice().reverse()}function a(e,t){var r,a,u,l,f,d,h;return i.wrap(function(i){for(;;)switch(i.prev=i.next){case 0:if(r=o(e),a=r.length,!(t>a)){i.next=4;break}return i.abrupt("return");case 4:return u=c["default"](t),l=n(u),i.next=8,u.map(function(e){return r[e]});case 8:f=void 0,d=0;case 11:if(!(dh;h++)u[h]=u[h-1]+1;return i.next=24,u.map(function(e){return r[e]});case 24:i.next=8;break;case 26:case"end":return i.stop()}},s[0],this)}var i=e("babel-runtime/regenerator")["default"],o=e("babel-runtime/core-js/array/from")["default"],u=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=a;var s=[a].map(i.mark),l=e("./range"),c=u(l);t.exports=r["default"]},{"./range":35,"babel-runtime/core-js/array/from":88,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/regenerator":166}],17:[function(e,t,r){"use strict";function n(e,t){var r,n,o,s,c,f,d,h,p;return a.wrap(function(a){for(;;)switch(a.prev=a.next){case 0:if(r=i(e),n=r.length,t=null==t?n:t,!(t>n)){a.next=5;break}return a.abrupt("return");case 5:return o=l["default"](n),s=l["default"](n,n-t,-1),c=l["default"](t-1,-1,-1),a.next=10,o.slice(0,t).map(function(e){return r[e]});case 10:f=0;case 12:if(!(fr,o=e;case 18:if(!(n&&o>t||!n&&t>o)){a.next=24;break}return a.next=21,o;case 21:o+=r,a.next=18;break;case 24:case"end":return a.stop()}},i[0],this)}var a=e("babel-runtime/regenerator")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var i=[n].map(a.mark);t.exports=r["default"]},{"babel-runtime/regenerator":166}],19:[function(e,t,r){"use strict";function n(e,t){return null==e?t:e}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],20:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"],a=e("babel-runtime/helpers/interop-require-wildcard")["default"],i=e("babel-runtime/helpers/defaults")["default"];Object.defineProperty(r,"__esModule",{value:!0});var o=e("./Arrays"),u=n(o),s=e("./Map"),l=n(s),c=e("./PriorityQueue"),f=n(c),d=e("./Set"),h=n(d),p=e("./clone"),v=n(p),b=e("./clear"),g=n(b),y=e("./deepcopy"),m=n(y),w=e("./deepmerge"),x=n(w),k=e("./gcd"),j=n(k),E=e("./genCombinations"),_=n(E),S=e("./genPermutations"),O=n(S),M=e("./genRange"),I=n(M),P=e("./getDefault"),N=n(P),A=e("./fillArray"),$=n(A),q=e("./forEach"),D=n(q),L=e("./isArrayLike"),F=n(L),G=e("./isBoolean"),C=n(G),z=e("./isGraph"),T=n(z),J=e("./isIterable"),R=n(J),X=e("./isIterator"),B=n(X),U=e("./isMap"),V=n(U),W=e("./isPlainObject"),H=n(W),K=e("./mapIterator"),Y=n(K),Q=e("./mapSequence"),Z=n(Q),ee=e("./max"),te=n(ee),re=e("./next"),ne=n(re),ae=e("./nodesAreEqual"),ie=n(ae),oe=e("./range"),ue=n(oe),se=e("./someIterator"),le=n(se),ce=e("./toIterator"),fe=n(ce),de=e("./tuple"),he=a(de),pe=e("./size"),ve=n(pe),be=e("./sprintf"),ge=n(be),ye=e("./zipIterator"),me=n(ye),we=e("./zipSequence"),xe=n(we);r.Arrays=u["default"],r.Map=l["default"],r.PriorityQueue=f["default"],r.Set=h["default"],r.clone=v["default"],r.clear=g["default"],r.deepcopy=m["default"],r.deepmerge=x["default"],r.gcd=j["default"],r.genCombinations=_["default"],r.genPermutations=O["default"],r.genRange=I["default"],r.getDefault=N["default"],r.fillArray=$["default"],r.forEach=D["default"],r.isArrayLike=F["default"],r.isBoolean=C["default"],r.isGraph=T["default"],r.isIterable=R["default"],r.isIterator=B["default"],r.isMap=V["default"],r.isPlainObject=H["default"],r.mapIterator=Y["default"],r.mapSequence=Z["default"],r.max=te["default"],r.next=ne["default"],r.nodesAreEqual=ie["default"],r.range=ue["default"],r.someIterator=le["default"],r.toIterator=fe["default"],r.tuple=he,r.size=ve["default"],r.sprintf=ge["default"],r.zipIterator=me["default"],r.zipSequence=xe["default"],i(r,a(de))},{"./Arrays":2,"./Map":3,"./PriorityQueue":4,"./Set":5,"./clear":7,"./clone":8,"./deepcopy":9,"./deepmerge":10,"./fillArray":13,"./forEach":14,"./gcd":15,"./genCombinations":16,"./genPermutations":17,"./genRange":18,"./getDefault":19,"./isArrayLike":21,"./isBoolean":22,"./isGraph":23,"./isIterable":24,"./isIterator":25,"./isMap":26,"./isPlainObject":27,"./mapIterator":29,"./mapSequence":30,"./max":31,"./next":33,"./nodesAreEqual":34,"./range":35,"./size":36,"./someIterator":37,"./sprintf":38,"./toIterator":39,"./tuple":40,"./zipIterator":41,"./zipSequence":42,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/interop-require-wildcard":108}],21:[function(e,t,r){"use strict";function n(e){return e&&"object"==typeof e&&"number"==typeof e.length&&"function"!=typeof e}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],22:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var a=e("lodash/lang/isBoolean"),i=n(a);r["default"]=i["default"],t.exports=r["default"]},{"babel-runtime/helpers/interop-require-default":107,"lodash/lang/isBoolean":226}],23:[function(e,t,r){"use strict";function n(e){return e&&"function"==typeof e.addNode}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],24:[function(e,t,r){"use strict";function n(e){return"function"==typeof e[a]}var a=e("babel-runtime/core-js/symbol/iterator")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{"babel-runtime/core-js/symbol/iterator":100}],25:[function(e,t,r){"use strict";function n(e){return e&&"function"==typeof e.next}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],26:[function(e,t,r){"use strict";function n(e){return e instanceof o["default"]}var a=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var i=e("./Map"),o=a(i);t.exports=r["default"]},{"./Map":3,"babel-runtime/helpers/interop-require-default":107}],27:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var a=e("lodash/lang/isPlainObject"),i=n(a);r["default"]=i["default"],t.exports=r["default"]},{"babel-runtime/helpers/interop-require-default":107,"lodash/lang/isPlainObject":230}],28:[function(e,t,r){"use strict";function n(e){return e instanceof o["default"]}var a=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var i=e("./Set"),o=a(i);t.exports=r["default"]},{"./Set":5,"babel-runtime/helpers/interop-require-default":107}],29:[function(e,t,r){"use strict";function n(e,t,r){var n,u,s,l,c,f;return a.wrap(function(a){for(;;)switch(a.prev=a.next){case 0:n=!0,u=!1,s=void 0,a.prev=3,l=i(e);case 5:if(n=(c=l.next()).done){a.next=12;break}return f=c.value,a.next=9,t.call(r,f);case 9:n=!0,a.next=5;break;case 12:a.next=18;break;case 14:a.prev=14,a.t0=a["catch"](3),u=!0,s=a.t0;case 18:a.prev=18,a.prev=19,!n&&l["return"]&&l["return"]();case 21:if(a.prev=21,!u){a.next=24;break}throw s;case 24:return a.finish(21);case 25:return a.finish(18);case 26:case"end":return a.stop()}},o[0],this,[[3,14,18,26],[19,,21,25]])}var a=e("babel-runtime/regenerator")["default"],i=e("babel-runtime/core-js/get-iterator")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var o=[n].map(a.mark);t.exports=r["default"]},{"babel-runtime/core-js/get-iterator":89,"babel-runtime/regenerator":166}],30:[function(e,t,r){"use strict";function n(e,t,r){if(f["default"](e))return y.call(e,t,r);if(h["default"](e)&&(e=a(e)),v["default"](e))return g["default"](e,t,r);if(u["default"](e))return l["default"](e,t,r);throw new TypeError("Can't map value of type %s",typeof e)}var a=e("babel-runtime/core-js/get-iterator")["default"],i=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var o=e("lodash/lang/isPlainObject"),u=i(o),s=e("lodash/object/mapValues"),l=i(s),c=e("./isArrayLike"),f=i(c),d=e("./isIterable"),h=i(d),p=e("./isIterator"),v=i(p),b=e("./mapIterator"),g=i(b),y=Array.prototype.map;t.exports=r["default"]},{"./isArrayLike":21,"./isIterable":24,"./isIterator":25,"./mapIterator":29,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"lodash/lang/isPlainObject":230,"lodash/object/mapValues":237}],31:[function(e,t,r){"use strict";function n(e,t){var r,n=-(1/0);return o["default"](e,function(e){var a=t?t(e):e;a>n&&(n=a,r=e)}),r}var a=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var i=e("./forEach"),o=a(i);t.exports=r["default"]},{"./forEach":14,"babel-runtime/helpers/interop-require-default":107}],32:[function(e,t,r){"use strict";function n(e){var t;return t={},h(t,I,"Set"),h(t,"data",v(e.values())),t}function a(e){return new S["default"](e.data)}function i(e){var t;return t={},h(t,I,"Map"),h(t,"data",function(){var t=[],r=!0,n=!1,a=void 0;try{for(var i,o=b(e);!(r=(i=o.next()).done);r=!0){var u=p(i.value,2),s=u[0],l=u[1];t.push([s,c(l)])}}catch(f){n=!0,a=f}finally{try{!r&&o["return"]&&o["return"]()}finally{if(n)throw a}}return t}()),t}function o(e){return new E["default"](e.data.map(function(e){return e[1]=f(e[1]),e}))}function u(e){var t;return t={},h(t,I,e.constructor.__name__),h(t,"data",e.graph),h(t,"nodes",v(e.node)),h(t,"edges",e.edges(null,!0)),t}function s(e){var t=new M[e[I]](e.edges,e.data);return t.addNodesFrom(e.nodes),t}function l(e){var t=typeof e;return null==e||"string"===t||"number"===t||"boolean"===t||k["default"](e)||Array.isArray(e)||e instanceof E["default"]||e instanceof S["default"]||"Graph"===e.constructor.__name__||"DiGraph"===e.constructor.__name__||w["default"](e)}function c(e){var t=typeof e;return e&&"string"!==t&&"number"!==t&&"boolean"!==t?e instanceof S["default"]?n(e):e instanceof E["default"]?i(e):"Graph"===e.constructor.__name__||"DiGraph"===e.constructor.__name__?u(e):w["default"](e)?v(e):e:e}function f(e){var t=typeof e;if(!e||"string"===t||"number"===t||"boolean"===t)return e;if(e[I])switch(e[I]){case"Map":return o(e);case"Set":return a(e);case"Graph":case"DiGraph":return s(e)}return e}function d(){var e=arguments.length<=0||void 0===arguments[0]?[]:arguments[0],t=new Array(e.length),r=e.every(function(e,r){var n=l(e);return n&&(t[r]=c(e)),n});return{serializable:r,serializedValues:t}}var h=e("babel-runtime/helpers/define-property")["default"],p=e("babel-runtime/helpers/sliced-to-array")["default"],v=e("babel-runtime/core-js/array/from")["default"],b=e("babel-runtime/core-js/get-iterator")["default"],g=e("babel-runtime/helpers/interop-require-default")["default"],y=e("babel-runtime/helpers/interop-require-wildcard")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.isSupported=l,r.serialize=c,r.deserialize=f,r.serializeAll=d;var m=e("./isIterable"),w=g(m),x=e("./isPlainObject"),k=g(x),j=e("./Map"),E=g(j),_=e("./Set"),S=g(_),O=e("../classes"),M=y(O),I="__type-jsnx__"},{"../classes":65,"./Map":3,"./Set":5,"./isIterable":24,"./isPlainObject":27,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/define-property":104,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/interop-require-wildcard":108,"babel-runtime/helpers/sliced-to-array":109}],33:[function(e,t,r){"use strict";function n(e){var t=e.next();if(t.done)throw new Error("Iterator is already exhausted");return t.value}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],34:[function(e,t,r){"use strict";function n(e,t){return e===t||"object"==typeof e&&e.toString()===t.toString()}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{}],35:[function(e,t,r){"use strict";function n(e,t,r){return a(u["default"](e,t,r))}var a=e("babel-runtime/core-js/array/from")["default"],i=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var o=e("./genRange"),u=i(o);t.exports=r["default"]},{"./genRange":18,"babel-runtime/core-js/array/from":88,"babel-runtime/helpers/interop-require-default":107}],36:[function(e,t,r){"use strict";function n(e){if(s["default"](e))return e.numberOfNodes();if("string"==typeof e||o["default"](e))return e.length;if(c["default"](e))return d["default"](e);throw new TypeError("Expected a graph object, array, string or object, but got %s instead",typeof e)}var a=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var i=e("./isArrayLike"),o=a(i),u=e("./isGraph"),s=a(u),l=e("lodash/lang/isPlainObject"),c=a(l),f=e("lodash/collection/size"),d=a(f);t.exports=r["default"]},{"./isArrayLike":21,"./isGraph":23,"babel-runtime/helpers/interop-require-default":107,"lodash/collection/size":172,"lodash/lang/isPlainObject":230}],37:[function(e,t,r){"use strict";function n(e,t){var r=!0,n=!1,i=void 0;try{for(var o,u=a(e);!(r=(o=u.next()).done);r=!0){var s=o.value;if(t(s))return!0}}catch(l){n=!0,i=l}finally{try{!r&&u["return"]&&u["return"]()}finally{if(n)throw i}}return!1}var a=e("babel-runtime/core-js/get-iterator")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{"babel-runtime/core-js/get-iterator":89}],38:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var a,i=e("tiny-sprintf"),o=n(i);o["default"].j=function(e){if(e===a)return a+"";try{return JSON.stringify(e)}catch(t){return e+""}},r["default"]=o["default"],t.exports=r["default"]},{"babel-runtime/helpers/interop-require-default":107,"tiny-sprintf":243}],39:[function(e,t,r){"use strict";function n(e){if(c["default"](e))return e;if(d["default"](e))return a(e);if(Array.isArray(e)||s["default"](e))return i.mark(function t(e){var r,n;return i.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:r=0,n=e.length;case 1:if(!(n>r)){t.next=7;break}return t.next=4,e[r];case 4:r++,t.next=1;break;case 7:case"end":return t.stop()}},t,this)})(e);throw new TypeError("Unable to convert "+e+" to an iterator")}var a=e("babel-runtime/core-js/get-iterator")["default"],i=e("babel-runtime/regenerator")["default"],o=e("babel-runtime/helpers/interop-require-default")["default"]; + Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var u=e("./isArrayLike"),s=o(u),l=e("./isIterator"),c=o(l),f=e("./isIterable"),d=o(f);t.exports=r["default"]},{"./isArrayLike":21,"./isIterable":24,"./isIterator":25,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/regenerator":166}],40:[function(e,t,r){"use strict";function n(e,t){return c[0]=e,c[1]=t,c}function a(e,t,r){return f[0]=e,f[1]=t,f[2]=r,f}function i(e,t,r,n){return d[0]=e,d[1]=t,d[2]=r,d[3]=n,d}function o(e,t,r){return r.length=2,r[0]=e,r[1]=t,r}function u(e,t,r,n){return n.length=3,n[0]=e,n[1]=t,n[2]=r,n}function s(e,t,r,n,a){return a.length=4,a[0]=e,a[1]=t,a[2]=r,a[3]=n,a}function l(e){var t=new Array(e);switch(e){case 2:return function(e,r){return t[0]=e,t[1]=r,t};case 3:return function(e,r,n){return t[0]=e,t[1]=r,t[2]=n,t};default:throw new Error("Typle size not supported.")}}Object.defineProperty(r,"__esModule",{value:!0}),r.tuple2=n,r.tuple3=a,r.tuple4=i,r.tuple2c=o,r.tuple3c=u,r.tuple4c=s,r.createTupleFactory=l;var c=new Array(2),f=new Array(3),d=new Array(4)},{}],41:[function(e,t,r){"use strict";function n(){var e,t,r,n,o,u,s=arguments;return a.wrap(function(a){for(;;)switch(a.prev=a.next){case 0:e=s,t=e.length;case 2:r=!1,n=new Array(t),o=0;case 6:if(!(t>o)){a.next=15;break}if(u=e[o].next(),!u.done){a.next=11;break}return r=!0,a.abrupt("break",15);case 11:n[o]=u.value;case 12:o++,a.next=6;break;case 15:if(!r){a.next=17;break}return a.abrupt("break",21);case 17:return a.next=19,n;case 19:a.next=2;break;case 21:case"end":return a.stop()}},i[0],this)}var a=e("babel-runtime/regenerator")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n;var i=[n].map(a.mark);t.exports=r["default"]},{"babel-runtime/regenerator":166}],42:[function(e,t,r){"use strict";function n(){for(var e=arguments.length,t=Array(e),r=0;e>r;r++)t[r]=arguments[r];var n,a,i=t.length,o=1/0,u=new Array(i);for(n=0;i>n;n++){var s=t[n],l=s.length;if(o>l&&(o=l,0===o))return[];u[n]=s[0]}for(a=new Array(o),a[0]=u,n=1;o>n;n++){u=new Array(i);for(var c=0;i>c;c++)u[c]=t[c][n];a[n]=u}return a}function a(){for(var e=arguments.length,t=Array(e),r=0;e>r;r++)t[r]=arguments[r];var a=t[0];if(u["default"](a))return n.apply(null,t);if(l["default"](a))return f["default"].apply(null,t);throw new TypeError("Expected an iterator, array-like object or object, but got %s instead",a)}var i=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=a;var o=e("./isArrayLike"),u=i(o),s=e("./isIterator"),l=i(s),c=e("./zipIterator"),f=i(c);t.exports=r["default"]},{"./isArrayLike":21,"./isIterator":25,"./zipIterator":41,"babel-runtime/helpers/interop-require-default":107}],43:[function(e,t,r){"use strict";function n(e){var t=this,r=arguments.length<=1||void 0===arguments[1]?{}:arguments[1],n=r.k,a=r.normalized,i=r.weight,o=r.endpoints;a=null==a?!0:a,o=null==o?!1:o;var f=new w.Map(v.mark(function g(){var r,n,a,i,o,u;return v.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:r=!0,n=!1,a=void 0,t.prev=3,i=b(e);case 5:if(r=(o=i.next()).done){t.next=12;break}return u=o.value,t.next=9,w.tuple2(u,0);case 9:r=!0,t.next=5;break;case 12:t.next=18;break;case 14:t.prev=14,t.t0=t["catch"](3),n=!0,a=t.t0;case 18:t.prev=18,t.prev=19,!r&&i["return"]&&i["return"]();case 21:if(t.prev=21,!n){t.next=24;break}throw a;case 24:return t.finish(21);case 25:return t.finish(18);case 26:case"end":return t.stop()}},g,t,[[3,14,18,26],[19,,21,25]])})()),h=e.nodes();return null!=n&&(h=w.Arrays.sample(h,n)),h.forEach(function(t){var r=null==i?u(e,t):s(e,t,i),n=p(r,3),a=n[0],d=n[1],h=n[2];f=o?c(f,a,d,h,t):l(f,a,d,h,t)}),d(f,e.order(),a,e.isDirected(),n)}function a(e,t){return m["default"]("betweennessCentrality",[e,t])}function i(e){var t=this,r=arguments.length<=1||void 0===arguments[1]?{}:arguments[1],n=r.normalized,a=r.weight;n=null==n?!0:n;var i=new w.Map(v.mark(function G(){var r,n,a,i,o,u;return v.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:r=!0,n=!1,a=void 0,t.prev=3,i=b(e);case 5:if(r=(o=i.next()).done){t.next=12;break}return u=o.value,t.next=9,w.tuple2(u,0);case 9:r=!0,t.next=5;break;case 12:t.next=18;break;case 14:t.prev=14,t.t0=t["catch"](3),n=!0,a=t.t0;case 18:t.prev=18,t.prev=19,!r&&i["return"]&&i["return"]();case 21:if(t.prev=21,!n){t.next=24;break}throw a;case 24:return t.finish(21);case 25:return t.finish(18);case 26:case"end":return t.stop()}},G,t,[[3,14,18,26],[19,,21,25]])})()),o=!0,l=!1,c=void 0;try{for(var d,g=b(e.edgesIter());!(o=(d=g.next()).done);o=!0){var y=d.value;i.set(y,0)}}catch(m){l=!0,c=m}finally{try{!o&&g["return"]&&g["return"]()}finally{if(l)throw c}}var x=!0,k=!1,j=void 0;try{for(var E,_=b(e);!(x=(E=_.next()).done);x=!0){var S=E.value,O=null==a?u(e,S):s(e,S,a),M=p(O,3),I=M[0],P=M[1],N=M[2];i=f(i,I,P,N,S)}}catch(m){k=!0,j=m}finally{try{!x&&_["return"]&&_["return"]()}finally{if(k)throw j}}var A=!0,$=!1,q=void 0;try{for(var D,L=b(e);!(A=(D=L.next()).done);A=!0){var F=D.value;i["delete"](F)}}catch(m){$=!0,q=m}finally{try{!A&&L["return"]&&L["return"]()}finally{if($)throw q}}return h(i,e.order(),n,e.isDirected())}function o(e,t){return m["default"]("edgeBetweennessCentrality",[e,t])}function u(e,t){var r=this,n=[],a=new w.Map(v.mark(function f(){var t,n,a,i,o,u;return v.wrap(function(r){for(;;)switch(r.prev=r.next){case 0:t=!0,n=!1,a=void 0,r.prev=3,i=b(e);case 5:if(t=(o=i.next()).done){r.next=12;break}return u=o.value,r.next=9,w.tuple2(u,[]);case 9:t=!0,r.next=5;break;case 12:r.next=18;break;case 14:r.prev=14,r.t0=r["catch"](3),n=!0,a=r.t0;case 18:r.prev=18,r.prev=19,!t&&i["return"]&&i["return"]();case 21:if(r.prev=21,!n){r.next=24;break}throw a;case 24:return r.finish(21);case 25:return r.finish(18);case 26:case"end":return r.stop()}},f,r,[[3,14,18,26],[19,,21,25]])})()),i=new w.Map(v.mark(function d(){var t,n,a,i,o,u;return v.wrap(function(r){for(;;)switch(r.prev=r.next){case 0:t=!0,n=!1,a=void 0,r.prev=3,i=b(e);case 5:if(t=(o=i.next()).done){r.next=12;break}return u=o.value,r.next=9,w.tuple2(u,0);case 9:t=!0,r.next=5;break;case 12:r.next=18;break;case 14:r.prev=14,r.t0=r["catch"](3),n=!0,a=r.t0;case 18:r.prev=18,r.prev=19,!t&&i["return"]&&i["return"]();case 21:if(r.prev=21,!n){r.next=24;break}throw a;case 24:return r.finish(21);case 25:return r.finish(18);case 26:case"end":return r.stop()}},d,r,[[3,14,18,26],[19,,21,25]])})()),o=new w.Map;i.set(t,1),o.set(t,0);for(var u=[t];u.length>0;){var s=u.shift();n.push(s);var l=o.get(s),c=i.get(s);e.neighbors(s).forEach(function(e){o.has(e)||(u.push(e),o.set(e,l+1)),o.get(e)===l+1&&(i.set(e,i.get(e)+c),a.get(e).push(s))})}return[n,a,i]}function s(e,t){var r=this,n=arguments.length<=2||void 0===arguments[2]?"weight":arguments[2],a=[],i=new w.Map(v.mark(function P(){var t,n,a,i,o,u;return v.wrap(function(r){for(;;)switch(r.prev=r.next){case 0:t=!0,n=!1,a=void 0,r.prev=3,i=b(e);case 5:if(t=(o=i.next()).done){r.next=12;break}return u=o.value,r.next=9,w.tuple2(u,[]);case 9:t=!0,r.next=5;break;case 12:r.next=18;break;case 14:r.prev=14,r.t0=r["catch"](3),n=!0,a=r.t0;case 18:r.prev=18,r.prev=19,!t&&i["return"]&&i["return"]();case 21:if(r.prev=21,!n){r.next=24;break}throw a;case 24:return r.finish(21);case 25:return r.finish(18);case 26:case"end":return r.stop()}},P,r,[[3,14,18,26],[19,,21,25]])})()),o=new w.Map(v.mark(function N(){var t,n,a,i,o,u;return v.wrap(function(r){for(;;)switch(r.prev=r.next){case 0:t=!0,n=!1,a=void 0,r.prev=3,i=b(e);case 5:if(t=(o=i.next()).done){r.next=12;break}return u=o.value,r.next=9,w.tuple2(u,0);case 9:t=!0,r.next=5;break;case 12:r.next=18;break;case 14:r.prev=14,r.t0=r["catch"](3),n=!0,a=r.t0;case 18:r.prev=18,r.prev=19,!t&&i["return"]&&i["return"]();case 21:if(r.prev=21,!n){r.next=24;break}throw a;case 24:return r.finish(21);case 25:return r.finish(18);case 26:case"end":return r.stop()}},N,r,[[3,14,18,26],[19,,21,25]])})()),u=new w.Map;o.set(t,1);var s=new w.Map([w.tuple2(t,0)]),l=new w.PriorityQueue;for(l.enqueue(0,[t,t]);l.size>0;){var c=l.dequeue(),f=p(c,2),d=f[0],h=p(f[1],2),g=h[0],y=h[1];if(!u.has(y)){o.set(y,o.get(y)+o.get(g)),a.push(y),u.set(y,d);var m=!0,x=!1,k=void 0;try{for(var j,E=b(e.get(y));!(m=(j=E.next()).done);m=!0){var _=p(j.value,2),S=_[0],O=_[1],M=d+w.getDefault(O[n],1);u.has(S)||s.has(S)&&!(M0;){var u=t.pop(),s=(1+o.get(u))/n.get(u);r.get(u).forEach(function(e){o.set(e,o.get(e)+n.get(e)*s)}),(u!==a||"object"==typeof u&&u.toString()!==a.toString())&&e.set(u,e.get(u)+o.get(u))}return e}function c(e,t,r,n,a){var i=this;e.set(a,e.get(a)+t.length-1);for(var o=new w.Map(v.mark(function l(){var e,r,n,a,o,u;return v.wrap(function(i){for(;;)switch(i.prev=i.next){case 0:e=!0,r=!1,n=void 0,i.prev=3,a=b(t);case 5:if(e=(o=a.next()).done){i.next=12;break}return u=o.value,i.next=9,w.tuple2(u,0);case 9:e=!0,i.next=5;break;case 12:i.next=18;break;case 14:i.prev=14,i.t0=i["catch"](3),r=!0,n=i.t0;case 18:i.prev=18,i.prev=19,!e&&a["return"]&&a["return"]();case 21:if(i.prev=21,!r){i.next=24;break}throw n;case 24:return i.finish(21);case 25:return i.finish(18);case 26:case"end":return i.stop()}},l,i,[[3,14,18,26],[19,,21,25]])})());t.length>0;){var u=t.pop(),s=(1+o.get(u))/n.get(u);r.get(u).forEach(function(e){o.set(e,o.get(e)+n.get(e)*s)}),(u!==a||"object"==typeof u&&u.toString()!==a.toString())&&e.set(u,e.get(u)+o.get(u)+1)}return e}function f(e,t,r,n,a){for(var i=this,o=new w.Map(v.mark(function l(){var e,r,n,a,o,u;return v.wrap(function(i){for(;;)switch(i.prev=i.next){case 0:e=!0,r=!1,n=void 0,i.prev=3,a=b(t);case 5:if(e=(o=a.next()).done){i.next=12;break}return u=o.value,i.next=9,w.tuple2(u,0);case 9:e=!0,i.next=5;break;case 12:i.next=18;break;case 14:i.prev=14,i.t0=i["catch"](3),r=!0,n=i.t0;case 18:i.prev=18,i.prev=19,!e&&a["return"]&&a["return"]();case 21:if(i.prev=21,!r){i.next=24;break}throw n;case 24:return i.finish(21);case 25:return i.finish(18);case 26:case"end":return i.stop()}},l,i,[[3,14,18,26],[19,,21,25]])})());t.length>0;){var u=t.pop(),s=(1+o.get(u))/n.get(u);r.get(u).forEach(function(t){var r=n.get(t)*s,a=[t,u];e.has(a)||(a=[u,t]),e.set(a,e.get(a)+r),o.set(t,o.get(t)+r)}),(u!==a||"object"==typeof u&&u.toString()!==a.toString())&&e.set(u,e.get(u)+o.get(u))}return e}function d(e,t,r,n,a){void 0===n&&(n=!1);var i;return i=r?2>=t?null:1/((t-1)*(t-2)):n?null:.5,null!=i&&(null!=a&&(i=i*t/a),e.forEach(function(t,r){return e.set(r,t*i)})),e}function h(e,t,r,n){var a;return a=r?1>=t?null:1/(t*(t-1)):n?null:.5,null!=a&&e.forEach(function(t,r){return e.set(r,t*a)}),e}var p=e("babel-runtime/helpers/sliced-to-array")["default"],v=e("babel-runtime/regenerator")["default"],b=e("babel-runtime/core-js/get-iterator")["default"],g=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.betweennessCentrality=n,r.genBetweennessCentrality=a,r.edgeBetweennessCentrality=i,r.genEdgeBetweennessCentrality=o;var y=e("../../_internals/delegate"),m=g(y),w=e("../../_internals")},{"../../_internals":20,"../../_internals/delegate":12,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109,"babel-runtime/regenerator":166}],44:[function(e,t,r){"use strict";function n(e){var t=arguments.length<=1||void 0===arguments[1]?{}:arguments[1],r=t.maxIter,n=void 0===r?100:r,a=t.tolerance,u=void 0===a?1e-6:a,s=t.nstart,l=t.weight,d=Math.sqrt,h=Math.pow,p=Math.abs;if(e.isMultigraph())throw new c.JSNetworkXException("Not defined for multigraphs.");if(0===e.order())throw new c.JSNetworkXException("Empty graph.");var v=void 0,b=new f.Map;if(s){v=s;var g=!0,y=!1,m=void 0;try{for(var w,x=o(v.keys());!(g=(w=x.next()).done);g=!0){var k=w.value;b.set(k,0)}}catch(j){y=!0,m=j}finally{try{!g&&x["return"]&&x["return"]()}finally{if(y)throw m}}}else{var E=1/e.order();v=new f.Map;var _=!0,S=!1,O=void 0;try{for(var M,I=o(e);!(_=(M=I.next()).done);_=!0){var k=M.value;v.set(k,E),b.set(k,0)}}catch(j){S=!0,O=j}finally{try{!_&&I["return"]&&I["return"]()}finally{if(S)throw O}}}var P=0,N=!0,A=!1,$=void 0;try{for(var q,D=o(v.values());!(N=(q=D.next()).done);N=!0){var L=q.value;P+=L}}catch(j){A=!0,$=j}finally{try{!N&&D["return"]&&D["return"]()}finally{if(A)throw $}}P=1/P;var F=!0,G=!1,C=void 0;try{for(var z,T=o(v);!(F=(z=T.next()).done);F=!0){var J=i(z.value,2),R=J[0],L=J[1];v.set(R,L*P)}}catch(j){G=!0,C=j}finally{try{!F&&T["return"]&&T["return"]()}finally{if(G)throw C}}u=e.order()*u;for(var X=0;n>X;X++){var B=v;v=new f.Map(b);var U=!0,V=!1,W=void 0;try{for(var H,K=o(v);!(U=(H=K.next()).done);U=!0){var Y=i(H.value,2),k=Y[0],L=Y[1],Q=!0,Z=!1,ee=void 0;try{for(var te,re=o(e.get(k));!(Q=(te=re.next()).done);Q=!0){var ne=i(te.value,2),ae=ne[0],ie=ne[1];v.set(ae,v.get(ae)+B.get(k)*f.getDefault(l&&ie[l],1))}}catch(j){Z=!0,ee=j}finally{try{!Q&&re["return"]&&re["return"]()}finally{if(Z)throw ee}}}}catch(j){V=!0,W=j}finally{try{!U&&K["return"]&&K["return"]()}finally{if(V)throw W}}var oe=0,ue=!0,se=!1,le=void 0;try{for(var ce,fe=o(v.values());!(ue=(ce=fe.next()).done);ue=!0){var L=ce.value;oe+=h(L,2)}}catch(j){se=!0,le=j}finally{try{!ue&&fe["return"]&&fe["return"]()}finally{if(se)throw le}}oe=d(oe),oe=0===oe?1:1/oe;var de=0,he=!0,pe=!1,ve=void 0;try{for(var be,ge=o(v);!(he=(be=ge.next()).done);he=!0){var ye=i(be.value,2),k=ye[0],L=ye[1];L*=oe,v.set(k,L),de+=p(L-B.get(k))}}catch(j){pe=!0,ve=j}finally{try{!he&&ge["return"]&&ge["return"]()}finally{if(pe)throw ve}}if(u>de)return v}throw new c.JSNetworkXError("eigenvectorCentrality(): power iteration failed to converge in "+(n+" iterations."))}function a(e,t){return l["default"]("eigenvectorCentrality",[e,t])}var i=e("babel-runtime/helpers/sliced-to-array")["default"],o=e("babel-runtime/core-js/get-iterator")["default"],u=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.eigenvectorCentrality=n,r.genEigenvectorCentrality=a;var s=e("../../_internals/delegate"),l=u(s),c=e("../../exceptions"),f=e("../../_internals")},{"../../_internals":20,"../../_internals/delegate":12,"../../exceptions":78,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109}],45:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-wildcard")["default"],a=e("babel-runtime/helpers/defaults")["default"];Object.defineProperty(r,"__esModule",{value:!0});var i=e("./betweenness"),o=n(i),u=e("./eigenvector"),s=n(u);r.betweenness=o,r.eigenvector=s,a(r,n(i)),a(r,n(u))},{"./betweenness":43,"./eigenvector":44,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-wildcard":108}],46:[function(e,t,r){"use strict";function n(e){var t,r,n,a,i,o,u,s,l,c,f,d,v;return p.wrap(function(p){for(;;)switch(p.prev=p.next){case 0:if(0!==e.numberOfNodes()){p.next=2;break}return p.abrupt("return",[]);case 2:t=new x.Map(x.mapIterator(e,function(t){var r=new x.Set(e.neighborsIter(t));return r["delete"](t),x.tuple2(t,r)})),r=new x.Set(e),n=new x.Set(e),a=[null],i=x.max(r,function(e){return n.intersection(t.get(e)).size}),o=n.difference(t.get(i)),u=[];case 9:if(!(o.size>0)){p.next=25;break}if(s=o.pop(),n["delete"](s),a[a.length-1]=s,l=t.get(s),c=r.intersection(l),0!==c.size){p.next=21;break}return p.next=19,a.slice();case 19:p.next=23;break;case 21:f=n.intersection(l),f.size>0&&(u.push([r,n,o]),a.push(null),r=c,n=f,i=x.max(r,function(e){return n.intersection(t.get(e)).size}),o=n.difference(t.get(i)));case 23:p.next=33;break;case 25:if(0!==a.length&&0!==u.length){p.next=27;break}return p.abrupt("break",35);case 27:a.pop(),d=u.pop(),v=h(d,3),r=v[0],n=v[1],o=v[2];case 33:p.next=9;break;case 35:case"end":return p.stop()}},y[0],this)}function a(e){return w["default"]("findCliques",[e])}function i(e){var t,r,n,a;return p.wrap(function(i){for(;;)switch(i.prev=i.next){case 0:if(a=function o(e,a){var i,u,s,l,c,f,d,h,b,g;return p.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:i=x.max(e,function(e){return a.intersection(r.get(e)).size}),u=!0,s=!1,l=void 0,t.prev=4,c=v(a.difference(r.get(i)));case 6:if(u=(f=c.next()).done){t.next=24;break}if(d=f.value,a["delete"](d),n.push(d),h=r.get(d),b=e.intersection(h),0!==b.size){t.next=17;break}return t.next=15,n.slice();case 15:t.next=20;break;case 17:if(g=a.intersection(h),!(g.size>0)){t.next=20;break}return t.delegateYield(o(b,g),"t0",20);case 20:n.pop();case 21:u=!0,t.next=6;break;case 24:t.next=30;break;case 26:t.prev=26,t.t1=t["catch"](4),s=!0,l=t.t1;case 30:t.prev=30,t.prev=31,!u&&c["return"]&&c["return"]();case 33:if(t.prev=33,!s){t.next=36;break}throw l;case 36:return t.finish(33);case 37:return t.finish(30);case 38:case"end":return t.stop()}},t[0],this,[[4,26,30,38],[31,,33,37]])},t=[a].map(p.mark),0!==e.size){i.next=5;break}return i.next=5,[];case 5:return r=new x.Map(x.mapIterator(e,function(t){var r=new x.Set(e.neighborsIter(t));return r["delete"](t),x.tuple2(t,r)})),n=[],i.delegateYield(a(new x.Set(e),new x.Set(e)),"t0",8);case 8:case"end":return i.stop()}},y[1],this)}function o(e){return w["default"]("findCliquesRecursive",[e])}function u(e,t){return null==t&&(t=n(e)),x.max(t,function(e){return e.length}).length}function s(e,t){return w["default"]("graphCliqueNumber",[e,t])}function l(e,t){return null==t&&(t=n(e)),b(t).length}function c(e,t){return w["default"]("graphNumberOfCliques",[e,t])}function f(e,t,r){r=b(r||n(e)),null==t&&(t=e.nodes());var a;if(Array.isArray(t))r=r.map(function(e){return new x.Set(e)}),a=new x.Map,t.forEach(function(e){a.set(e,r.filter(function(t){return t.has(e)}).length)});else{var i=t;a=r.filter(function(e){return new x.Set(e).has(i)}).length}return a}function d(e,t,r){return w["default"]("numberOfCliques",[e,t,r])}var h=e("babel-runtime/helpers/sliced-to-array")["default"],p=e("babel-runtime/regenerator")["default"],v=e("babel-runtime/core-js/get-iterator")["default"],b=e("babel-runtime/core-js/array/from")["default"],g=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.findCliques=n,r.genFindCliques=a,r.findCliquesRecursive=i,r.genFindCliquesRecursive=o,r.graphCliqueNumber=u,r.genGraphCliqueNumber=s,r.graphNumberOfCliques=l,r.genGraphNumberOfCliques=c,r.numberOfCliques=f,r.genNumberOfCliques=d;var y=[n,i].map(p.mark),m=e("../_internals/delegate"),w=g(m),x=e("../_internals")},{"../_internals":20,"../_internals/delegate":12,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109,"babel-runtime/regenerator":166}],47:[function(e,t,r){"use strict";function n(e,t){if(e.isDirected())throw new E["default"]("triangles() is not defined for directed graphs.");return null!=t&&e.hasNode(t)?Math.floor(_.next(i(e,t))[2]/2):new _.Map(_.mapIterator(i(e,t),function(e){var t=v(e,3),r=t[0],n=(t[1],t[2]);return _.tuple2(r,Math.floor(n/2),r)}))}function a(e,t){return k["default"]("triangles",[e,t])}function i(e,t){var r,n,a,i,o,u,s,l,c,f,d,h,p,y,m,x,k,j;return b.wrap(function(b){for(;;)switch(b.prev=b.next){case 0:if(!e.isMultigraph()){b.next=2;break}throw new E["default"]("Not defined for multigraphs.");case 2:r=_.mapIterator(null==t?e:e.nbunchIter(t),function(t){return _.tuple2(t,e.get(t))}),n=!0,a=!1,i=void 0,b.prev=6,o=g(r);case 8:if(n=(u=o.next()).done){b.next=39;break}for(s=v(u.value,2),l=s[0],c=s[1],f=new _.Set(c.keys()),f["delete"](l),d=0,h=!0,p=!1,y=void 0,b.prev=18,m=g(f);!(h=(x=m.next()).done);h=!0)k=x.value,j=new _.Set(e.get(k).keys()),j["delete"](k),d+=f.intersection(j).size;b.next=26;break;case 22:b.prev=22,b.t0=b["catch"](18),p=!0,y=b.t0;case 26:b.prev=26,b.prev=27,!h&&m["return"]&&m["return"]();case 29:if(b.prev=29,!p){b.next=32;break}throw y;case 32:return b.finish(29);case 33:return b.finish(26);case 34:return b.next=36,_.tuple3(l,f.size,d);case 36:n=!0,b.next=8;break;case 39:b.next=45;break;case 41:b.prev=41,b.t1=b["catch"](6),a=!0,i=b.t1;case 45:b.prev=45,b.prev=46,!n&&o["return"]&&o["return"]();case 48:if(b.prev=48,!a){b.next=51;break}throw i;case 51:return b.finish(48);case 52:return b.finish(45);case 53:case"end":return b.stop()}},w[0],this,[[6,41,45,53],[18,22,26,34],[27,,29,33],[46,,48,52]])}function o(e,t){var r,n,a,i,o,u,s,l,c,f,d,h,p,y,m,x,k,j,S,O,M,I,P,N,A,$,q,D,L,F=arguments.length<=2||void 0===arguments[2]?"weight":arguments[2];return b.wrap(function(b){for(;;)switch(b.prev=b.next){case 0:if(!e.isMultigraph()){b.next=2;break}throw new E["default"]("Not defined for multigraphs.");case 2:r=null==F||0===e.edges().length?1:_.max(_.mapIterator(e.edgesIter(!0),function(e){var t=v(e,3),r=(t[0],t[1],t[2]);return _.getDefault(r[F],1)})),n=_.mapIterator(null==t?e:e.nbunchIter(t),function(t){return _.tuple2(t,e.get(t))}),a=!0,i=!1,o=void 0,b.prev=7,u=g(n);case 9:if(a=(s=u.next()).done){b.next=67;break}l=v(s.value,2),c=l[0],f=l[1],d=new _.Set(f.keys()).difference([c]),h=0,p=new _.Set,y=!0,m=!1,x=void 0,b.prev=19,k=g(d);case 21:if(y=(j=k.next()).done){b.next=48;break}for(S=j.value,O=_.getDefault(f.get(S)[F],1)/r,p.add(S),M=new _.Set(e.get(S).keys()).difference(p),I=!0,P=!1,N=void 0,b.prev=29,A=g(d.intersection(M));!(I=($=A.next()).done);I=!0)q=$.value,D=_.getDefault(e.get(S).get(q)[F],1)/r,L=_.getDefault(f.get(q)[F],1)/r,h+=Math.pow(O*D*L,1/3);b.next=37;break;case 33:b.prev=33,b.t0=b["catch"](29),P=!0,N=b.t0;case 37:b.prev=37,b.prev=38,!I&&A["return"]&&A["return"]();case 40:if(b.prev=40,!P){b.next=43;break}throw N;case 43:return b.finish(40);case 44:return b.finish(37);case 45:y=!0,b.next=21;break;case 48:b.next=54;break;case 50:b.prev=50,b.t1=b["catch"](19),m=!0,x=b.t1;case 54:b.prev=54,b.prev=55,!y&&k["return"]&&k["return"]();case 57:if(b.prev=57,!m){b.next=60;break}throw x;case 60:return b.finish(57);case 61:return b.finish(54);case 62:return b.next=64,_.tuple3(c,d.size,2*h);case 64:a=!0,b.next=9;break;case 67:b.next=73;break;case 69:b.prev=69,b.t2=b["catch"](7),i=!0,o=b.t2;case 73:b.prev=73,b.prev=74,!a&&u["return"]&&u["return"]();case 76:if(b.prev=76,!i){b.next=79;break}throw o;case 79:return b.finish(76);case 80:return b.finish(73);case 81:case"end":return b.stop()}},w[1],this,[[7,69,73,81],[19,50,54,62],[29,33,37,45],[38,,40,44],[55,,57,61],[74,,76,80]])}function u(e,t,r){var n=arguments.length<=3||void 0===arguments[3]?!0:arguments[3],a=y(l(e,t,r).values());return n||(a=a.filter(function(e){return e>0})),a.reduce(function(e,t){return e+t},0)/a.length}function s(e,t,r,n){return k["default"]("averageClustering",[e,t,r,n])}function l(e,t,r){if(e.isDirected())throw new E["default"]("Clustering algorithms are not defined for directed graphs.");var n=null==r?i(e,t):o(e,t,r),a=new _.Map(_.mapIterator(n,function(e){var t=v(e,3),r=t[0],n=t[1],a=t[2];return _.tuple2(r,0===a?0:a/(n*(n-1)))}));return e.hasNode(t)?_.next(a.values()):a}function c(e,t,r){return k["default"]("clustering",[e,t,r])}function f(e){var t=0,r=0,n=!0,a=!1,o=void 0;try{for(var u,s=g(i(e));!(n=(u=s.next()).done);n=!0){var l=v(u.value,3),c=(l[0],l[1]),f=l[2];r+=c*(c-1),t+=f}}catch(d){a=!0,o=d}finally{try{!n&&s["return"]&&s["return"]()}finally{if(a)throw o}}return 0===t?0:t/r}function d(e){return k["default"]("transitivity",[e])}function h(e,t){var r=null==t?e:e.nbunchIter(t),n=new _.Map,a=!0,i=!1,o=void 0;try{for(var u,s=g(r);!(a=(u=s.next()).done);a=!0){var l=u.value;n.set(l,0);var c=0,f=!0,d=!1,h=void 0;try{for(var p,b=g(_.genCombinations(e.get(l).keys(),2));!(f=(p=b.next()).done);f=!0){var y=v(p.value,2),m=y[0],w=y[1],x=new _.Set(e.get(m).keys()).intersection(new _.Set(e.get(w).keys()));x["delete"](l),x=x.size,n.set(l,n.get(l)+x);var k=x+1;e.get(m).has(w)&&(k+=1),c+=(e.get(m).size-k)*(e.get(w).size-k)+x}}catch(j){d=!0,h=j}finally{try{!f&&b["return"]&&b["return"]()}finally{if(d)throw h}}c>0&&n.set(l,n.get(l)/c)}}catch(j){i=!0,o=j}finally{try{!a&&s["return"]&&s["return"]()}finally{if(i)throw o}}return e.hasNode(t)?_.next(n.values()):n}function p(e,t){return k["default"]("squareClustering",[e,t])}var v=e("babel-runtime/helpers/sliced-to-array")["default"],b=e("babel-runtime/regenerator")["default"],g=e("babel-runtime/core-js/get-iterator")["default"],y=e("babel-runtime/core-js/array/from")["default"],m=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.triangles=n,r.genTriangles=a,r.averageClustering=u,r.genAverageClustering=s,r.clustering=l,r.genClustering=c,r.transitivity=f,r.genTransitivity=d,r.squareClustering=h,r.genSquareClustering=p;var w=[i,o].map(b.mark),x=e("../_internals/delegate"),k=m(x),j=e("../exceptions/JSNetworkXError"),E=m(j),_=e("../_internals")},{"../_internals":20,"../_internals/delegate":12,"../exceptions/JSNetworkXError":73,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109,"babel-runtime/regenerator":166}],48:[function(e,t,r){"use strict";function n(e){try{return i(e),!0}catch(t){if(t instanceof g["default"])return!1;throw t}}function a(e){return h["default"]("isDirectedAcyclicGraph",[e])}function i(e,t){if(!e.isDirected())throw new v["default"]("Topological sort not defined on undirected graphs.");var r=new y.Set,n=[],a=new y.Set;return null==t&&(t=e.nodesIter()),y.forEach(t,function(t){if(!a.has(t))for(var i=[t];i.length>0;){var o=i[i.length-1];if(a.has(o))i.pop();else{r.add(o);var u=[];e.get(o).forEach(function(e,t){if(!a.has(t)){if(r.has(t))throw new g["default"]("Graph contains a cycle.");u.push(t)}}),u.length>0?i.push.apply(i,u):(a.add(o),n.unshift(o))}}}),n}function o(e,t){return h["default"]("topologicalSort",[e,t])}function u(e,t){function r(e,t,n,a){return t.add(a),e.get(a).forEach(function(a,i){if(t.has(i)){if(t.has(i)&&-1===n.indexOf(i))throw new g["default"]("Graph contains a cycle.")}else if(!r(e,t,n,i))return!1}),n.unshift(a),!0}if(!e.isDirected())throw new v["default"]("Topological sort not defined on undirected graphs.");var n=new y.Set,a=[];return null==t&&(t=e.nodesIter()),y.forEach(t,function(t){if(-1===a.indexOf(t)&&!r(e,n,a,t))throw new g["default"]("Graph contains a cycle.")}),a}function s(e,t){return h["default"]("topologicalSortRecursive",[e,t])}function l(e){for(var t,r=!0;r;){var n=e;if(a=i=o=u=s=l=c=f=void 0,r=!1,!n.isDirected())throw new v["default"]("is_aperiodic not defined for undirected graphs.");var a=n.nodesIter().next();if(a.done)return!0;var i=new y.Map;i.set(a.value,0);for(var o=[a.value],u=0,s=1;o.length>0;){for(var l=[],c=0;cu;u++){var s=e[u];if(0>s||s>=t)throw new g.JSNetworkXUnfeasible;s>0&&(n=Math.max(n,s),a=Math.min(a,s),i+=s,o+=1,r[s]+=1)}if(i%2===1||i>o*(o-1))throw new g.JSNetworkXUnfeasible;return[n,a,i,o,r]}function s(e){var t,r,n,a,i;try{var o=u(e),s=d(o,5);r=s[0],n=s[1],t=s[2],a=s[3],i=s[4]}catch(l){if(l instanceof g.JSNetworkXUnfeasible)return!1;throw l}if(0===a||4*n*a>=Math.pow(r+n+1,2))return!0;for(var c=m["default"](r+1,0);a>0;){for(;0===i[r];)r-=1;if(r>a-1)return!1;i[r]-=1,a-=1;for(var f=0,h=r,p=0;r>p;p++){for(;0===i[h];)h-=1;i[h]-=1,a-=1,h>1&&(c[f]=h-1,f+=1)}for(p=0;f>p;p++){var v=c[p];i[v]+=1,a+=1}}return!0}function l(e){return b["default"]("isValidDegreeSequenceHavelHakimi",[e])}function c(e){var t,r,n,a,i;try{var o=u(e),s=d(o,5);t=s[0],r=s[1],n=s[2],a=s[3],i=s[4]}catch(l){if(l instanceof g.JSNetworkXUnfeasible)return!1;throw l}if(0===a||4*r*a>=Math.pow(t+r+1,2))return!0;for(var c=0,f=0,h=0,p=0,v=t;v>=r;v-=1){if(c+1>v)return!0;if(i[v]>0){var b=i[v];c+b>v&&(b=v-c),f+=b*v;for(var y=0;b>y;y++)h+=i[c+y],p+=(c+y)*i[c+y];if(c+=b,f>c*(a-1)-c*h+p)return!1}}return!0}function f(e){return b["default"]("isValidDegreeSequenceErdosGallai",[e])}var d=e("babel-runtime/helpers/sliced-to-array")["default"],h=e("babel-runtime/core-js/array/from")["default"],p=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.isGraphical=n,r.genIsGraphical=a,r.isValidDegreeSequence=i,r.genIsValidDegreeSequence=o,r.isValidDegreeSequenceHavelHakimi=s,r.genIsValidDegreeSequenceHavelHakimi=l,r.isValidDegreeSequenceErdosGallai=c,r.genIsValidDegreeSequenceErdosGallai=f;var v=e("../_internals/delegate"),b=p(v),g=e("../exceptions"),y=e("../_internals/fillArray"),m=p(y)},{"../_internals/delegate":12,"../_internals/fillArray":13,"../exceptions":78,"babel-runtime/core-js/array/from":88,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109}],50:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-wildcard")["default"],a=e("babel-runtime/helpers/defaults")["default"];Object.defineProperty(r,"__esModule",{value:!0});var i=e("./centrality"),o=n(i),u=e("./clique"),s=n(u),l=e("./cluster"),c=n(l),f=e("./dag"),d=n(f),h=e("./graphical"),p=n(h),v=e("./isomorphism"),b=n(v),g=e("./operators"),y=n(g),m=e("./shortestPaths"),w=n(m);r.centrality=o,r.clique=s,r.cluster=c,r.dag=d,r.graphical=p,r.isomorphism=b,r.operators=y,r.shortestPaths=w,a(r,n(i)),a(r,n(u)),a(r,n(l)),a(r,n(f)),a(r,n(h)),a(r,n(v)),a(r,n(g)),a(r,n(m))},{"./centrality":45,"./clique":46,"./cluster":47,"./dag":48,"./graphical":49,"./isomorphism":51,"./operators":54,"./shortestPaths":56,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-wildcard":108}],51:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-wildcard")["default"],a=e("babel-runtime/helpers/defaults")["default"];Object.defineProperty(r,"__esModule",{value:!0});var i=e("./isomorph"),o=n(i);r.isomorph=o,a(r,n(i))},{"./isomorph":52,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-wildcard":108}],52:[function(e,t,r){"use strict";function n(e,t){if(e.order()!==t.order())return!1; + var r=e.degree(),n=p.triangles(e),a=h.numberOfCliques(e),i=[];r.forEach(function(e,t){i.push([r.get(t),n.get(t),a.get(t)])}),i.sort(function(e,t){return e[0]-t[0]||e[1]-t[1]||e[2]-t[2]});var o=t.degree(),u=p.triangles(t),s=h.numberOfCliques(t),l=[];return o.forEach(function(e,t){l.push([o.get(t),u.get(t),s.get(t)])}),l.sort(function(e,t){return e[0]-t[0]||e[1]-t[1]||e[2]-t[2]}),i.every(function(e,t){var r=l[t];return e[0]===r[0]&&e[1]===r[1]&&e[2]===r[2]})}function a(e,t){return d["default"]("couldBeIsomorphic",[e,t])}function i(e,t){if(e.order()!==t.order())return!1;var r=e.degree(),n=p.triangles(e),a=[];r.forEach(function(e,t){a.push([r.get(t),n.get(t)])}),a.sort(function(e,t){return e[0]-t[0]||e[1]-t[1]});var i=t.degree(),o=p.triangles(t),u=[];return i.forEach(function(e,t){u.push([i.get(t),o.get(t)])}),u.sort(function(e,t){return e[0]-t[0]||e[1]-t[1]}),a.every(function(e,t){var r=u[t];return e[0]===r[0]&&e[1]===r[1]})}function o(e,t){return d["default"]("fastCouldBeIsomorphic",[e,t])}function u(e,t){if(e.order()!==t.order())return!1;var r=l(e.degree().values());r.sort(function(e,t){return e-t});var n=l(t.degree().values());return n.sort(function(e,t){return e-t}),r.every(function(e,t){return e===n[t]})}function s(e,t){return d["default"]("fasterCouldBeIsomorphic",[e,t])}var l=e("babel-runtime/core-js/array/from")["default"],c=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.couldBeIsomorphic=n,r.genCouldBeIsomorphic=a,r.fastCouldBeIsomorphic=i,r.genFastCouldBeIsomorphic=o,r.fasterCouldBeIsomorphic=u,r.genFasterCouldBeIsomorphic=s;var f=e("../../_internals/delegate"),d=c(f),h=e("../clique"),p=e("../cluster")},{"../../_internals/delegate":12,"../clique":46,"../cluster":47,"babel-runtime/core-js/array/from":88,"babel-runtime/helpers/interop-require-default":107}],53:[function(e,t,r){"use strict";function n(e,t){var r=new S["default"](t),n=new S["default"](e);if(r.size!==n.size||O.someIterator(n.values(),function(e){return!r.has(e)}))throw new k["default"]("Node sets of graphs are not equal.")}function a(e,t){function r(e,t){return t?j.relabelNodes(e,function(e){return t+e.toString()}):e}var n=arguments.length<=2||void 0===arguments[2]?{}:arguments[2],a=n.rename,i=void 0===a?[null,null]:a;if(e.isMultigraph()!==t.isMultigraph())throw new k["default"]("G and H must both be graphs or multigraphs");var o=new e.constructor;if(o.name="union("+e.name+", "+t.name+")",e=r(e,i[0]),t=r(t,i[1]),new S["default"](e).intersection(new S["default"](t)).size>0)throw new k["default"]("The node sets of G and H are not disjoint. Use appropriate {rename: [Gprefix, Hprefix]} or use disjointUnion({G, H})");return o.addNodesFrom(e.nodesIter(!0)),o.addNodesFrom(t.nodesIter(!0)),o.addEdgesFrom(e.isMultigraph()?e.edgesIter(!0,!0):e.edgesIter(!0)),o.addEdgesFrom(t.isMultigraph()?t.edgesIter(!0,!0):t.edgesIter(!0)),b(o.graph,e.graph,t.graph),o}function i(e,t,r){return w["default"]("union",[e,t,r])}function o(e,t){var r=j.convertNodeLabelsToIntegers(e),n=j.convertNodeLabelsToIntegers(t,r.order()),i=a(r,n);return i.name="disjointUnion("+e.name+", "+t.name+")",b(i.graph,e.graph,t.graph),i}function u(e,t){return w["default"]("disjointUnion",[e,t])}function s(e,t){if(e.isMultigraph()!==t.isMultigraph())throw new k["default"]("G and H must both be graphs or multigraphs");var r=E.createEmptyCopy(e);r.name="Intersection of ("+e.name+" and "+t.name+")",n(e,t);var a=e.numberOfEdges()0;){var o=i;i=new k.Map;var u=!0,s=!1,l=void 0;try{for(var c,f=g(o.keys());!(u=(c=f.next()).done);u=!0){var d=c.value;n.has(d)||(n.set(d,a),e.get(d).forEach(function(e,t){return i.set(t,1)}))}}catch(h){s=!0,l=h}finally{try{!u&&f["return"]&&f["return"]()}finally{if(s)throw l}}if(null!=r&&a>=r)break;a+=1}return n}function a(e,t,r){return w["default"]("singleSourceShortestPathLength",[e,t,r])}function i(e,t){var r=new k.Map,a=!0,i=!1,o=void 0;try{for(var u,s=g(e);!(a=(u=s.next()).done);a=!0){var l=u.value;r.set(l,n(e,l,t))}}catch(c){i=!0,o=c}finally{try{!a&&s["return"]&&s["return"]()}finally{if(i)throw o}}return r}function o(e,t){return w["default"]("allPairsShortestPathLength",[e,t])}function u(e,t,r){for(var n=l(e,t,r),a=b(n,3),i=a[0],o=a[1],u=a[2],s=[];null!=u;)s.push(u),u=i.get(u);for(u=o.get(s[0]),s.reverse();null!=u;)s.push(u),u=o.get(u);return s}function s(e,t,r){return w["default"]("bidirectionalShortestPath",[e,t,r])}function l(e,t,r){if(k.nodesAreEqual(t,r))return[new k.Map([[t,null]]),new k.Map([[r,null]]),t];var n,a;e.isDirected()?(n=e.predecessorsIter.bind(e),a=e.successorsIter.bind(e)):(n=e.neighborsIter.bind(e),a=e.neighborsIter.bind(e));for(var i,o=new k.Map([[t,null]]),u=new k.Map([[r,null]]),s=[t],l=[r];s.length>0&&l.length>0;)if(s.length<=l.length){i=s,s=[];var c=!0,f=!1,d=void 0;try{for(var h,p=g(i);!(c=(h=p.next()).done);c=!0){var v=h.value,b=!0,y=!1,m=void 0;try{for(var w,j=g(a(v));!(b=(w=j.next()).done);b=!0){var E=w.value;if(o.has(E)||(s.push(E),o.set(E,v)),u.has(E))return[o,u,E]}}catch(_){y=!0,m=_}finally{try{!b&&j["return"]&&j["return"]()}finally{if(y)throw m}}}}catch(_){f=!0,d=_}finally{try{!c&&p["return"]&&p["return"]()}finally{if(f)throw d}}}else{i=l,l=[];var S=!0,O=!1,M=void 0;try{for(var I,P=g(i);!(S=(I=P.next()).done);S=!0){var v=I.value,N=!0,A=!1,$=void 0;try{for(var q,D=g(n(v));!(N=(q=D.next()).done);N=!0){var E=q.value;if(u.has(E)||(l.push(E),u.set(E,v)),o.has(E))return[o,u,E]}}catch(_){A=!0,$=_}finally{try{!N&&D["return"]&&D["return"]()}finally{if(A)throw $}}}}catch(_){O=!0,M=_}finally{try{!S&&P["return"]&&P["return"]()}finally{if(O)throw M}}}throw new x.JSNetworkXNoPath(k.sprintf("No path between `%j` and `%j`.",t,r))}function c(e,t,r){var n=0,a=new k.Map([[t,1]]),i=new k.Map([[t,[t]]]);if(0===r)return i;for(;a.size>0;){var o=a;a=new k.Map;var u=!0,s=!1,l=void 0;try{for(var c,f=g(o.keys());!(u=(c=f.next()).done);u=!0){var d=c.value,h=!0,p=!1,v=void 0;try{for(var b,y=g(e.get(d).keys());!(h=(b=y.next()).done);h=!0){var m=b.value;i.has(m)||(i.set(m,i.get(d).concat([m])),a.set(m,1))}}catch(w){p=!0,v=w}finally{try{!h&&y["return"]&&y["return"]()}finally{if(p)throw v}}}}catch(w){s=!0,l=w}finally{try{!u&&f["return"]&&f["return"]()}finally{if(s)throw l}}if(n+=1,null!=r&&n>=r)break}return i}function f(e,t,r){return w["default"]("singleSourceShortestPath",[e,t,r])}function d(e,t){var r=new k.Map,n=!0,a=!1,i=void 0;try{for(var o,u=g(e);!(n=(o=u.next()).done);n=!0){var s=o.value;r.set(s,c(e,s,t))}}catch(l){a=!0,i=l}finally{try{!n&&u["return"]&&u["return"]()}finally{if(a)throw i}}return r}function h(e,t){return w["default"]("allPairsShortestPath",[e,t])}function p(e,t){for(var r=arguments.length<=2||void 0===arguments[2]?{}:arguments[2],n=r.target,a=r.cutoff,i=r.returnSeen,o=0,u=[t],s=new k.Map([[t,o]]),l=new k.Map([[t,[]]]);u.length>0;){o+=1;var c=u;if(u=[],c.forEach(function(t){e.get(t).forEach(function(e,r){s.has(r)?s.get(r)===o&&l.get(r).push(t):(l.set(r,[t]),s.set(r,o),u.push(r))})}),null!=a&&o>=a)break}return null!=n?i?l.has(n)?[l.get(n),s.get(n)]:[[],-1]:k.getDefault(l.get(n),[]):i?[l,s]:l}function v(e,t,r){return w["default"]("predecessor",[e,t,r])}var b=e("babel-runtime/helpers/sliced-to-array")["default"],g=e("babel-runtime/core-js/get-iterator")["default"],y=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.singleSourceShortestPathLength=n,r.genSingleSourceShortestPathLength=a,r.allPairsShortestPathLength=i,r.genAllPairsShortestPathLength=o,r.bidirectionalShortestPath=u,r.genBidirectionalShortestPath=s,r.singleSourceShortestPath=c,r.genSingleSourceShortestPath=f,r.allPairsShortestPath=d,r.genAllPairsShortestPath=h,r.predecessor=p,r.genPredecessor=v;var m=e("../../_internals/delegate"),w=y(m),x=e("../../exceptions"),k=e("../../_internals")},{"../../_internals":20,"../../_internals/delegate":12,"../../exceptions":78,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109}],58:[function(e,t,r){"use strict";function n(e,t){var r=t.source,n=t.target,a=t.weight,i=void 0===a?"weight":a,o=d(e,{source:r,target:n,weight:i}),u=m(o,2),s=(u[0],u[1]),l=s.get(n);if(!l)throw new S["default"](E.sprintf("Node %j is not reachable from %j",r,n));return l}function a(e,t){return j["default"]("dijkstraPath",[e,t])}function i(e,t){var r=t.source,n=t.target,a=t.weight,i=void 0===a?"weight":a,o=c(e,{source:r,weight:i}),u=o.get(n);if(null==u)throw new S["default"](E.sprintf("Node %j is not reachable from %j",r,n));return u}function o(e,t){return j["default"]("dijkstraPathLength",[e,t])}function u(e,t){var r=1/0;for(var n in e){var a=E.getDefault(e[n][t],1);r>a&&(r=a)}return r}function s(e,t){var r=t.source,n=t.cutoff,a=t.weight,i=void 0===a?"weight":a,o=d(e,{source:r,cutoff:n,weight:i}),u=m(o,2),s=(u[0],u[1]);return s}function l(e,t){return j["default"]("singleSourceDijkstraPath",[e,t])}function c(e,t){var r=t.source,n=t.cutoff,a=t.weight,i=void 0===a?"weight":a,o=new E.Map,s=new E.Map([[r,0]]),l=new E.PriorityQueue,c=0;for(l.enqueue(0,[c++,r]);l.size>0;){var f=l.dequeue(),d=m(f,2),h=d[0],p=m(d[1],2),v=(p[0],p[1]);if(!o.has(v)){o.set(v,h);var b=void 0;b=e.isMultigraph()?E.mapIterator(e.get(v),function(e){var t=m(e,2),r=t[0],n=t[1];return[r,y({},i,u(n,i))]}):e.get(v);var g=!0,x=!1,k=void 0;try{for(var j,_=w(b);!(g=(j=_.next()).done);g=!0){var S=m(j.value,2),O=S[0],M=S[1],I=h+E.getDefault(M[i],1);if(!(null!=n&&I>n))if(o.has(O)){if(I0;){var h=f.dequeue(),p=m(h,2),v=p[0],b=m(p[1],2),g=(b[0],b[1]);if(!s.has(g)){if(s.set(g,v),E.nodesAreEqual(g,n))break;var x=void 0;x=e.isMultigraph()?E.mapIterator(e.get(g),function(e){var t=m(e,2),r=t[0],n=t[1];return[r,y({},o,u(n,o))]}):e.get(g);var k=!0,j=!1,_=void 0;try{for(var S,O=w(x);!(k=(S=O.next()).done);k=!0){var M=m(S.value,2),I=M[0],P=M[1],N=v+E.getDefault(P[o],1);if(!(null!=a&&N>a))if(s.has(I)){if(N0)if(e.isDirected()){var i=0,o=0,u=!0,s=!1,l=void 0;try{for(var c,f=E(e.inDegree().values());!(u=(c=f.next()).done);u=!0){var d=c.value;i+=d}}catch(h){s=!0,l=h}finally{try{!u&&f["return"]&&f["return"]()}finally{if(s)throw l}}var p=!0,v=!1,b=void 0;try{for(var g,y=E(e.outDegree().values());!(p=(g=y.next()).done);p=!0){var m=g.value;o+=m}}catch(h){v=!0,b=h}finally{try{!p&&y["return"]&&y["return"]()}finally{if(v)throw b}}r+=I.sprintf("Average in degree: %s\nAverage out degree: %s",(i/a).toFixed(4),(o/a).toFixed(4))}else{var w=0,x=!0,k=!1,j=void 0;try{for(var _,S=E(e.degree().values());!(x=(_=S.next()).done);x=!0){var O=_.value;w+=O}}catch(h){k=!0,j=h}finally{try{!x&&S["return"]&&S["return"]()}finally{if(k)throw j}}r+=I.sprintf("Average degree: %s",(w/a).toFixed(4))}}else{if(!e.hasNode(t))throw new M["default"](I.sprintf("Node %j not in graph.",t));r=I.sprintf("Node %j has the following properties:\nDegree: %s\nNeighbors: %s",t,e.degree(t),e.neighbors(t).map(function(e){return JSON.stringify(e)}).join(" "))}return r}function m(e,t,r){if(I.isMap(r))r.forEach(function(r,n){return e.node.get(n)[t]=r});else{if(!I.isPlainObject(r))throw new TypeError("Attributes must be a Map or a plain object");for(var n in r)n=isNaN(n)?n:+n,e.node.get(n)[t]=r[n]}}function w(e,t){var r=new _;return e.node.forEach(function(e,n){n.hasOwnProperty(t)&&r.set(e,n[t])}),r}function x(e,t,r){r.forEach(function(r,n){e.get(r[0]).get(r[1])[t]=n})}function k(e,t){var r=new _;return e.edges(null,!0).forEach(function(e){if(e[2].hasOwnProperty(t)){var n=e[2][t];e.length=2,r.set(e,n)}}),r}var j=e("babel-runtime/core-js/array/from")["default"],E=e("babel-runtime/core-js/get-iterator")["default"],_=e("babel-runtime/core-js/map")["default"],S=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.nodes=n,r.nodesIter=a,r.edges=i,r.edgesIter=o,r.degree=u,r.neighbors=s,r.numberOfNodes=l,r.numberOfEdges=c,r.density=f,r.degreeHistogram=d,r.isDirected=h,r.freeze=p,r.isFrozen=v,r.subgraph=b,r.createEmptyCopy=g,r.info=y,r.setNodeAttributes=m,r.getNodeAttributes=w,r.setEdgeAttributes=x,r.getEdgeAttributes=k;var O=e("../exceptions/JSNetworkXError"),M=S(O),I=e("../_internals")},{"../_internals":20,"../exceptions/JSNetworkXError":73,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/core-js/map":91,"babel-runtime/helpers/interop-require-default":107}],65:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"],a=e("babel-runtime/helpers/interop-require-wildcard")["default"],i=e("babel-runtime/helpers/defaults")["default"];Object.defineProperty(r,"__esModule",{value:!0});var o=e("./Graph"),u=n(o),s=e("./DiGraph"),l=n(s),c=e("./MultiGraph"),f=n(c),d=e("./MultiDiGraph"),h=n(d),p=e("./functions"),v=a(p);r.Graph=u["default"],r.DiGraph=l["default"],r.MultiGraph=f["default"],r.MultiDiGraph=h["default"],r.functions=v,i(r,a(p))},{"./DiGraph":60,"./Graph":61,"./MultiDiGraph":62,"./MultiGraph":63,"./functions":64,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/interop-require-wildcard":108}],66:[function(e,t,r){"use strict";function n(e,t){var r=new h.Map;if(null!=t)s(t).forEach(function(n){return r.set(n,e.neighbors(n).filter(function(e){return t.indexOf(e)>-1}))});else{var n=!0,a=!1,i=void 0;try{for(var o,u=l(e);!(n=(o=u.next()).done);n=!0){var c=o.value;r.set(c,e.neighbors(c))}}catch(f){a=!0,i=f}finally{try{!n&&u["return"]&&u["return"]()}finally{if(a)throw i}}}return r}function a(e,t){var r=d["default"](t);if(r.addNodesFrom(e.keys()),r.isMultigraph()&&!r.isDirected()){var n=new h.Set;e.forEach(function(e,t){e.forEach(function(e){n.has(e)||r.addEdge(t,e)}),n.add(t)})}else e.forEach(function(e,t){e.forEach(function(e){return r.addEdge(t,e)})});return r}function i(e,t,r){var n=new h.Map;if(null!=t)t=s(t),t.forEach(function(a){var i=n.set(a,new h.Map);e.get(a).forEach(function(e,n){t.indexOf(e)>-1&&i.set(e,null==r?n:r)})});else{var a=!0,i=!1,o=void 0;try{for(var c,f=function(){var e=u(c.value,2),t=e[0],a=e[1],i=n.set(a,new h.Map);t.forEach(function(e,t){i.set(t,null==r?e:r)})},d=l(e.adjacencyIter());!(a=(c=d.next()).done);a=!0)f()}catch(p){i=!0,o=p}finally{try{!a&&d["return"]&&d["return"]()}finally{if(i)throw o}}}return n}function o(e,t,r){var n=d["default"](t),a=new h.Set;if(n.addNodesFrom(e.keys()),r)if(n.isDirected())e.forEach(function(e,t){if(h.isArrayLike(e))throw new TypeError("Value is not a map.");e.forEach(function(e,r){for(var a in e){var i=e[a];n.isMultigraph()?n.addEdge(t,r,a,i):n.addEdge(t,r,i)}})});else{var i=n.isMultigraph();e.forEach(function(e,t){if(h.isArrayLike(e))throw new TypeError("Not a map");e.forEach(function(e,r){if(!a.has(h.tuple2(t,r))){for(var o in e){var u=e[o];i?n.addEdge(t,r,o,u):n.addEdge(t,r,u)}a.add(h.tuple2(r,t))}})})}else n.isMultigraph()&&!n.isDirected()?e.forEach(function(e,t){if(h.isArrayLike(e))throw new TypeError("Value is not a map");e.forEach(function(e,r){a.has(h.tuple2(t,r))||(n.addEdge(t,r,e),a.add(h.tuple2(r,t)))})}):e.forEach(function(e,t){if(h.isArrayLike(e))throw new TypeError("Value is not a map");e.forEach(function(e,r){n.addEdge(t,r,e)})});return n}var u=e("babel-runtime/helpers/sliced-to-array")["default"],s=e("babel-runtime/core-js/array/from")["default"],l=e("babel-runtime/core-js/get-iterator")["default"],c=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.toMapOfLists=n,r.fromMapOfLists=a,r.toMapOfMaps=i,r.fromMapOfMaps=o;var f=e("./prepCreateUsing"),d=c(f),h=e("../_internals")},{"../_internals":20,"./prepCreateUsing":68,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109}],67:[function(e,t,r){"use strict";function n(e){function t(e,t,a,i){var o=r[e.type];if(o){for(var u=0,s=o.length;s>u&&!e.isPropgationStopped();u+=3)o[u+2]&&o[u].call(o[u+1]||t,e);if(!e.isDefaultPrevented()&&(i?n[a].apply(t,i):n[a].call(t),!e.isPropgationStopped()))for(u=0,s=o.length;s>u&&!e.isPropgationStopped();u+=3)o[u+2]||o[u].call(o[u+1]||t,e)}}if("function"==typeof e.on)return e;var r={addNodes:[],removeNodes:[],addEdges:[],removeEdges:[],clear:[]},n=e.constructor.prototype;return e.on=function(e,t,n,a){if(!r[e])throw new Error('Event "'+e+'" is not supported.');r[e].push(t,n,!!a)},e.off=function(e,t,n){var a,i,o;if(1===arguments.length)r[e].length=0;else if(2===arguments.length)for(a=r[e],i=a.length-2,"function"!=typeof t&&(i+=1),o=i;o>0;o-=2)a[o]===t&&a.splice(o,3);else for(a=r[e],i=a.length-2,o=i;o>0;o-=2)a[o]===t&&a[o+1]===n&&a.splice(o,2)},e.addNode=function(r){var n=e.hasNode(r)?[]:[r],a=new c("addNodes",this);a.nodes=[r],a.newNodes=n,t(a,this,"addNode",arguments)},e.addNodesFrom=function(r){var n=[],a=[],i=!0,o=!1,u=void 0;try{for(var f,d=s(r);!(i=(f=d.next()).done);i=!0){var h=f.value,p=Array.isArray(h)?h[0]:h;n.push(Array.isArray(h)?h.slice():h),e.hasNode(p)||a.push(p)}}catch(v){o=!0,u=v}finally{try{!i&&d["return"]&&d["return"]()}finally{if(o)throw u}}var b=new c("addNodes",this);b.nodes=n.filter(function(e){return Array.isArray(e)?e[0]:e}),b.newNodes=a;var g=l(arguments);g[0]=n,t(b,this,"addNodesFrom",g)},e.addEdge=function(e,r){var n=[[e,r]],a=this.hasEdge(e,r)?[]:n,i=new c("addEdges",this);i.edges=n,i.newEdges=a,t(i,this,"addEdge",arguments)},e.addEdgesFrom=function(e){var r=[],n=[],a=!0,i=!1,o=void 0;try{for(var u,f=s(e);!(a=(u=f.next()).done);a=!0){var d=u.value;r.push(d.slice()),this.hasEdge(d[0],d[1])||n.push(d.slice(0,2))}}catch(h){i=!0,o=h}finally{try{!a&&f["return"]&&f["return"]()}finally{if(i)throw o}}var p=new c("addEdges",this);p.edges=r,p.newEdges=n;var v=l(arguments);v[0]=r,t(p,this,"addEdgesFrom",v)},e.removeNode=function(e){var r=new c("removeNodes",this);r.nodes=[e],t(r,this,"removeNode",arguments)},e.removeNodesFrom=function(e){var r=[],n=!0,a=!1,i=void 0;try{for(var o,u=s(e);!(n=(o=u.next()).done);n=!0){var f=o.value;r.push(Array.isArray(f)?f.slice():f)}}catch(d){a=!0,i=d}finally{try{!n&&u["return"]&&u["return"]()}finally{if(a)throw i}}var h=new c("removeNodes",this);h.nodes=r;var p=l(arguments);p[0]=r,t(h,this,"removeNodesFrom",p)},e.removeEdge=function(e,r){var n=new c("removeEdges",this);n.edges=[[e,r]],t(n,this,"removeEdge",arguments)},e.removeEdgesFrom=function(e){var r=[],n=!0,a=!1,i=void 0;try{for(var o,u=s(e);!(n=(o=u.next()).done);n=!0){var f=o.value;r.push(f.slice())}}catch(d){a=!0,i=d}finally{try{!n&&u["return"]&&u["return"]()}finally{if(a)throw i}}var h=new c("removeEdges");h.edges=r;var p=l(arguments);p[0]=r,t(h,this,"removeEdgesFrom",p)},e.clear=function(){t(new c("clear",this),this,"clear")},e}function a(e){var t=e.constructor.prototype;return"function"!=typeof e.on?e:(e.addNode=t.addNode,e.addNodesFrome=t.addNodesFrom,e.addEdge=t.addEdge,e.addEdgesFrome=t.addEdgesFrom,e.removeNode=t.removeNode,e.removeEdge=t.removeEdge,e.removeNodesFrom=t.removeNodesFrom,e.removeEdgesFrom=t.removeEdgesFrom,e.clear=t.clear,delete e.on,delete e.off,e)}function i(e){return"function"==typeof e.on&&"function"==typeof e.off}var o=e("babel-runtime/helpers/create-class")["default"],u=e("babel-runtime/helpers/class-call-check")["default"],s=e("babel-runtime/core-js/get-iterator")["default"],l=e("babel-runtime/core-js/array/from")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.observe=n,r.unobserve=a,r.isObservable=i;var c=function(){function e(t,r){u(this,e),this.type=t,this.target=r,this._defaultAction=!0,this._propagate=!0}return o(e,[{key:"stopPropagation",value:function(){this._propagate=!1}},{key:"isPropgationStopped",value:function(){return!this._propagate}},{key:"preventDefault",value:function(){this._defaultAction=!1}},{key:"isDefaultPrevented",value:function(){return!this._defaultAction}}]),e}()},{"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/class-call-check":101,"babel-runtime/helpers/create-class":102}],68:[function(e,t,r){"use strict";function n(t){var r,n=e("../classes/Graph");if(null==t)r=new n;else{r=t;try{r.clear()}catch(a){throw new TypeError("Input graph is not a jsnx graph type")}}return r}Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=n,t.exports=r["default"]},{"../classes/Graph":61}],69:[function(e,t,r){"use strict";function n(e,t){var r=arguments.length<=2||void 0===arguments[2]?!1:arguments[2],n=null;if(S.call(e,"adj"))try{return n=w.fromMapOfMaps(e.adj,t,e.isMultigraph()),S.call(e,"graph")&&"object"==typeof e.graph&&(n.graph=_.clone(e.graph)),S.call(e,"node")&&_.isMap(e.node)&&(n.node=new _.Map,e.node.forEach(function(e,t){return n.node.set(t,_.clone(e))})),n}catch(a){throw a}if(_.isMap(e))try{return w.fromMapOfMaps(e,t,r)}catch(i){try{return w.fromMapOfLists(e,t)}catch(a){throw new Error("Map data structure cannot be converted to a graph.")}}if(_.isPlainObject(e))try{return l(e,t,r)}catch(i){try{return u(e,t)}catch(a){throw new Error("Object data structure cannot be converted to a graph.")}}if(_.isArrayLike(e))try{return f(e,t)}catch(i){throw new Error("Input is not a valid edge list")}return n}function a(e){return e.toUndirected()}function i(e){return e.toDirected()}function o(e,t){var r=function(e){return t.indexOf(e)>-1},n=h(null);null==t?(t=e,r=function(e){return t.hasNode(e)}):t=p(t);var a=!0,i=!1,o=void 0;try{for(var u,s=v(t);!(a=(u=s.next()).done);a=!0){var l=u.value;n[l]=e.neighbors(l).filter(r)}}catch(c){i=!0,o=c}finally{try{!a&&s["return"]&&s["return"]()}finally{if(i)throw o}}return n}function u(e,t){var r=k["default"](t);r.addNodesFrom(b.mark(function u(){var t;return b.wrap(function(r){for(;;)switch(r.prev=r.next){case 0:r.t0=b.keys(e);case 1:if((r.t1=r.t0()).done){r.next=7;break}return t=r.t1.value,r.next=5,isNaN(t)?t:+t;case 5:r.next=1;break;case 7:case"end":return r.stop()}},u,this)})());var n,a;if(r.isMultigraph()&&!r.isDirected()){var i=new _.Set;for(n in e)a=e[n],n=isNaN(n)?n:+n,_.forEach(a,function(e){i.has(e)||r.addEdge(n,e)}),i.add(n)}else{var o=[];for(n in e)a=e[n],n=isNaN(n)?n:+n,_.forEach(a,function(e){o.push([n,e])});r.addEdgesFrom(o)}return r}function s(e,t,r){var n={};if(null!=t)t=p(t),null!=r?t.forEach(function(a){n[a]={},e.get(a).forEach(function(e,i){t.indexOf(i)>-1&&(n[a][i]=r)})}):t.forEach(function(r){n[r]={},e.get(r).forEach(function(e,a){t.indexOf(a)>-1&&(n[r][a]=e)})});else if(null!=r){var a=!0,i=!1,o=void 0;try{for(var u,s=v(e.adjacencyIter());!(a=(u=s.next()).done);a=!0){var l=d(u.value,2),c=l[0],f=l[1];n[f]=E["default"](c,function(){return r})}}catch(h){i=!0,o=h}finally{try{!a&&s["return"]&&s["return"]()}finally{if(i)throw o}}}else{var b=!0,g=!1,y=void 0;try{for(var m,w=v(e.adjacencyIter());!(b=(m=w.next()).done);b=!0){var x=d(m.value,2),c=x[0],f=x[1];n[f]=_.clone(c)}}catch(h){g=!0,y=h}finally{try{!b&&w["return"]&&w["return"]()}finally{if(g)throw y}}}return n}function l(e,t){var r=arguments.length<=2||void 0===arguments[2]?!1:arguments[2],n=k["default"](t),a=new _.Set;if(n.addNodesFrom(b.mark(function f(){var t;return b.wrap(function(r){for(;;)switch(r.prev=r.next){case 0:r.t0=b.keys(e);case 1:if((r.t1=r.t0()).done){r.next=7;break}return t=r.t1.value,r.next=5,isNaN(t)?t:+t;case 5:r.next=1;break;case 7:case"end":return r.stop()}},f,this)})()),r)if(n.isDirected())for(var i in e){var o=e[i];if(_.isArrayLike(o))throw new TypeError("Inner object seems to be an array");i=isNaN(i)?i:+i;for(var u in o){var s=o[u];u=isNaN(u)?u:+u;for(var l in s)n.isMultigraph()?n.addEdge(i,u,l,s[l]):n.addEdge(i,u,s[l])}}else for(var i in e){var o=e[i];if(_.isArrayLike(o))throw new TypeError("Inner object seems to be an array");i=isNaN(i)?i:+i;for(var u in o){var s=o[u];if(u=isNaN(u)?u:+u,!a.has([i,u])){for(var l in s)n.isMultigraph()?n.addEdge(i,u,l,s[l]):n.addEdge(i,u,s[l]);a.add([u,i])}}}else if(n.isMultigraph()&&!n.isDirected())for(var i in e){var o=e[i];if(_.isArrayLike(o))throw new TypeError("Inner object seems to be an array");i=isNaN(i)?i:+i;for(var u in o){var c=o[u];u=isNaN(u)?u:+u,a.has([i,u])||(n.addEdge(i,u,c),a.add([u,i]))}}else for(var i in e){var o=e[i];if(_.isArrayLike(o))throw new TypeError("Inner object seems to be an array");i=isNaN(i)?i:+i;for(var u in o){var c=o[u];u=isNaN(u)?u:+u,n.addEdge(i,u,c)}}return n}function c(e,t){return null!=t?e.edges(t,!0):e.edges(null,!0)}function f(e,t){var r=k["default"](t);return r.addEdgesFrom(e),r}var d=e("babel-runtime/helpers/sliced-to-array")["default"],h=e("babel-runtime/core-js/object/create")["default"],p=e("babel-runtime/core-js/array/from")["default"],v=e("babel-runtime/core-js/get-iterator")["default"],b=e("babel-runtime/regenerator")["default"],g=e("babel-runtime/helpers/interop-require-wildcard")["default"],y=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.toNetworkxGraph=n,r.convertToUndirected=a,r.convertToDirected=i,r.toDictOfLists=o,r.fromDictOfLists=u,r.toDictOfDicts=s,r.fromDictOfDicts=l,r.toEdgelist=c,r.fromEdgelist=f;var m=e("./contrib/convert"),w=g(m),x=e("./contrib/prepCreateUsing"),k=y(x),j=e("lodash/object/mapValues"),E=y(j),_=e("./_internals"),S=Object.prototype.hasOwnProperty},{"./_internals":20,"./contrib/convert":66,"./contrib/prepCreateUsing":68,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/core-js/object/create":93,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/interop-require-wildcard":108,"babel-runtime/helpers/sliced-to-array":109,"babel-runtime/regenerator":166,"lodash/object/mapValues":237}],70:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"],a=e("babel-runtime/helpers/defaults")["default"],i=e("babel-runtime/helpers/interop-require-wildcard")["default"];Object.defineProperty(r,"__esModule",{value:!0});var o=e("./svg"),u=n(o);r.svg=u["default"],a(r,i(o))},{"./svg":71,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/interop-require-wildcard":108}],71:[function(e,t,r){(function(t){"use strict";function n(e,t,r,n){return 180*Math.atan2(n-t,r-e)/Math.PI}function a(e){if(Array.isArray(e))return e;var t=[],r=0,n=!0,a=!1,i=void 0;try{for(var o,u=m(e);!(n=(o=u.next()).done);n=!0){var s=o.value;t[r++]=Array.isArray(s)?w(s):s}}catch(l){a=!0,i=l}finally{try{!n&&u["return"]&&u["return"]()}finally{if(a)throw i}}return t}function i(e,t,r){if("boolean"==typeof t&&(r=t,t=null),t=t||_||{},_=t,t.d3&&(M=t.d3),t=k.deepmerge({},I,t),!M)throw new Error("D3 requried for draw()");if(null==t.element&&null==E)throw new Error("Output element required for draw()");E=t.element||E,M.select(E).select("svg.jsnx").remove();var a,i,c,f=M.select(E),d=[],h=[],y=f.append("svg").classed("jsnx",!0).attr("pointer-events","all"),w=y.append("g"),x=w.append("g").classed("edges",!0).selectAll("g.edge"),S=w.append("g").classed("nodes",!0).selectAll("g.node"),O=M.layout.force(),P=t.width||parseInt(f.style("width"),10),N=t.height||parseInt(f.style("height"),10),A=t.layoutAttr,$=t.nodelist||null,q=e.isDirected(),D=t.weighted,L={nodeSelection:S,edgeSelection:x};if(t.withLabels){var F=t.labels;switch(typeof F){case"object":a=function(e){return k.getDefault(F[e.node],"")};break;case"function":a=F;break;case"string":a=function(e){return e.data[F]};break;default:a=function(e){return e.node}}}if(t.labels=a,D){var G=t.weights;switch(typeof weigths){case"object":c=function(e){return k.getDefault(G[e.node],1)};break;case"function":c=G;break;case"string":c=function(e){return k.getDefault(e.data[G],1)};break;default:c=function(e){return 1}}}if(t.withEdgeLabels){var C=t.edgeLabels;if(D&&null==C)i=c;else switch(typeof C){case"object":i=function(e){return k.getDefault(F[e.node],"")};break;case"function":i=C;break;case"string":i=function(e){return e.data[C]};break;default:i=function(e){return e.edge}}t.edgeLabels=i}if(D&&t.weightedStroke){var z=1,T=!0,J=!1,R=void 0;try{for(var X,B=m(e.edgesIter(null,!0));!(T=(X=B.next()).done);T=!0){var U=X.value,V=(U.u,U.v,U.data),W=c({data:V});W>z&&(z=W)}}catch(H){J=!0,R=H}finally{try{!T&&B["return"]&&B["return"]()}finally{if(J)throw R}}var K=M.scale.linear().range([2,t.edgeStyle["stroke-width"]]).domain([0,z]);t.edgeStyle["stroke-width"]=function(e){return K(c.call(this,e))}}y.select("svg.jsnx").remove(),y.attr("width",P+"px").attr("height",N+"px").style("opacity",1e-6).transition().duration(1e3).style("opacity",1);var Y={size:!0,nodes:!0,links:!0,start:!0};for(var Q in A)Y[Q]!==!0&&O[Q](A[Q]);O.nodes(d).links(h).size([P,N]);var Z=1,ee=1;t.panZoom.enabled&&!function(){var e=t.panZoom.scale,r=!1,n=1,a=Z;y.call(M.behavior.zoom().on("zoom",function(){if(M.event.sourceEvent){var t=M.event.sourceEvent.shiftKey,i=e&&t||!(e||t);i&&!r?(n=M.event.scale,a=Z,r=!0):!i&&r&&(r=!1),Z=i?a*(M.event.scale/n):Z,ee=i?ee:Z/M.event.scale;var o=M.event.translate;w.attr("transform","translate("+o[0]+","+o[1]+")scale("+M.event.scale+")"),ue()}}))}();var te=j,re=t.edgeOffset,ne=t.nodeAttr.r,ae=t.nodeStyle["stroke-width"];"circle"===t.nodeShape?("function"!=typeof ne&&(ne=function(){return t.nodeAttr.r}),"function"!=typeof ae&&(ae=function(){return t.nodeStyle["stroke-width"]}),re=function(e){return[ne(e.source)+ae(e.source),ne(e.target)+ae(e.target)]}):Array.isArray(re)?re=function(){return t.edgeOffset}:"number"==typeof re&&(re=function(){return[t.edgeOffset,t.edgeOffset]});var ie=t.edgeStyle["stroke-width"];"function"!=typeof ie&&(ie=function(){return t.edgeStyle["stroke-width"]});var oe=t.edgeLabelOffset;te=q?function(){L.edgeSelection.each(function(e){if(e.source!==e.target){var t=M.select(this),r=e.source.x,a=e.source.y,i=e.target.x,o=e.target.y,u=n(r,a,i,o),s=Math.sqrt(Math.pow(i-r,2)+Math.pow(o-a,2)),l=re(e);l=[l[0]*ee,l[1]*ee],t.attr("transform",["translate(",r,",",a,")","rotate(",u,")"].join(""));var c=ie(e)*ee,f=s-l[1]-2*c,d=c/2;t.select(".line").attr("d",["M",l[0],0,"L",l[0],-d,"L",f,-d,"L",f,-c,"L",s-l[1],0,"z"].join(" "));var h=1/ee;t.select("text").attr("x",oe.x*h+l[0]+(s*h-l[0]-l[1])/2).attr("y",-ie(e)/2+-oe.y*h).attr("transform","scale("+ee+")")}})}:function(){L.edgeSelection.each(function(e){if(e.source!==e.target){var r=M.select(this),a=e.source.x,i=e.source.y,o=e.target.x,u=e.target.y,s=n(a,i,o,u),l=Math.sqrt(Math.pow(o-a,2)+Math.pow(u-i,2)),c=l/2,f=re(e);f=[f[0]*ee,f[1]*ee];var d=1/ee,h=ie(e)*ee,p=s>90&&279>s;r.attr("transform",["translate(",a,",",i,")","rotate(",s,")"].join("")),r.select(".line").attr("d",["M",f[0],h/4,"L",f[0],-h/4,"L",l-f[1],-h/4,"L",l-f[1],h/4,"z"].join(" ")),t.withEdgeLabels&&r.select("text").attr("x",(p?1:-1)*oe.x*d+f[0]+(l*d-f[0]-f[1])/2).attr("y",-ie(e)/4+-oe.y*d).attr("transform","scale("+ee+")"+(p?"rotate(180,"+c*(1/ee)+",0)":""))}})};var ue=function(){L.nodeSelection.attr("transform",function(e){return["translate(",e.x,",",e.y,")","scale(",ee,")"].join("")}),te()};O.on("tick",ue);var se=e.nodesIter(),le=e.edgesIter();return $&&(r=!1,se=e.nbunch_iter($),le=e.edges_iter($)),L.nodeSelection=o(e,se,O,S,t),L.edgeSelection=u(e,le,O,x,i),s(L.nodeSelection,t),l(L.edgeSelection,t,null,q),r?p(e,O,t,L):v(e)?b(e):g(e),O.start(),O}function o(e,t,r,n,a){var i=r.nodes(),o=!0,u=!1,s=void 0;try{for(var l,f=m(t);!(o=(l=f.next()).done);o=!0){var d=l.value,h=e.node.get(d),p={node:d,data:h,G:e};i.push(p),h[O]=p}}catch(v){u=!0,s=v}finally{try{!o&&f["return"]&&f["return"]()}finally{if(u)throw s}}n=n.data(i,c);var b=r.drag().on("dragstart",function(e){M.event.sourceEvent.stopPropagation(),a.stickyDrag&&(e.fixed=!0,M.select(this).classed("fixed",!0))}),g=n.enter().append("g").classed("node",!0).call(b);return g.append(a.nodeShape).classed("node-shape",!0),a.labels&&g.append("text").text(a.labels),n}function u(e,t,r,n,a){var i=r.links(),o=!0,u=!1,s=void 0;try{for(var l,c=m(t);!(o=(l=c.next()).done);o=!0){var d=y(l.value,3),h=d[0],p=d[1],v=d[2];v=v||e.getEdgeData(h,p);var b={edge:[h,p],source:e.node.get(h)[O],target:e.node.get(p)[O],data:v,G:e};i.push(b),v[O]=b}}catch(g){u=!0,s=g}finally{try{!o&&c["return"]&&c["return"]()}finally{if(u)throw s}}n=n.data(i,f);var w=n.enter().append("g").classed("edge",!0);return w.append("path").classed("line",!0),a&&w.append("text").text(a),n}function s(e,t,r){if(null!=r){var n=new k.Set,a=!0,i=!1,o=void 0;try{for(var u,s=m(r);!(a=(u=s.next()).done);a=!0){var l=u.value;n.add(k.isArrayLike(l)?l[0]:l)}}catch(c){i=!0,o=c}finally{try{!a&&s["return"]&&s["return"]()}finally{if(i)throw o}}e=e.filter(function(e){return n.has(e.node)})}e.selectAll(".node-shape").attr(t.nodeAttr).style(t.nodeStyle),t.withLabels&&e.selectAll("text").attr(t.labelAttr).style(t.labelStyle)}function l(e,t,r,n){if(null!=r){var a=new k.Map,i=!0,o=!1,u=void 0;try{for(var s,l=m(r);!(i=(s=l.next()).done);i=!0){var c=y(s.value,2),f=c[0],d=c[1];a.set(f,d)}}catch(h){o=!0,u=h}finally{try{!i&&l["return"]&&l["return"]()}finally{if(o)throw u}}e=e.filter(function(e){var t=e.edge;return a.get(t[0])===t[1]||n||a.get(t[1])===t[0]})}e.selectAll(".line").attr(t.edgeAttr).style(t.edgeStyle).style("stroke-width",0),t.withEdgeLabels&&e.selectAll("text").attr(t.edgeLabelAttr).style(t.edgeLabelStyle)}function c(e){return e.node}function f(e){return e.edge}function d(e,t,r,n){var a=r.nodes(),i=!0,o=!1,u=void 0;try{for(var s,l=m(e.nbunchIter(t));!(i=(s=l.next()).done);i=!0){var f=s.value,d=a.indexOf(e.node.get(f)[O]);d>-1&&a.splice(d,1)}}catch(h){o=!0,u=h}finally{try{!i&&l["return"]&&l["return"]()}finally{if(o)throw u}}return n=n.data(a,c),n.exit().remove(),n}function h(e,t,r,n){var a=r.links(),i=!0,o=!1,u=void 0;try{for(var s,l=m(t);!(i=(s=l.next()).done);i=!0){var c=y(s.value,2),d=c[0],h=c[1],p=a.indexOf(e.getEdgeData(d,h,{})[O]);p>-1&&a.splice(p,1)}}catch(v){o=!0,u=v}finally{try{!i&&l["return"]&&l["return"]()}finally{if(o)throw u}}return n=n.data(a,f),n.exit().remove(),n}function p(e,t,r,n){b(e,!1);var i=e.constructor.prototype,p=r.edgeLabels,v=e.isDirected();e.addNode=function(e,a){var u=!this.hasNode(e);i.addNode.call(this,e,a),u&&(n.nodeSelection=o(this,[e],t,n.nodeSelection,r)),s(n.nodeSelection,r,[e]),t.start()},e.addNodesFrom=function(e,u){var l=this;e=a(e);var c=e.filter(function(e){return!l.hasNode(k.isArrayLike(e)?e[0]:e)});i.addNodesFrom.call(this,e,u),c.length>0&&(n.nodeSelection=o(this,c,t,n.nodeSelection,r)),s(n.nodeSelection,r,e),t.start()},e.addEdge=function(a,c,f){var d=this,h=!this.hasEdge(a,c),b=[[a,c]],g=h?(a===c?[a]:b[0]).filter(function(e){return!d.hasNode(e)}):[];i.addEdge.call(e,a,c,f),g.length>0&&(n.nodeSelection=o(this,g,t,n.nodeSelection,r),s(n.nodeSelection,r,g)),h&&(n.edgeSelection=u(this,b,t,n.edgeSelection,p)),l(n.edgeSelection,r,b,v),t.start()},e.addEdgesFrom=function(c,f){var d=[],h=[],b=new k.Map,g=new k.Set;c=a(c);var w=!0,x=!1,j=void 0;try{for(var E,_=m(c);!(w=(E=_.next()).done);w=!0){var S=y(E.value,2),O=S[0],M=S[1];this.hasEdge(O,M)||b.get(O)===M||!v&&b.get(M)!==O||(d.push([O,M]),b.set(O,M),this.hasNode(O)||g.has(O)||(h.push(O),g.add(O)),this.hasNode(M)||g.has(M)||(h.push(M),g.add(M)))}}catch(I){x=!0,j=I}finally{try{!w&&_["return"]&&_["return"]()}finally{if(x)throw j}}i.addEdgesFrom.call(e,c,f),h.length>0&&(n.nodeSelection=o(this,h,t,n.nodeSelection,r),s(n.nodeSelection,r,h)),d.length>0&&(n.edgeSelection=u(this,d,t,n.edgeSelection,p)),l(n.edgeSelection,r,d,v),t.start()},e.removeNode=function(e){if(this.hasNode(e)){n.nodeSelection=d(this,[e],t,n.nodeSelection);var r=this.edgesIter([e]);this.isDirected()&&(r=x.mark(function a(t,r){return x.wrap(function(n){for(;;)switch(n.prev=n.next){case 0:return n.delegateYield(r,"t0",1);case 1:return n.delegateYield(t.inEdgesIter([e]),"t1",2);case 2:case"end":return n.stop()}},a,this)})(this,r)),n.edgeSelection=h(this,r,t,n.edgeSelection),t.resume()}i.removeNode.call(this,e)},e.removeNodesFrom=function(e){e=a(e),n.nodeSelection=d(this,e,t,n.nodeSelection);var r=this.edgesIter(e);this.isDirected()&&(r=x.mark(function o(t,r){return x.wrap(function(n){for(;;)switch(n.prev=n.next){case 0:return n.delegateYield(r,"t0",1);case 1:return n.delegateYield(t.inEdgesIter(e),"t1",2);case 2:case"end":return n.stop()}},o,this)})(this,r)),n.edgeSelection=h(this,r,t,n.edgeSelection),t.resume(),i.removeNodesFrom.call(this,e)},e.removeEdge=function(e,r){n.edgeSelection=h(this,[[e,r]],t,n.edgeSelection),t.resume(),i.removeEdge.call(this,e,r)},e.removeEdgesFrom=function(r){r=a(r),n.edgeSelection=h(this,r,t,n.edgeSelection),t.resume(),i.removeEdgesFrom.call(e,r)},e.clear=function(){n.nodeSelection=n.nodeSelection.data([],c),n.nodeSelection.exit().remove(),n.edgeSelection=n.edgeSelection.data([],f),n.edgeSelection.exit().remove(),t.nodes([]).links([]).resume(),i.clear.call(this)},e.bound=!0}function v(e){return e.bound}function b(e){var t=arguments.length<=1||void 0===arguments[1]?!0:arguments[1];if(v(e)){var r=e.constructor.prototype;S.forEach(function(t){return e[t]=r[t]}),delete e.bound,t&&g(e)}}function g(e){var t=!0,r=!1,n=void 0;try{for(var a,i=m(e.nodesIter(!0));!(t=(a=i.next()).done);t=!0){var o=y(a.value,2),u=(o[0],o[1]);delete u[O]}}catch(s){r=!0,n=s}finally{try{!t&&i["return"]&&i["return"]()}finally{if(r)throw n}}var l=!0,c=!1,f=void 0; + try{for(var d,h=m(e.edgesIter(null,!0));!(l=(d=h.next()).done);l=!0){var p=y(d.value,3),u=(p[0],p[1],p[2]);delete u[O]}}catch(s){c=!0,f=s}finally{try{!l&&h["return"]&&h["return"]()}finally{if(c)throw f}}}var y=e("babel-runtime/helpers/sliced-to-array")["default"],m=e("babel-runtime/core-js/get-iterator")["default"],w=e("babel-runtime/core-js/array/from")["default"],x=e("babel-runtime/regenerator")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.draw=i;var k=e("../_internals"),j=function(){},E=null,_=null,S=["addNode","addNodesFrom","addEdge","addEdgesFrom","removeNode","removeNodesFrom","removeEdge","removeEdgesFrom","clear"],O="__d3datum__",M=t.d3,I={layoutAttr:{charge:-120,linkDistance:60},nodeShape:"circle",nodeAttr:{r:10},nodeStyle:{"stroke-width":2,stroke:"#333",fill:"#999",cursor:"pointer"},edgeAttr:{},edgeStyle:{fill:"#000","stroke-width":3},labelAttr:{},labelStyle:{"text-anchor":"middle","dominant-baseline":"central",cursor:"pointer","-webkit-user-select":"none",fill:"#000"},edgeLabelAttr:{},edgeLabelStyle:{"font-size":"0.8em","text-anchor":"middle","-webkit-user-select":"none"},edgeLabelOffset:{x:0,y:.5},withLabels:!1,withEdgeLabels:!1,edgeOffset:10,weighted:!1,weights:"weight",weightedStroke:!0,panZoom:{enabled:!0,scale:!0}}}).call(this,"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{"../_internals":20,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/get-iterator":89,"babel-runtime/helpers/sliced-to-array":109,"babel-runtime/regenerator":166}],72:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/get")["default"],a=e("babel-runtime/helpers/inherits")["default"],i=e("babel-runtime/helpers/class-call-check")["default"],o=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var u=e("./JSNetworkXException"),s=o(u),l=function(e){function t(e){i(this,t),n(Object.getPrototypeOf(t.prototype),"constructor",this).call(this,e),this.name="JSNetworkXAlgorithmError"}return a(t,e),t}(s["default"]);r["default"]=l,t.exports=r["default"]},{"./JSNetworkXException":74,"babel-runtime/helpers/class-call-check":101,"babel-runtime/helpers/get":105,"babel-runtime/helpers/inherits":106,"babel-runtime/helpers/interop-require-default":107}],73:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/get")["default"],a=e("babel-runtime/helpers/inherits")["default"],i=e("babel-runtime/helpers/class-call-check")["default"],o=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var u=e("./JSNetworkXException"),s=o(u),l=function(e){function t(e){i(this,t),n(Object.getPrototypeOf(t.prototype),"constructor",this).call(this,e),this.name="JSNetworkXError"}return a(t,e),t}(s["default"]);r["default"]=l,t.exports=r["default"]},{"./JSNetworkXException":74,"babel-runtime/helpers/class-call-check":101,"babel-runtime/helpers/get":105,"babel-runtime/helpers/inherits":106,"babel-runtime/helpers/interop-require-default":107}],74:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/class-call-check")["default"];Object.defineProperty(r,"__esModule",{value:!0});var a=function i(e){n(this,i),this.name="JSNetworkXException",this.message=e};r["default"]=a,t.exports=r["default"]},{"babel-runtime/helpers/class-call-check":101}],75:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/get")["default"],a=e("babel-runtime/helpers/inherits")["default"],i=e("babel-runtime/helpers/class-call-check")["default"],o=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var u=e("./JSNetworkXUnfeasible"),s=o(u),l=function(e){function t(e){i(this,t),n(Object.getPrototypeOf(t.prototype),"constructor",this).call(this,e),this.name="JSNetworkXNoPath"}return a(t,e),t}(s["default"]);r["default"]=l,t.exports=r["default"]},{"./JSNetworkXUnfeasible":76,"babel-runtime/helpers/class-call-check":101,"babel-runtime/helpers/get":105,"babel-runtime/helpers/inherits":106,"babel-runtime/helpers/interop-require-default":107}],76:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/get")["default"],a=e("babel-runtime/helpers/inherits")["default"],i=e("babel-runtime/helpers/class-call-check")["default"],o=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var u=e("./JSNetworkXAlgorithmError"),s=o(u),l=function(e){function t(e){i(this,t),n(Object.getPrototypeOf(t.prototype),"constructor",this).call(this,e),this.name="JSNetworkXUnfeasible"}return a(t,e),t}(s["default"]);r["default"]=l,t.exports=r["default"]},{"./JSNetworkXAlgorithmError":72,"babel-runtime/helpers/class-call-check":101,"babel-runtime/helpers/get":105,"babel-runtime/helpers/inherits":106,"babel-runtime/helpers/interop-require-default":107}],77:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/get")["default"],a=e("babel-runtime/helpers/inherits")["default"],i=e("babel-runtime/helpers/class-call-check")["default"];Object.defineProperty(r,"__esModule",{value:!0});var o=function(e){function t(e){i(this,t),n(Object.getPrototypeOf(t.prototype),"constructor",this).call(this),this.name="KeyError",this.message=e}return a(t,e),t}(Error);r["default"]=o,t.exports=r["default"]},{"babel-runtime/helpers/class-call-check":101,"babel-runtime/helpers/get":105,"babel-runtime/helpers/inherits":106}],78:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0});var a=e("./KeyError"),i=n(a),o=e("./JSNetworkXAlgorithmError"),u=n(o),s=e("./JSNetworkXError"),l=n(s),c=e("./JSNetworkXException"),f=n(c),d=e("./JSNetworkXNoPath"),h=n(d),p=e("./JSNetworkXUnfeasible"),v=n(p);r.KeyError=i["default"],r.JSNetworkXAlgorithmError=u["default"],r.JSNetworkXError=l["default"],r.JSNetworkXException=f["default"],r.JSNetworkXNoPath=h["default"],r.JSNetworkXUnfeasible=v["default"]},{"./JSNetworkXAlgorithmError":72,"./JSNetworkXError":73,"./JSNetworkXException":74,"./JSNetworkXNoPath":75,"./JSNetworkXUnfeasible":76,"./KeyError":77,"babel-runtime/helpers/interop-require-default":107}],79:[function(e,t,r){"use strict";function n(e,t){var r,n,a,i,o;return h.wrap(function(u){for(;;)switch(u.prev=u.next){case 0:if(r=y.genRange(e),0!==e){u.next=3;break}return u.abrupt("return");case 3:n=[y.next(r)];case 4:if(!(n.length>0)){u.next=20;break}a=n.shift(),i=0;case 7:if(!(t>i)){u.next=18;break}if(o=r.next(),!o.done){u.next=11;break}return u.abrupt("return");case 11:return o=o.value,n.push(o),u.next=15,y.tuple2(a,o);case 15:i++,u.next=7;break;case 18:u.next=4;break;case 20:case"end":return u.stop()}},v[0],this)}function a(e,t,r){var a=s(t,r);return a.addEdgesFrom(n(t,e)),a}function i(e,t,r){var a=1===e?t:Math.floor((1-Math.pow(e,t+1))/(1-e)),i=s(a,r);return i.addEdgesFrom(n(a,e)),i}function o(e,t){var r=s(e,t);return r.name="complete_graph("+e+")",e>1&&r.addEdgesFrom(r.isDirected()?y.genPermutations(y.range(e),2):y.genCombinations(y.range(e),2)),r}function u(e,t){var r=f(e,t);return r.name="cycle_graph("+e+")",e>1&&r.addEdge(e-1,0),r}function s(e,t){y.isGraph(e)&&(t=e,e=null),null==e&&(e=0);var r;return null==t?r=new g["default"]:(r=t,r.clear()),r.addNodesFrom(y.genRange(e)),r.name="emptyGraph("+e+")",r}function l(e,t){var r=arguments.length<=2||void 0===arguments[2]?!1:arguments[2],n=arguments.length<=3||void 0===arguments[3]?null:arguments[3],a=s(0,n);a.name="grid2dGraph";var i,o;for(i=0;e>i;i++)for(o=0;t>o;o++)a.addNode([i,o]);for(i=1;e>i;i++)for(o=0;t>o;o++)a.addEdge([i,o],[i-1,o]);for(i=0;e>i;i++)for(o=1;t>o;o++)a.addEdge([i,o],[i,o-1]);if(a.isDirected()){for(i=0;e-1>i;i++)for(o=0;t>o;o++)a.addEdge([i,o],[i+1,o]);for(i=0;e>i;i++)for(o=0;t-1>o;o++)a.addEdge([i,o],[i,o+1])}if(r){if(t>2){for(i=0;e>i;i++)a.addEdge([i,0],[i,t-1]);if(a.isDirected())for(i=0;e>i;i++)a.addEdge([i,t-1],[i,0])}if(e>2){for(o=0;t>o;o++)a.addEdge([0,o],[e-1,o]);if(a.isDirected())for(o=0;t>o;o++)a.addEdge([e-1,o],[0,o])}a.name="periodicGrid2dGraph("+e+", "+t+")"}return a}function c(e){var t=s(0,e);return t.name="nullGraph()",t}function f(e,t){var r=s(e,t);return r.name="pathGraph("+e+")",r.addEdgesFrom(y.mapIterator(y.genRange(e-1),function(e){return y.tuple2(e,e+1)})),r}function d(e){var t=s(1,e);return t.name="nullGraph()",t}var h=e("babel-runtime/regenerator")["default"],p=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.fullRaryTree=a,r.balancedTree=i,r.completeGraph=o,r.cycleGraph=u,r.emptyGraph=s,r.grid2dGraph=l,r.nullGraph=c,r.pathGraph=f,r.trivialGraph=d;var v=[n].map(h.mark),b=e("../classes/Graph"),g=p(b),y=e("../_internals")},{"../_internals":20,"../classes/Graph":61,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/regenerator":166}],80:[function(e,t,r){"use strict";function n(e,t){if(e=o(e),!d.isValidDegreeSequence(e))throw new f["default"]("Invalid degree sequence");if(null!=t&&t.isDirected())throw new f["default"]("Directed Graph not supported");for(var r=e.length,n=h.emptyGraph(r,t),a=new Array(r),u=0;r>u;u++)a[u]=[];var s=0,l=0,c=0;for(u=0;r>u;u++){var p=e[u];p>0&&(a[p].push(c),s=Math.max(s,p),l+=p,c+=1)}if(0===c)return n;var b=new Array(s+1);for(u=0;s+1>u;u++)b[u]=[0,0];for(;c>0;){for(;0===a[s].length;)s-=1;if(s>c-1)throw new f["default"]("Non-graphical integer sequence");var g=a[s].pop();c-=1;var y=0,m=s;for(u=0;s>u;u++){for(;0===a[m].length;)m-=1;var w=a[m].pop();n.addEdge(g,w),c-=1,m>1&&(b[y]=[m-1,w],y+=1)}for(u=0;y>u;u++){var x=i(b[u],2),k=x[0],j=x[1];a[k].push(j),c+=1}}return n.name=v["default"]("havelHakimiGraph %s nodes %d edges",n.order(),n.size()),n}function a(e,t){return l["default"]("havelHakimiGraph",[e,t])}var i=e("babel-runtime/helpers/sliced-to-array")["default"],o=e("babel-runtime/core-js/array/from")["default"],u=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.havelHakimiGraph=n,r.genHavelHakimiGraph=a;var s=e("../_internals/delegate"),l=u(s),c=e("../exceptions/JSNetworkXError"),f=u(c),d=e("../algorithms/graphical"),h=e("./classic"),p=e("../_internals/sprintf"),v=u(p)},{"../_internals/delegate":12,"../_internals/sprintf":38,"../algorithms/graphical":49,"../exceptions/JSNetworkXError":73,"./classic":79,"babel-runtime/core-js/array/from":88,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109}],81:[function(e,t,r){"use strict";var n=e("babel-runtime/helpers/interop-require-wildcard")["default"],a=e("babel-runtime/helpers/defaults")["default"];Object.defineProperty(r,"__esModule",{value:!0});var i=e("./classic"),o=n(i),u=e("./degreeSequence"),s=n(u),l=e("./randomGraphs"),c=n(l),f=e("./small"),d=n(f),h=e("./social"),p=n(h);r.classic=o,r.degreeSequence=s,r.randomGraphs=c,r.small=d,r.social=p,a(r,n(i)),a(r,n(u)),a(r,n(l)),a(r,n(f)),a(r,n(h))},{"./classic":79,"./degreeSequence":80,"./randomGraphs":82,"./small":83,"./social":84,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-wildcard":108}],82:[function(e,t,r){"use strict";function n(e,t){var r=arguments.length<=2||void 0===arguments[2]?!1:arguments[2],n=m.emptyGraph(e);if(n.name=w.sprintf("fastGnpRandomGraph(%s, %s)",e,t),0>=t||t>=1)return i(e,t,r);var a,o,u=-1,s=Math.log(1-t);if(r)for(a=0,n=new b["default"](n);e>a;){for(o=Math.log(1-Math.random()),u=u+1+Math.floor(o/s),a===u&&(u+=1);u>=e&&e>a;)u-=e,a+=1,a===u&&(u+=1);e>a&&n.addEdge(a,u)}else for(a=1;e>a;){for(o=Math.log(1-Math.random()),u=u+1+Math.floor(o/s);u>=a&&e>a;)u-=a,a+=1;e>a&&n.addEdge(a,u)}return n}function a(e,t,r){return p["default"]("fastGnpRandomGraph",[e,t,r])}function i(e,t){var r,n=arguments.length<=2||void 0===arguments[2]?!1:arguments[2],a=n?new b["default"]:new y["default"],i=w.range(e);if(a.addNodesFrom(i),a.name=w.sprintf("gnpRandomGraph(%s, %s)",e,t),0>=t)return a;if(t>=1)return m.completeGraph(e,a);r=a.isDirected()?w.genPermutations(i,2):w.genCombinations(i,2);var o=!0,u=!1,s=void 0;try{for(var l,c=f(r);!(o=(l=c.next()).done);o=!0){var d=l.value;Math.random()r||r>a-1||0>n||n>a-1)throw new c["default"]("invalid graphDescription");o.addEdge(r,n)});return o.name=n,o}function i(e){var t="adjacencylist",r="Bull Graph",a=5,i=[[2,3],[1,3,4],[1,2,5],[2],[3]];return n({type:t,name:r,n:a,list:i},e)}function o(e){var t="adjacencylist",r="Krackhardt Kite Social Network",a=10,i=[[2,3,4,6],[1,4,5,7],[1,4,6],[1,2,3,5,6,7],[2,4,7],[1,3,4,7,8],[2,4,5,6,8],[6,7,9],[8,10],[9]];return n({type:t,name:r,n:a,list:i},e)}var u=e("babel-runtime/helpers/sliced-to-array")["default"],s=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.makeSmallUndirectedGraph=n,r.makeSmallGraph=a,r.bullGraph=i,r.krackhardtKiteGraph=o;var l=e("../exceptions/JSNetworkXError"),c=s(l),f=e("./classic"),d=e("../_internals")},{"../_internals":20,"../exceptions/JSNetworkXError":73,"./classic":79,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/sliced-to-array":109}],84:[function(e,t,r){"use strict";function n(){var e=new s["default"];e.addNodesFrom(c["default"](34)),e.name="Zachary's Karate Club";var t=["0 1 1 1 1 1 1 1 1 0 1 1 1 1 0 0 0 1 0 1 0 1 0 0 0 0 0 0 0 0 0 1 0 0","1 0 1 1 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 1 0 1 0 0 0 0 0 0 0 0 1 0 0 0","1 1 0 1 0 0 0 1 1 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 0 1 0","1 1 1 0 0 0 0 1 0 0 0 0 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 0 0 0 1 1 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 1 1","0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1","1 0 0 0 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1","0 0 0 0 0 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1","1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1","1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 1 0 1 0 0 1 1","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 1 0 0 0 1 0 0","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 0 0 0 0 1 0 0","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 1","0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 0 0 0 0 0 0 1","0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 1","0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 1 0 0 0 0 0 1 1","0 1 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1","1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 1 0 0 0 1 1","0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 1 0 0 1 0 1 0 1 1 0 0 0 0 0 1 1 1 0 1","0 0 0 0 0 0 0 0 1 1 0 0 0 1 1 1 0 0 1 1 1 0 1 1 0 0 1 1 1 1 1 1 1 0"];return t.forEach(function(t,r){var n=t.split(" ");n.forEach(function(t,n){"1"===t&&e.addEdge(r,n)})}),e.addNodesFrom([0,1,2,3,4,5,6,7,8,10,11,12,13,16,17,19,21],{club:"Mr. Hi"}),e.addNodesFrom([9,14,15,18,20,22,23,24,25,26,27,28,29,30,31,32,33],{club:"Officer"}),e}function a(){var e=new s["default"];return e.addNodesFrom(["Evelyn Jefferson","Laura Mandeville","Theresa Anderson","Brenda Rogers","Charlotte McDowd","Frances Anderson","Eleanor Nye","Pearl Oglethorpe","Ruth DeSand","Verne Sanderson","Myra Liddel","Katherina Rogers","Sylvia Avondale","Nora Fayette","Helen Lloyd","Dorothy Murchison","Olivia Carleton","Flora Price"],{bipartite:0}),e.addNodesFrom(["E1","E2","E3","E4","E5","E6","E7","E8","E9","E10","E11","E12","E13","E14"],{bipartite:1}),e.add_edges_from([["Evelyn Jefferson","E1"],["Evelyn Jefferson","E2"],["Evelyn Jefferson","E3"],["Evelyn Jefferson","E4"],["Evelyn Jefferson","E5"],["Evelyn Jefferson","E6"],["Evelyn Jefferson","E8"],["Evelyn Jefferson","E9"],["Laura Mandeville","E1"],["Laura Mandeville","E2"],["Laura Mandeville","E3"],["Laura Mandeville","E5"],["Laura Mandeville","E6"],["Laura Mandeville","E7"],["Laura Mandeville","E8"],["Theresa Anderson","E2"],["Theresa Anderson","E3"],["Theresa Anderson","E4"],["Theresa Anderson","E5"],["Theresa Anderson","E6"],["Theresa Anderson","E7"],["Theresa Anderson","E8"],["Theresa Anderson","E9"],["Brenda Rogers","E1"],["Brenda Rogers","E3"],["Brenda Rogers","E4"],["Brenda Rogers","E5"],["Brenda Rogers","E6"],["Brenda Rogers","E7"],["Brenda Rogers","E8"],["Charlotte McDowd","E3"],["Charlotte McDowd","E4"],["Charlotte McDowd","E5"],["Charlotte McDowd","E7"],["Frances Anderson","E3"],["Frances Anderson","E5"],["Frances Anderson","E6"],["Frances Anderson","E8"],["Eleanor Nye","E5"],["Eleanor Nye","E6"],["Eleanor Nye","E7"],["Eleanor Nye","E8"],["Pearl Oglethorpe","E6"],["Pearl Oglethorpe","E8"],["Pearl Oglethorpe","E9"],["Ruth DeSand","E5"],["Ruth DeSand","E7"],["Ruth DeSand","E8"],["Ruth DeSand","E9"],["Verne Sanderson","E7"],["Verne Sanderson","E8"],["Verne Sanderson","E9"],["Verne Sanderson","E12"],["Myra Liddel","E8"],["Myra Liddel","E9"],["Myra Liddel","E10"],["Myra Liddel","E12"],["Katherina Rogers","E8"],["Katherina Rogers","E9"],["Katherina Rogers","E10"],["Katherina Rogers","E12"],["Katherina Rogers","E13"],["Katherina Rogers","E14"],["Sylvia Avondale","E7"],["Sylvia Avondale","E8"],["Sylvia Avondale","E9"],["Sylvia Avondale","E10"],["Sylvia Avondale","E12"],["Sylvia Avondale","E13"],["Sylvia Avondale","E14"],["Nora Fayette","E6"],["Nora Fayette","E7"],["Nora Fayette","E9"],["Nora Fayette","E10"],["Nora Fayette","E11"],["Nora Fayette","E12"],["Nora Fayette","E13"],["Nora Fayette","E14"],["Helen Lloyd","E7"],["Helen Lloyd","E8"],["Helen Lloyd","E10"],["Helen Lloyd","E11"],["Helen Lloyd","E12"],["Dorothy Murchison","E8"],["Dorothy Murchison","E9"],["Olivia Carleton","E9"],["Olivia Carleton","E11"],["Flora Price","E9"],["Flora Price","E11"]]),e}function i(){var e=new s["default"];return e.addEdge("Acciaiuoli","Medici"),e.addEdge("Castellani","Peruzzi"),e.addEdge("Castellani","Strozzi"),e.addEdge("Castellani","Barbadori"),e.addEdge("Medici","Barbadori"),e.addEdge("Medici","Ridolfi"),e.addEdge("Medici","Tornabuoni"),e.addEdge("Medici","Albizzi"),e.addEdge("Medici","Salviati"),e.addEdge("Salviati","Pazzi"),e.addEdge("Peruzzi","Strozzi"),e.addEdge("Peruzzi","Bischeri"),e.addEdge("Strozzi","Ridolfi"),e.addEdge("Strozzi","Bischeri"),e.addEdge("Ridolfi","Tornabuoni"),e.addEdge("Tornabuoni","Guadagni"),e.addEdge("Albizzi","Ginori"),e.addEdge("Albizzi","Guadagni"),e.addEdge("Bischeri","Guadagni"),e.addEdge("Guadagni","Lamberteschi"),e}var o=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.karateClubGraph=n,r.davisSouthernWomenGraph=a,r.florentineFamiliesGraph=i;var u=e("../classes/Graph"),s=o(u),l=e("../_internals/range"),c=o(l)},{"../_internals/range":35,"../classes/Graph":61,"babel-runtime/helpers/interop-require-default":107}],85:[function(e,t,r){"use strict";var n=e("babel-runtime/core-js/array/from")["default"],a=e("babel-runtime/helpers/interop-require-wildcard")["default"],i=e("babel-runtime/helpers/interop-require-default")["default"],o=e("babel-runtime/helpers/defaults")["default"];Object.defineProperty(r,"__esModule",{value:!0});var u=e("./algorithms"),s=a(u),l=e("./classes"),c=a(l),f=e("./convert"),d=a(f),h=e("./drawing"),p=a(h),v=e("./exceptions"),b=a(v),g=e("./generators"),y=a(g),m=e("./relabel"),w=a(m),x=e("./_internals/Map"),k=i(x),j=e("./_internals/Set"),E=i(j),_=e("./_internals/forEach"),S=i(_);r.Map=k["default"],r.Set=E["default"],r.forEach=S["default"],r.algorithms=s,r.classes=c,r.convert=d,r.drawing=p,r.exceptions=b,r.generators=y,r.relabel=w;var O=n;r.toArray=O,o(r,a(u)),o(r,a(l)),o(r,a(f)),o(r,a(h));var M=e("./contrib/observer");o(r,a(M)),o(r,a(v)),o(r,a(g)),o(r,a(m))},{"./_internals/Map":3,"./_internals/Set":5,"./_internals/forEach":14,"./algorithms":50,"./classes":65,"./contrib/observer":67,"./convert":69,"./drawing":70,"./exceptions":78,"./generators":81,"./relabel":87,"babel-runtime/core-js/array/from":88,"babel-runtime/helpers/defaults":103,"babel-runtime/helpers/interop-require-default":107,"babel-runtime/helpers/interop-require-wildcard":108}],86:[function(e,t,r){(function(n){"use strict";function a(){n.document||(n.onmessage=function(e){var t=e.data.args.map(o.deserialize),r=s["default"].methodLookupFunction(e.data.method).apply(null,t);n.postMessage(o.serialize(r)),n.close()})}var i=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r["default"]=a;var o=e("./_internals/message"),u=e("./WorkerSettings"),s=i(u);t.exports=r["default"]}).call(this,"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{"./WorkerSettings":1,"./_internals/message":32,"babel-runtime/helpers/interop-require-default":107}],87:[function(e,t,r){"use strict";function n(e,t){var r=arguments.length<=2||void 0===arguments[2]?!0:arguments[2],n=t;return"function"!=typeof t?h.isMap(n)||(n=new h.Map(n)):n=new h.Map(h.mapIterator(e.nodesIter(),function(e){return h.tuple2(e,t(e))})),r?i(e,n):a(e,n)}function a(e,t){var r,n=new h.Set(t.keys());if(h.someIterator(t.values(),function(e){return n.has(e)})){var a=new f["default"](t);a.removeEdgesFrom(a.selfloopEdges());try{r=h.topologicalSort(a)}catch(i){if(i instanceof d.JSNetworkXUnfeasible)throw new d.JSNetworkXUnfeasible("The node label sets are overlapping and no ordering can resolve the mapping. Use copy=True.")}r.reverse()}else r=n.values();var o,u=e.isMultigraph(),s=e.isDirected();return h.forEach(r,function(r){var n;if(t.has(r)){if(n=t.get(r),!e.hasNode(r))throw new d.JSNetworkXError(h.sprintf("Node %j is not in the graph.",r));e.addNode(n,e.node.get(r)),u?(o=e.edges(r,!0,!0).map(function(e){return h.tuple4c(n,e[1],e[2],e[3],e)}),s&&(o=o.concat(e.inEdges(r,!0,!0).map(function(e){return h.tuple4c(e[0],n,e[2],e[3],e)})))):(o=e.edges(r,!0).map(function(e){return h.tuple3c(n,e[1],e[2],e)}),s&&(o=o.concat(e.inEdges(r,!0).map(function(e){return h.tuple3c(e[0],n,e[2],e)})))),e.removeNode(r),e.addEdgesFrom(o)}}),e}function i(e,t){var r=new e.constructor;return r.name="("+e.name+")",e.isMultigraph()?r.addEdgesFrom(h.mapIterator(e.edgesIter(null,!0,!0),function(e){return h.tuple4c(t.has(e[0])?t.get(e[0]):e[0],t.has(e[1])?t.get(e[1]):e[1],e[2],h.clone(e[3]),e)})):r.addEdgesFrom(h.mapIterator(e.edgesIter(null,!0),function(e){return h.tuple3c(t.has(e[0])?t.get(e[0]):e[0],t.has(e[1])?t.get(e[1]):e[1],h.clone(e[3]),e)})),e.node.forEach(function(e,n){return r.addNode(t.has(n)?t.get(n):n,h.clone(e))}),u(r.graph,h.clone(e.graph)),r}function o(e){var t=arguments.length<=1||void 0===arguments[1]?0:arguments[1],r=arguments.length<=2||void 0===arguments[2]?"default":arguments[2],a=arguments.length<=3||void 0===arguments[3]?!0:arguments[3];switch("boolean"==typeof r&&(a=r,r="default"),typeof t){case"string":r=t,t=0;break;case"boolean":a=t,t=0}var i,o,u,l,c,f=new h.Map;switch(r){case"default":for(i=e.nodes(),u=0,l=t,c=i.length;c>u;u++,l++)f.set(i[u],l);break;case"sorted":for(i=e.nodes(),i.sort(),u=0,l=t,c=i.length;c>u;u++,l++)f.set(i[u],l);break;case"increasing degree":for(o=s(e.degreeIter()),o.sort(function(e,t){return e[1]-t[1]}),u=0,l=t,c=o.length;c>u;u++,l++)f.set(o[u][0],l);break;case"decreasing degree":for(o=s(e.degreeIter()),o.sort(function(e,t){return t[1]-e[1]}),u=0,l=t,c=o.length;c>u;u++,l++)f.set(o[u][0],l);break;default:throw new d.JSNetworkXError(h.sprintf('Unkown node ordering: "%s"',r))}var p=n(e,f);return p.name="("+e.name+")WithIntLabels",a||(p.nodeLabels=f),p}var u=e("babel-runtime/core-js/object/assign")["default"],s=e("babel-runtime/core-js/array/from")["default"],l=e("babel-runtime/helpers/interop-require-default")["default"];Object.defineProperty(r,"__esModule",{value:!0}),r.relabelNodes=n,r.convertNodeLabelsToIntegers=o;var c=e("./classes/DiGraph"),f=l(c),d=e("./exceptions"),h=e("./_internals")},{"./_internals":20,"./classes/DiGraph":60,"./exceptions":78,"babel-runtime/core-js/array/from":88,"babel-runtime/core-js/object/assign":92,"babel-runtime/helpers/interop-require-default":107}],88:[function(e,t,r){t.exports={"default":e("core-js/library/fn/array/from"),__esModule:!0}},{"core-js/library/fn/array/from":110}],89:[function(e,t,r){t.exports={"default":e("core-js/library/fn/get-iterator"),__esModule:!0}},{"core-js/library/fn/get-iterator":111}],90:[function(e,t,r){t.exports={"default":e("core-js/library/fn/is-iterable"),__esModule:!0}},{"core-js/library/fn/is-iterable":112}],91:[function(e,t,r){t.exports={"default":e("core-js/library/fn/map"),__esModule:!0}},{"core-js/library/fn/map":113}],92:[function(e,t,r){t.exports={"default":e("core-js/library/fn/object/assign"),__esModule:!0}},{"core-js/library/fn/object/assign":114}],93:[function(e,t,r){t.exports={"default":e("core-js/library/fn/object/create"),__esModule:!0}},{"core-js/library/fn/object/create":115}],94:[function(e,t,r){t.exports={"default":e("core-js/library/fn/object/define-property"),__esModule:!0}},{"core-js/library/fn/object/define-property":116}],95:[function(e,t,r){t.exports={"default":e("core-js/library/fn/object/get-own-property-descriptor"),__esModule:!0}},{"core-js/library/fn/object/get-own-property-descriptor":117}],96:[function(e,t,r){t.exports={"default":e("core-js/library/fn/object/get-own-property-names"),__esModule:!0}},{"core-js/library/fn/object/get-own-property-names":118}],97:[function(e,t,r){t.exports={"default":e("core-js/library/fn/object/keys"),__esModule:!0}},{"core-js/library/fn/object/keys":119}],98:[function(e,t,r){t.exports={"default":e("core-js/library/fn/promise"),__esModule:!0}},{"core-js/library/fn/promise":120}],99:[function(e,t,r){t.exports={"default":e("core-js/library/fn/symbol"),__esModule:!0}},{"core-js/library/fn/symbol":121}],100:[function(e,t,r){t.exports={"default":e("core-js/library/fn/symbol/iterator"),__esModule:!0}},{"core-js/library/fn/symbol/iterator":122}],101:[function(e,t,r){"use strict";r["default"]=function(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")},r.__esModule=!0},{}],102:[function(e,t,r){"use strict";var n=e("babel-runtime/core-js/object/define-property")["default"];r["default"]=function(){function e(e,t){for(var r=0;ro;)for(var u,s=n.ES5Object(arguments[o++]),l=a(s),c=l.length,f=0;c>f;)r[u=l[f++]]=s[u];return r}},{"./$":141,"./$.enum-keys":132}],125:[function(e,t,r){function n(e){return o.call(e).slice(8,-1)}var a=e("./$"),i=e("./$.wks")("toStringTag"),o={}.toString;n.classof=function(e){var t,r;return void 0==e?void 0===e?"Undefined":"Null":"string"==typeof(r=(t=Object(e))[i])?r:n(t)},n.set=function(e,t,r){e&&!a.has(e=r?e:e.prototype,i)&&a.hide(e,i,t)},t.exports=n},{"./$":141,"./$.wks":153}],126:[function(e,t,r){"use strict";function n(e,t){if(!h(e))return"symbol"==typeof e?e:("string"==typeof e?"S":"P")+e;if(!f(e,b)){if(!v(e))return"F";if(!t)return"E";p(e,b,++k)}return"O"+e[b]}function a(e,t){var r,a=n(t);if("F"!==a)return e[g][a];for(r=e[m];r;r=r.n)if(r.k==t)return r}var i=e("./$"),o=e("./$.ctx"),u=e("./$.uid").safe,s=e("./$.assert"),l=e("./$.for-of"),c=e("./$.iter").step,f=i.has,d=i.set,h=i.isObject,p=i.hide,v=Object.isExtensible||h,b=u("id"),g=u("O1"),y=u("last"),m=u("first"),w=u("iter"),x=i.DESC?u("size"):"size",k=0;t.exports={getConstructor:function(t,r,n,u){var c=t(function(e,t){s.inst(e,c,r),d(e,g,i.create(null)),d(e,x,0),d(e,y,void 0),d(e,m,void 0),void 0!=t&&l(t,n,e[u],e)});return e("./$.mix")(c.prototype,{clear:function(){for(var e=this,t=e[g],r=e[m];r;r=r.n)r.r=!0,r.p&&(r.p=r.p.n=void 0),delete t[r.i];e[m]=e[y]=void 0,e[x]=0},"delete":function(e){var t=this,r=a(t,e);if(r){var n=r.n,i=r.p;delete t[g][r.i],r.r=!0,i&&(i.n=n),n&&(n.p=i),t[m]==r&&(t[m]=n),t[y]==r&&(t[y]=i),t[x]--}return!!r},forEach:function(e){for(var t,r=o(e,arguments[1],3);t=t?t.n:this[m];)for(r(t.v,t.k,this);t&&t.r;)t=t.p},has:function(e){return!!a(this,e)}}),i.DESC&&i.setDesc(c.prototype,"size",{get:function(){return s.def(this[x])}}),c},def:function(e,t,r){var i,o,u=a(e,t);return u?u.v=r:(e[y]=u={i:o=n(t,!0),k:t,v:r,p:i=e[y],n:void 0,r:!1},e[m]||(e[m]=u),i&&(i.n=u),e[x]++,"F"!==o&&(e[g][o]=u)),e},getEntry:a,setIter:function(t,r,n){e("./$.iter-define")(t,r,function(e,t){d(this,w,{o:e,k:t})},function(){for(var e=this[w],t=e.k,r=e.l;r&&r.r;)r=r.p;return e.o&&(e.l=r=r?r.n:e.o[m])?"keys"==t?c(0,r.k):"values"==t?c(0,r.v):c(0,[r.k,r.v]):(e.o=void 0,c(1))},n?"entries":"values",!n,!0)}}},{"./$":141,"./$.assert":123,"./$.ctx":129,"./$.for-of":133,"./$.iter":140,"./$.iter-define":138,"./$.mix":143,"./$.uid":151}],127:[function(e,t,r){var n=e("./$.def"),a=e("./$.for-of");t.exports=function(e){n(n.P,e,{toJSON:function(){var e=[];return a(this,!1,e.push,e),e}})}},{"./$.def":130,"./$.for-of":133}],128:[function(e,t,r){"use strict";var n=e("./$"),a=e("./$.def"),i=e("./$.iter"),o=i.BUGGY,u=e("./$.for-of"),s=e("./$.assert").inst,l=e("./$.uid").safe("internal");t.exports=function(t,r,i,c,f,d){var h=n.g[t],p=h,v=f?"set":"add",b=p&&p.prototype,g={};return n.DESC&&n.isFunction(p)&&(d||!o&&b.forEach&&b.entries)?(p=r(function(e,r){s(e,p,t),e[l]=new h,void 0!=r&&u(r,f,e[v],e)}),n.each.call("add,clear,delete,forEach,get,has,set,keys,values,entries".split(","),function(e){var t="add"==e||"set"==e;e in b&&n.hide(p.prototype,e,function(r,n){var a=this[l][e](0===r?0:r,n);return t?this:a})}),"size"in b&&n.setDesc(p.prototype,"size",{get:function(){return this[l].size}})):(p=c.getConstructor(r,t,f,v),e("./$.mix")(p.prototype,i)),e("./$.cof").set(p,t),g[t]=p,a(a.G+a.W+a.F,g),e("./$.species")(p),d||c.setIter(p,t,f),p}},{"./$":141,"./$.assert":123,"./$.cof":125,"./$.def":130,"./$.for-of":133,"./$.iter":140,"./$.mix":143,"./$.species":148,"./$.uid":151}],129:[function(e,t,r){var n=e("./$.assert").fn;t.exports=function(e,t,r){if(n(e),~r&&void 0===t)return e;switch(r){case 1:return function(r){return e.call(t,r)};case 2:return function(r,n){return e.call(t,r,n)};case 3:return function(r,n,a){return e.call(t,r,n,a)}}return function(){return e.apply(t,arguments)}}},{"./$.assert":123}],130:[function(e,t,r){function n(e,t){return function(){return e.apply(t,arguments)}}function a(e,t,r){var i,l,c,f,d=e&a.G,h=e&a.P,p=d?o:e&a.S?o[t]:(o[t]||{}).prototype,v=d?u:u[t]||(u[t]={});d&&(r=t);for(i in r)l=!(e&a.F)&&p&&i in p,l&&i in v||(c=l?p[i]:r[i],d&&!s(p[i])?f=r[i]:e&a.B&&l?f=n(c,o):e&a.W&&p[i]==c?!function(e){f=function(t){return this instanceof e?new e(t):e(t)},f.prototype=e.prototype}(c):f=h&&s(c)?n(Function.call,c):c,v[i]=f,h&&((v.prototype||(v.prototype={}))[i]=c))}var i=e("./$"),o=i.g,u=i.core,s=i.isFunction;a.F=1,a.G=2,a.S=4,a.P=8,a.B=16,a.W=32,t.exports=a},{"./$":141}],131:[function(e,t,r){var n=e("./$"),a=n.g.document,i=n.isObject,o=i(a)&&i(a.createElement);t.exports=function(e){return o?a.createElement(e):{}}},{"./$":141}],132:[function(e,t,r){var n=e("./$");t.exports=function(e){var t=n.getKeys(e),r=n.getDesc,a=n.getSymbols;return a&&n.each.call(a(e),function(n){r(e,n).enumerable&&t.push(n)}),t}},{"./$":141}],133:[function(e,t,r){var n=e("./$.ctx"),a=e("./$.iter").get,i=e("./$.iter-call");t.exports=function(e,t,r,o){for(var u,s=a(e),l=n(r,o,t?2:1);!(u=s.next()).done;)if(i(s,l,u.value,t)===!1)return i.close(s)}},{"./$.ctx":129,"./$.iter":140,"./$.iter-call":137}],134:[function(e,t,r){t.exports=function(e){return e.FW=!1,e.path=e.core,e}},{}],135:[function(e,t,r){function n(e){try{return o(e)}catch(t){return u.slice()}}var a=e("./$"),i={}.toString,o=a.getNames,u="object"==typeof window&&Object.getOwnPropertyNames?Object.getOwnPropertyNames(window):[];t.exports.get=function(e){return u&&"[object Window]"==i.call(e)?n(e):o(a.toObject(e))}},{"./$":141}],136:[function(e,t,r){t.exports=function(e,t,r){var n=void 0===r;switch(t.length){case 0:return n?e():e.call(r);case 1:return n?e(t[0]):e.call(r,t[0]);case 2:return n?e(t[0],t[1]):e.call(r,t[0],t[1]);case 3:return n?e(t[0],t[1],t[2]):e.call(r,t[0],t[1],t[2]);case 4:return n?e(t[0],t[1],t[2],t[3]):e.call(r,t[0],t[1],t[2],t[3]);case 5:return n?e(t[0],t[1],t[2],t[3],t[4]):e.call(r,t[0],t[1],t[2],t[3],t[4])}return e.apply(r,t)}},{}],137:[function(e,t,r){function n(e){var t=e["return"];void 0!==t&&i(t.call(e))}function a(e,t,r,a){try{return a?t(i(r)[0],r[1]):t(r)}catch(o){throw n(e),o}}var i=e("./$.assert").obj;a.close=n,t.exports=a},{"./$.assert":123}],138:[function(e,t,r){var n=e("./$.def"),a=e("./$.redef"),i=e("./$"),o=e("./$.cof"),u=e("./$.iter"),s=e("./$.wks")("iterator"),l="@@iterator",c="keys",f="values",d=u.Iterators;t.exports=function(e,t,r,h,p,v,b){function g(e){function t(t){return new r(t,e)}switch(e){case c:return function(){return t(this)};case f:return function(){return t(this)}}return function(){return t(this)}}u.create(r,t,h);var y,m,w=t+" Iterator",x=e.prototype,k=x[s]||x[l]||p&&x[p],j=k||g(p);if(k){var E=i.getProto(j.call(new e));o.set(E,w,!0),i.FW&&i.has(x,l)&&u.set(E,i.that)}if((i.FW||b)&&u.set(x,j),d[t]=j,d[w]=i.that,p)if(y={keys:v?j:g(c),values:p==f?j:g(f),entries:p!=f?j:g("entries")},b)for(m in y)m in x||a(x,m,y[m]);else n(n.P+n.F*u.BUGGY,t,y)}},{"./$":141,"./$.cof":125,"./$.def":130,"./$.iter":140,"./$.redef":144,"./$.wks":153}],139:[function(e,t,r){var n=e("./$.wks")("iterator"),a=!1;try{var i=[7][n]();i["return"]=function(){a=!0},Array.from(i,function(){throw 2})}catch(o){}t.exports=function(e){if(!a)return!1;var t=!1;try{var r=[7],i=r[n]();i.next=function(){t=!0},r[n]=function(){return i},e(r)}catch(o){}return t}},{"./$.wks":153}],140:[function(e,t,r){"use strict";function n(e,t){a.hide(e,l,t),c in[]&&a.hide(e,c,t)}var a=e("./$"),i=e("./$.cof"),o=i.classof,u=e("./$.assert"),s=u.obj,l=e("./$.wks")("iterator"),c="@@iterator",f=e("./$.shared")("iterators"),d={};n(d,a.that),t.exports={BUGGY:"keys"in[]&&!("next"in[].keys()),Iterators:f,step:function(e,t){return{value:t,done:!!e}},is:function(e){var t=Object(e),r=a.g.Symbol;return(r&&r.iterator||c)in t||l in t||a.has(f,o(t))},get:function(e){var t,r=a.g.Symbol;return void 0!=e&&(t=e[r&&r.iterator||c]||e[l]||f[o(e)]),u(a.isFunction(t),e," is not iterable!"),s(t.call(e))},set:n,create:function(e,t,r,n){e.prototype=a.create(n||d,{next:a.desc(1,r)}),i.set(e,t+" Iterator")}}},{"./$":141,"./$.assert":123,"./$.cof":125,"./$.shared":147,"./$.wks":153}],141:[function(e,t,r){"use strict";function n(e){return isNaN(e=+e)?0:(e>0?v:p)(e)}function a(e,t){return{enumerable:!(1&e),configurable:!(2&e),writable:!(4&e),value:t}}function i(e,t,r){return e[t]=r,e}function o(e){return y?function(t,r,n){return w.setDesc(t,r,a(e,n))}:i}function u(e){return null!==e&&("object"==typeof e||"function"==typeof e)}function s(e){return"function"==typeof e}function l(e){if(void 0==e)throw TypeError("Can't call method on "+e);return e}var c="undefined"!=typeof self?self:Function("return this")(),f={},d=Object.defineProperty,h={}.hasOwnProperty,p=Math.ceil,v=Math.floor,b=Math.max,g=Math.min,y=!!function(){try{return 2==d({},"a",{get:function(){return 2}}).a}catch(e){}}(),m=o(1),w=t.exports=e("./$.fw")({g:c,core:f,html:c.document&&document.documentElement,isObject:u,isFunction:s,that:function(){return this},toInteger:n,toLength:function(e){return e>0?g(n(e),9007199254740991):0},toIndex:function(e,t){return e=n(e),0>e?b(e+t,0):g(e,t)},has:function(e,t){return h.call(e,t)},create:Object.create,getProto:Object.getPrototypeOf,DESC:y,desc:a,getDesc:Object.getOwnPropertyDescriptor,setDesc:d,setDescs:Object.defineProperties,getKeys:Object.keys,getNames:Object.getOwnPropertyNames,getSymbols:Object.getOwnPropertySymbols,assertDefined:l,ES5Object:Object,toObject:function(e){return w.ES5Object(l(e))},hide:m,def:o(0),set:c.Symbol?i:m,each:[].forEach});"undefined"!=typeof __e&&(__e=f),"undefined"!=typeof __g&&(__g=c)},{"./$.fw":134}],142:[function(e,t,r){var n=e("./$");t.exports=function(e,t){for(var r,a=n.toObject(e),i=n.getKeys(a),o=i.length,u=0;o>u;)if(a[r=i[u++]]===t)return r}},{"./$":141}],143:[function(e,t,r){var n=e("./$.redef");t.exports=function(e,t){for(var r in t)n(e,r,t[r]);return e}},{"./$.redef":144}],144:[function(e,t,r){t.exports=e("./$").hide},{"./$":141}],145:[function(e,t,r){t.exports=Object.is||function(e,t){return e===t?0!==e||1/e===1/t:e!=e&&t!=t}},{}],146:[function(e,t,r){function n(e,t){i.obj(e),i(null===t||a.isObject(t),t,": can't set as prototype!")}var a=e("./$"),i=e("./$.assert");t.exports={set:Object.setPrototypeOf||("__proto__"in{}?function(t,r){try{r=e("./$.ctx")(Function.call,a.getDesc(Object.prototype,"__proto__").set,2),r({},[])}catch(i){t=!0}return function(e,a){return n(e,a),t?e.__proto__=a:r(e,a),e}}():void 0),check:n}},{"./$":141,"./$.assert":123,"./$.ctx":129}],147:[function(e,t,r){var n=e("./$"),a="__core-js_shared__",i=n.g[a]||(n.g[a]={});t.exports=function(e){return i[e]||(i[e]={})}},{"./$":141}],148:[function(e,t,r){var n=e("./$"),a=e("./$.wks")("species");t.exports=function(e){!n.DESC||a in e||n.setDesc(e,a,{configurable:!0,get:n.that})}},{"./$":141,"./$.wks":153}],149:[function(e,t,r){var n=e("./$");t.exports=function(e){return function(t,r){var a,i,o=String(n.assertDefined(t)),u=n.toInteger(r),s=o.length;return 0>u||u>=s?e?"":void 0:(a=o.charCodeAt(u),55296>a||a>56319||u+1===s||(i=o.charCodeAt(u+1))<56320||i>57343?e?o.charAt(u):a:e?o.slice(u,u+2):(a-55296<<10)+(i-56320)+65536)}}},{"./$":141}],150:[function(e,t,r){"use strict";function n(){var e=+this;if(s.has(x,e)){var t=x[e];delete x[e],t()}}function a(e){n.call(e.data)}var i,o,u,s=e("./$"),l=e("./$.ctx"),c=e("./$.cof"),f=e("./$.invoke"),d=e("./$.dom-create"),h=s.g,p=s.isFunction,v=s.html,b=h.process,g=h.setImmediate,y=h.clearImmediate,m=h.MessageChannel,w=0,x={},k="onreadystatechange";p(g)&&p(y)||(g=function(e){for(var t=[],r=1;arguments.length>r;)t.push(arguments[r++]);return x[++w]=function(){f(p(e)?e:Function(e),t)},i(w),w},y=function(e){delete x[e]},"process"==c(b)?i=function(e){b.nextTick(l(n,e,1))}:h.addEventListener&&p(h.postMessage)&&!h.importScripts?(i=function(e){h.postMessage(e,"*")},h.addEventListener("message",a,!1)):p(m)?(o=new m,u=o.port2,o.port1.onmessage=a,i=l(u.postMessage,u,1)):i=k in d("script")?function(e){v.appendChild(d("script"))[k]=function(){v.removeChild(this),n.call(e)}}:function(e){setTimeout(l(n,e,1),0)}),t.exports={set:g,clear:y}},{"./$":141,"./$.cof":125,"./$.ctx":129,"./$.dom-create":131,"./$.invoke":136}],151:[function(e,t,r){function n(e){return"Symbol(".concat(void 0===e?"":e,")_",(++a+Math.random()).toString(36))}var a=0;n.safe=e("./$").g.Symbol||n,t.exports=n},{"./$":141}],152:[function(e,t,r){t.exports=function(){}},{}],153:[function(e,t,r){var n=e("./$").g,a=e("./$.shared")("wks");t.exports=function(t){return a[t]||(a[t]=n.Symbol&&n.Symbol[t]||e("./$.uid").safe("Symbol."+t))}},{"./$":141,"./$.shared":147,"./$.uid":151}],154:[function(e,t,r){var n=e("./$").core,a=e("./$.iter");n.isIterable=a.is,n.getIterator=a.get},{"./$":141,"./$.iter":140}],155:[function(e,t,r){var n=e("./$"),a=e("./$.ctx"),i=e("./$.def"),o=e("./$.iter"),u=e("./$.iter-call");i(i.S+i.F*!e("./$.iter-detect")(function(e){Array.from(e)}),"Array",{from:function(e){var t,r,i,s,l=Object(n.assertDefined(e)),c=arguments[1],f=void 0!==c,d=f?a(c,arguments[2],2):void 0,h=0;if(o.is(l))for(s=o.get(l),r=new("function"==typeof this?this:Array);!(i=s.next()).done;h++)r[h]=f?u(s,d,[i.value,h],!0):i.value;else for(r=new("function"==typeof this?this:Array)(t=n.toLength(l.length));t>h;h++)r[h]=f?d(l[h],h):l[h];return r.length=h,r}})},{"./$":141,"./$.ctx":129,"./$.def":130,"./$.iter":140,"./$.iter-call":137,"./$.iter-detect":139}],156:[function(e,t,r){var n=e("./$"),a=e("./$.unscope"),i=e("./$.uid").safe("iter"),o=e("./$.iter"),u=o.step,s=o.Iterators;e("./$.iter-define")(Array,"Array",function(e,t){n.set(this,i,{o:n.toObject(e),i:0,k:t})},function(){var e=this[i],t=e.o,r=e.k,n=e.i++;return!t||n>=t.length?(e.o=void 0,u(1)):"keys"==r?u(0,n):"values"==r?u(0,t[n]):u(0,[n,t[n]])},"values"),s.Arguments=s.Array,a("keys"),a("values"),a("entries")},{"./$":141,"./$.iter":140,"./$.iter-define":138,"./$.uid":151,"./$.unscope":152}],157:[function(e,t,r){"use strict";var n=e("./$.collection-strong");e("./$.collection")("Map",function(e){return function(){return e(this,arguments[0])}},{get:function(e){var t=n.getEntry(this,e);return t&&t.v},set:function(e,t){return n.def(this,0===e?0:e,t)}},n,!0)},{"./$.collection":128,"./$.collection-strong":126}],158:[function(e,t,r){var n=e("./$.def");n(n.S,"Object",{assign:e("./$.assign")})},{"./$.assign":124,"./$.def":130}],159:[function(e,t,r){var n=e("./$"),a=e("./$.def"),i=n.isObject,o=n.toObject;n.each.call("freeze,seal,preventExtensions,isFrozen,isSealed,isExtensible,getOwnPropertyDescriptor,getPrototypeOf,keys,getOwnPropertyNames".split(","),function(t,r){var u=(n.core.Object||{})[t]||Object[t],s=0,l={};l[t]=0==r?function(e){return i(e)?u(e):e}:1==r?function(e){return i(e)?u(e):e}:2==r?function(e){return i(e)?u(e):e}:3==r?function(e){return i(e)?u(e):!0}:4==r?function(e){return i(e)?u(e):!0}:5==r?function(e){return i(e)?u(e):!1}:6==r?function(e,t){return u(o(e),t)}:7==r?function(e){return u(Object(n.assertDefined(e)))}:8==r?function(e){return u(o(e))}:e("./$.get-names").get;try{u("z")}catch(c){s=1}a(a.S+a.F*s,"Object",l)})},{"./$":141,"./$.def":130,"./$.get-names":135}],160:[function(e,t,r){"use strict";var n=e("./$.cof"),a={};a[e("./$.wks")("toStringTag")]="z",e("./$").FW&&"z"!=n(a)&&e("./$.redef")(Object.prototype,"toString",function(){return"[object "+n.classof(this)+"]"},!0)},{"./$":141,"./$.cof":125,"./$.redef":144,"./$.wks":153}],161:[function(e,t,r){"use strict";function n(e){var t=new I(function(){});return e&&(t.constructor=Object),I.resolve(t)===t}function a(e){return N(e)&&(q?"Promise"==v.classof(e):j in e)}function i(e,t){return h.FW||e!==I||t!==d?w(e,t):!0}function o(e){var t=$(e)[k];return void 0!=t?t:e}function u(e){var t;return N(e)&&(t=e.then),P(t)?t:!1}function s(e){var t=e.c;t.length&&M.call(_,function(){function r(t){var r,i,o=a?t.ok:t.fail;try{o?(a||(e.h=!0),r=o===!0?n:o(n),r===t.P?t.rej(TypeError("Promise-chain cycle")):(i=u(r))?i.call(r,t.res,t.rej):t.res(r)):t.rej(n)}catch(s){t.rej(s)}}for(var n=e.v,a=1==e.s,i=0;t.length>i;)r(t[i++]);t.length=0})}function l(e){var t,r=e[j],n=r.a||r.c,a=0;if(r.h)return!1;for(;n.length>a;)if(t=n[a++],t.fail||!l(t.P))return!1;return!0}function c(e){var t,r=this;r.d||(r.d=!0,r=r.r||r,r.v=e,r.s=2,r.a=r.c.slice(),setTimeout(function(){M.call(_,function(){l(t=r.p)&&(O?S.emit("unhandledRejection",e,t):_.console&&console.error&&console.error("Unhandled promise rejection",e)),r.a=void 0})},1),s(r))}function f(e){var t,r=this;if(!r.d){r.d=!0,r=r.r||r;try{(t=u(e))?M.call(_,function(){var n={r:r,d:!1};try{t.call(e,p(f,n,1),p(c,n,1))}catch(a){c.call(n,a)}}):(r.v=e,r.s=1,s(r))}catch(n){c.call({r:r,d:!1},n)}}}var d,h=e("./$"),p=e("./$.ctx"),v=e("./$.cof"),b=e("./$.def"),g=e("./$.assert"),y=e("./$.for-of"),m=e("./$.set-proto").set,w=e("./$.same"),x=e("./$.species"),k=e("./$.wks")("species"),j=e("./$.uid").safe("record"),E="Promise",_=h.g,S=_.process,O="process"==v(S),M=S&&S.nextTick||e("./$.task").set,I=_[E],P=h.isFunction,N=h.isObject,A=g.fn,$=g.obj,q=function(){function e(t){var r=new I(t);return m(r,e.prototype),r}var t=!1;try{if(t=P(I)&&P(I.resolve)&&n(),m(e,I),e.prototype=h.create(I.prototype,{constructor:{value:e}}),e.resolve(5).then(function(){})instanceof e||(t=!1),t&&h.DESC){var r=!1;I.resolve(h.setDesc({},"then",{get:function(){r=!0}})),t=r}}catch(a){t=!1}return t}();q||(I=function(e){A(e);var t={p:g.inst(this,I,E),c:[],a:void 0,s:0,d:!1,v:void 0,h:!1};h.hide(this,j,t);try{e(p(f,t,1),p(c,t,1))}catch(r){c.call(t,r)}},e("./$.mix")(I.prototype,{then:function(e,t){var r=$($(this).constructor)[k],n={ok:P(e)?e:!0,fail:P(t)?t:!1},a=n.P=new(void 0!=r?r:I)(function(e,t){n.res=A(e),n.rej=A(t)}),i=this[j];return i.c.push(n),i.a&&i.a.push(n),i.s&&s(i),a},"catch":function(e){return this.then(void 0,e)}})),b(b.G+b.W+b.F*!q,{Promise:I}),v.set(I,E),x(I),x(d=h.core[E]),b(b.S+b.F*!q,E,{reject:function(e){return new(o(this))(function(t,r){r(e)})}}),b(b.S+b.F*(!q||n(!0)),E,{resolve:function(e){return a(e)&&i(e.constructor,this)?e:new this(function(t){t(e)})}}),b(b.S+b.F*!(q&&e("./$.iter-detect")(function(e){I.all(e)["catch"](function(){})})),E,{all:function(e){var t=o(this),r=[];return new t(function(n,a){y(e,!1,r.push,r);var i=r.length,o=Array(i);i?h.each.call(r,function(e,r){t.resolve(e).then(function(e){o[r]=e,--i||n(o)},a)}):n(o)})},race:function(e){var t=o(this);return new t(function(r,n){y(e,!1,function(e){t.resolve(e).then(r,n)})})}})},{"./$":141,"./$.assert":123,"./$.cof":125,"./$.ctx":129,"./$.def":130,"./$.for-of":133,"./$.iter-detect":139,"./$.mix":143,"./$.same":145,"./$.set-proto":146,"./$.species":148,"./$.task":150,"./$.uid":151,"./$.wks":153}],162:[function(e,t,r){var n=e("./$").set,a=e("./$.string-at")(!0),i=e("./$.uid").safe("iter"),o=e("./$.iter"),u=o.step;e("./$.iter-define")(String,"String",function(e){n(this,i,{o:String(e),i:0})},function(){var e,t=this[i],r=t.o,n=t.i;return n>=r.length?u(1):(e=a(r,n),t.i+=e.length,u(0,e))})},{"./$":141,"./$.iter":140,"./$.iter-define":138,"./$.string-at":149,"./$.uid":151}],163:[function(e,t,r){"use strict";function n(e){var t=L[e]=f.set(j(P.prototype),A,e);return x&&N&&G(w,e,{configurable:!0,set:function(t){k(this,$)&&k(this[$],e)&&(this[$][e]=!1),G(this,e,S(1,t))}}),t}function a(e,t,r){return r&&k(L,t)?(r.enumerable?(k(e,$)&&e[$][t]&&(e[$][t]=!1),r=j(r,{enumerable:S(0,!1)})):(k(e,$)||_(e,$,S(1,{})),e[$][t]=!0),G(e,t,r)):_(e,t,r)}function i(e,t){m(e);for(var r,n=y(t=I(t)),i=0,o=n.length;o>i;)a(e,r=n[i++],t[r]);return e}function o(e,t){return void 0===t?j(e):i(j(e),t)}function u(e){var t=q.call(this,e);return t||!k(this,e)||!k(L,e)||k(this,$)&&this[$][e]?t:!0}function s(e,t){var r=E(e=I(e),t);return!r||!k(L,t)||k(e,$)&&e[$][t]||(r.enumerable=!0),r}function l(e){for(var t,r=M(I(e)),n=[],a=0;r.length>a;)k(L,t=r[a++])||t==$||n.push(t);return n}function c(e){for(var t,r=M(I(e)),n=[],a=0;r.length>a;)k(L,t=r[a++])&&n.push(L[t]);return n}var f=e("./$"),d=e("./$.cof").set,h=e("./$.uid"),p=e("./$.shared"),v=e("./$.def"),b=e("./$.redef"),g=e("./$.keyof"),y=e("./$.enum-keys"),m=e("./$.assert").obj,w=Object.prototype,x=f.DESC,k=f.has,j=f.create,E=f.getDesc,_=f.setDesc,S=f.desc,O=e("./$.get-names"),M=O.get,I=f.toObject,P=f.g.Symbol,N=!1,A=h("tag"),$=h("hidden"),q={}.propertyIsEnumerable,D=p("symbol-registry"),L=p("symbols"),F=f.isFunction(P),G=x?function(){try{return j(_({},$,{get:function(){return _(this,$,{value:!1})[$]}}))[$]||_}catch(e){return function(e,t,r){var n=E(w,t);n&&delete w[t],_(e,t,r),n&&e!==w&&_(w,t,n)}}}():_;F||(P=function(){if(this instanceof P)throw TypeError("Symbol is not a constructor");return n(h(arguments[0]))},b(P.prototype,"toString",function(){return this[A]}),f.create=o,f.setDesc=a,f.getDesc=s,f.setDescs=i,f.getNames=O.get=l,f.getSymbols=c,f.DESC&&f.FW&&b(w,"propertyIsEnumerable",u,!0));var C={"for":function(e){return k(D,e+="")?D[e]:D[e]=P(e)},keyFor:function(e){return g(D,e)},useSetter:function(){N=!0},useSimple:function(){N=!1}};f.each.call("hasInstance,isConcatSpreadable,iterator,match,replace,search,species,split,toPrimitive,toStringTag,unscopables".split(","),function(t){var r=e("./$.wks")(t);C[t]=F?r:n(r)}),N=!0,v(v.G+v.W,{Symbol:P}),v(v.S,"Symbol",C),v(v.S+v.F*!F,"Object",{create:o,defineProperty:a,defineProperties:i,getOwnPropertyDescriptor:s,getOwnPropertyNames:l,getOwnPropertySymbols:c}),d(P,"Symbol"),d(Math,"Math",!0),d(f.g.JSON,"JSON",!0)},{"./$":141,"./$.assert":123,"./$.cof":125,"./$.def":130,"./$.enum-keys":132,"./$.get-names":135,"./$.keyof":142,"./$.redef":144,"./$.shared":147,"./$.uid":151,"./$.wks":153}],164:[function(e,t,r){e("./$.collection-to-json")("Map")},{"./$.collection-to-json":127}],165:[function(e,t,r){e("./es6.array.iterator");var n=e("./$"),a=e("./$.iter").Iterators,i=e("./$.wks")("iterator"),o=a.Array,u=n.g.NodeList,s=n.g.HTMLCollection,l=u&&u.prototype,c=s&&s.prototype;n.FW&&(!u||i in l||n.hide(l,i,o),!s||i in c||n.hide(c,i,o)),a.NodeList=a.HTMLCollection=o},{"./$":141,"./$.iter":140,"./$.wks":153,"./es6.array.iterator":156}],166:[function(e,t,r){(function(r){var n="object"==typeof r?r:"object"==typeof window?window:"object"==typeof self?self:this,a=n.regeneratorRuntime&&Object.getOwnPropertyNames(n).indexOf("regeneratorRuntime")>=0,i=a&&n.regeneratorRuntime;n.regeneratorRuntime=void 0,t.exports=e("./runtime"),a?n.regeneratorRuntime=i:delete n.regeneratorRuntime,t.exports={"default":t.exports,__esModule:!0}}).call(this,"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{"./runtime":167}],167:[function(e,t,r){(function(r,n){"use strict";var a=e("babel-runtime/core-js/symbol")["default"],i=e("babel-runtime/core-js/symbol/iterator")["default"],o=e("babel-runtime/core-js/object/create")["default"],u=e("babel-runtime/core-js/promise")["default"];!function(e){function n(e,t,r,n){var a=o((t||l).prototype);return a._invoke=v(e,r||null,new y(n||[])),a}function s(e,t,r){try{return{type:"normal",arg:e.call(t,r)}}catch(n){return{type:"throw",arg:n}}}function l(){}function c(){}function f(){}function d(e){["next","throw","return"].forEach(function(t){e[t]=function(e){return this._invoke(t,e)}})}function h(e){this.arg=e}function p(e){function t(t,r){var n=e[t](r),a=n.value;return a instanceof h?u.resolve(a.arg).then(i,o):u.resolve(a).then(function(e){return n.value=e,n})}function n(e,r){var n=a?a.then(function(){return t(e,r)}):new u(function(n){n(t(e,r))});return a=n["catch"](function(e){}),n}"object"==typeof r&&r.domain&&(t=r.domain.bind(t));var a,i=t.bind(e,"next"),o=t.bind(e,"throw");t.bind(e,"return");this._invoke=n}function v(e,t,r){var n=S;return function(a,i){if(n===M)throw new Error("Generator is already running");if(n===I){if("throw"===a)throw i;return w()}for(;;){var o=r.delegate;if(o){if("return"===a||"throw"===a&&o.iterator[a]===x){r.delegate=null;var u=o.iterator["return"];if(u){var l=s(u,o.iterator,i);if("throw"===l.type){a="throw",i=l.arg;continue}}if("return"===a)continue}var l=s(o.iterator[a],o.iterator,i);if("throw"===l.type){r.delegate=null,a="throw",i=l.arg;continue}a="next",i=x;var c=l.arg;if(!c.done)return n=O,c;r[o.resultName]=c.value,r.next=o.nextLoc,r.delegate=null}if("next"===a)n===O?r.sent=i:r.sent=x;else if("throw"===a){if(n===S)throw n=I,i;r.dispatchException(i)&&(a="next",i=x)}else"return"===a&&r.abrupt("return",i);n=M;var l=s(e,t,r);if("normal"===l.type){n=r.done?I:O;var c={value:l.arg,done:r.done};if(l.arg!==P)return c;r.delegate&&"next"===a&&(i=x)}else"throw"===l.type&&(n=I,a="throw",i=l.arg)}}}function b(e){var t={tryLoc:e[0]};1 in e&&(t.catchLoc=e[1]),2 in e&&(t.finallyLoc=e[2],t.afterLoc=e[3]),this.tryEntries.push(t)}function g(e){var t=e.completion||{};t.type="normal",delete t.arg,e.completion=t}function y(e){this.tryEntries=[{tryLoc:"root"}],e.forEach(b,this),this.reset(!0)}function m(e){if(e){var t=e[j];if(t)return t.call(e);if("function"==typeof e.next)return e;if(!isNaN(e.length)){var r=-1,n=function a(){for(;++r=0;--n){var a=this.tryEntries[n],i=a.completion;if("root"===a.tryLoc)return t("end");if(a.tryLoc<=this.prev){var o=k.call(a,"catchLoc"),u=k.call(a,"finallyLoc");if(o&&u){if(this.prev=0;--r){var n=this.tryEntries[r];if(n.tryLoc<=this.prev&&k.call(n,"finallyLoc")&&this.prev=0;--t){var r=this.tryEntries[t];if(r.finallyLoc===e)return this.complete(r.completion,r.afterLoc),g(r),P}},"catch":function(e){for(var t=this.tryEntries.length-1;t>=0;--t){var r=this.tryEntries[t];if(r.tryLoc===e){var n=r.completion;if("throw"===n.type){var a=n.arg;g(r)}return a}}throw new Error("illegal catch attempt")},delegateYield:function(e,t,r){return this.delegate={iterator:m(e),resultName:t,nextLoc:r},P}}}("object"==typeof n?n:"object"==typeof window?window:"object"==typeof self?self:void 0)}).call(this,e("_process"),"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{_process:168,"babel-runtime/core-js/object/create":93,"babel-runtime/core-js/promise":98,"babel-runtime/core-js/symbol":99,"babel-runtime/core-js/symbol/iterator":100}],168:[function(e,t,r){function n(){c=!1,u.length?l=u.concat(l):f=-1,l.length&&a()}function a(){if(!c){var e=setTimeout(n);c=!0;for(var t=l.length;t;){for(u=l,l=[];++f1)for(var r=1;r0?e[a(0,n-1)]:void 0}var l=-1,c=o(e),n=c.length,f=n-1;for(t=s(0>t?0:+t||0,n);++ln;)e=e[t[n++]];return n&&n==i?e:void 0}}var a=e("./toObject");t.exports=n},{"./toObject":221}],185:[function(e,t,r){function n(e,t,r,u,s,l){return e===t?!0:null==e||null==t||!i(e)&&!o(t)?e!==e&&t!==t:a(e,t,n,r,u,s,l)}var a=e("./baseIsEqualDeep"),i=e("../lang/isObject"),o=e("./isObjectLike");t.exports=n},{"../lang/isObject":229,"./baseIsEqualDeep":186,"./isObjectLike":217}],186:[function(e,t,r){function n(e,t,r,n,d,v,b){var g=u(e),y=u(t),m=c,w=c;g||(m=p.call(e),m==l?m=f:m!=f&&(g=s(e))),y||(w=p.call(t),w==l?w=f:w!=f&&(y=s(t)));var x=m==f,k=w==f,j=m==w;if(j&&!g&&!x)return i(e,t,m);if(!d){var E=x&&h.call(e,"__wrapped__"),_=k&&h.call(t,"__wrapped__");if(E||_)return r(E?e.value():e,_?t.value():t,n,d,v,b)}if(!j)return!1;v||(v=[]),b||(b=[]);for(var S=v.length;S--;)if(v[S]==e)return b[S]==t;v.push(e),b.push(t);var O=(g?a:o)(e,t,r,n,d,v,b);return v.pop(),b.pop(),O}var a=e("./equalArrays"),i=e("./equalByTag"),o=e("./equalObjects"),u=e("../lang/isArray"),s=e("../lang/isTypedArray"),l="[object Arguments]",c="[object Array]",f="[object Object]",d=Object.prototype,h=d.hasOwnProperty,p=d.toString;t.exports=n},{"../lang/isArray":225,"../lang/isTypedArray":232,"./equalArrays":203,"./equalByTag":204,"./equalObjects":205}],187:[function(e,t,r){function n(e,t,r){var n=t.length,o=n,u=!r;if(null==e)return!o;for(e=i(e);n--;){var s=t[n];if(u&&s[2]?s[1]!==e[s[0]]:!(s[0]in e))return!1}for(;++nt&&(t=-t>a?0:a+t),r=void 0===r||r>a?a:+r||0,0>r&&(r+=a),a=t>r?0:r-t>>>0,t>>>=0;for(var i=Array(a);++n2?r[o-2]:void 0,s=o>2?r[2]:void 0,l=o>1?r[o-1]:void 0;for("function"==typeof u?(u=a(u,l,5),o-=2):(u="function"==typeof l?l:void 0,o-=u?1:0),s&&i(r[0],r[1],s)&&(u=3>o?void 0:u,o=1);++nl))return!1;for(;++s-1&&e%1==0&&t>e}var a=/^\d+$/,i=9007199254740991;t.exports=n},{}],214:[function(e,t,r){function n(e,t,r){if(!o(r))return!1;var n=typeof t;if("number"==n?a(r)&&i(t,r.length):"string"==n&&t in r){var u=r[t];return e===e?e===u:u!==u}return!1}var a=e("./isArrayLike"),i=e("./isIndex"),o=e("../lang/isObject");t.exports=n},{"../lang/isObject":229,"./isArrayLike":212,"./isIndex":213}],215:[function(e,t,r){function n(e,t){var r=typeof e;if("string"==r&&u.test(e)||"number"==r)return!0;if(a(e))return!1;var n=!o.test(e);return n||null!=t&&e in i(t)}var a=e("../lang/isArray"),i=e("./toObject"),o=/\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\n\\]|\\.)*?\1)\]/,u=/^\w*$/;t.exports=n},{"../lang/isArray":225,"./toObject":221}],216:[function(e,t,r){function n(e){return"number"==typeof e&&e>-1&&e%1==0&&a>=e}var a=9007199254740991;t.exports=n},{}],217:[function(e,t,r){function n(e){return!!e&&"object"==typeof e}t.exports=n},{}],218:[function(e,t,r){function n(e){return e===e&&!a(e)}var a=e("../lang/isObject");t.exports=n},{"../lang/isObject":229}],219:[function(e,t,r){function n(e){for(var t=s(e),r=t.length,n=r&&e.length,l=!!n&&u(n)&&(i(e)||a(e)),f=-1,d=[];++f0;++no&&(t=r[4]?t.substr(0,o):t.substr(t.length-o)),l++}e=e.substr(0,o=r.index)+t+e.substr(a.lastIndex),a.lastIndex=t.length+o}return e};i.s=function(e,t){return t?n:e+""},t.exports=i},{}]},{},[59])(59)}); \ No newline at end of file diff --git a/tracker-server/public/javascripts/odometer.min.js b/tracker-server/public/javascripts/odometer.min.js new file mode 100644 index 0000000..48da2e9 --- /dev/null +++ b/tracker-server/public/javascripts/odometer.min.js @@ -0,0 +1,2 @@ +/*! odometer 0.4.8 */ +(function(){var a,b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u,v,w,x,y,z,A,B,C,D,E,F,G=[].slice;q='',n=''+q+"",d='8'+n+"",g='',c="(,ddd).dd",h=/^\(?([^)]*)\)?(?:(.)(d+))?$/,i=30,f=2e3,a=20,j=2,e=.5,k=1e3/i,b=1e3/a,o="transitionend webkitTransitionEnd oTransitionEnd otransitionend MSTransitionEnd",y=document.createElement("div").style,p=null!=y.transition||null!=y.webkitTransition||null!=y.mozTransition||null!=y.oTransition,w=window.requestAnimationFrame||window.mozRequestAnimationFrame||window.webkitRequestAnimationFrame||window.msRequestAnimationFrame,l=window.MutationObserver||window.WebKitMutationObserver||window.MozMutationObserver,s=function(a){var b;return b=document.createElement("div"),b.innerHTML=a,b.children[0]},v=function(a,b){return a.className=a.className.replace(new RegExp("(^| )"+b.split(" ").join("|")+"( |$)","gi")," ")},r=function(a,b){return v(a,b),a.className+=" "+b},z=function(a,b){var c;return null!=document.createEvent?(c=document.createEvent("HTMLEvents"),c.initEvent(b,!0,!0),a.dispatchEvent(c)):void 0},u=function(){var a,b;return null!=(a=null!=(b=window.performance)&&"function"==typeof b.now?b.now():void 0)?a:+new Date},x=function(a,b){return null==b&&(b=0),b?(a*=Math.pow(10,b),a+=.5,a=Math.floor(a),a/=Math.pow(10,b)):Math.round(a)},A=function(a){return 0>a?Math.ceil(a):Math.floor(a)},t=function(a){return a-x(a)},C=!1,(B=function(){var a,b,c,d,e;if(!C&&null!=window.jQuery){for(C=!0,d=["html","text"],e=[],b=0,c=d.length;c>b;b++)a=d[b],e.push(function(a){var b;return b=window.jQuery.fn[a],window.jQuery.fn[a]=function(a){var c;return null==a||null==(null!=(c=this[0])?c.odometer:void 0)?b.apply(this,arguments):this[0].odometer.update(a)}}(a));return e}})(),setTimeout(B,0),m=function(){function a(b){var c,d,e,g,h,i,l,m,n,o,p=this;if(this.options=b,this.el=this.options.el,null!=this.el.odometer)return this.el.odometer;this.el.odometer=this,m=a.options;for(d in m)g=m[d],null==this.options[d]&&(this.options[d]=g);null==(h=this.options).duration&&(h.duration=f),this.MAX_VALUES=this.options.duration/k/j|0,this.resetFormat(),this.value=this.cleanValue(null!=(n=this.options.value)?n:""),this.renderInside(),this.render();try{for(o=["innerHTML","innerText","textContent"],i=0,l=o.length;l>i;i++)e=o[i],null!=this.el[e]&&!function(a){return Object.defineProperty(p.el,a,{get:function(){var b;return"innerHTML"===a?p.inside.outerHTML:null!=(b=p.inside.innerText)?b:p.inside.textContent},set:function(a){return p.update(a)}})}(e)}catch(q){c=q,this.watchForMutations()}}return a.prototype.renderInside=function(){return this.inside=document.createElement("div"),this.inside.className="odometer-inside",this.el.innerHTML="",this.el.appendChild(this.inside)},a.prototype.watchForMutations=function(){var a,b=this;if(null!=l)try{return null==this.observer&&(this.observer=new l(function(a){var c;return c=b.el.innerText,b.renderInside(),b.render(b.value),b.update(c)})),this.watchMutations=!0,this.startWatchingMutations()}catch(c){a=c}},a.prototype.startWatchingMutations=function(){return this.watchMutations?this.observer.observe(this.el,{childList:!0}):void 0},a.prototype.stopWatchingMutations=function(){var a;return null!=(a=this.observer)?a.disconnect():void 0},a.prototype.cleanValue=function(a){var b;return"string"==typeof a&&(a=a.replace(null!=(b=this.format.radix)?b:".",""),a=a.replace(/[.,]/g,""),a=a.replace("","."),a=parseFloat(a,10)||0),x(a,this.format.precision)},a.prototype.bindTransitionEnd=function(){var a,b,c,d,e,f,g=this;if(!this.transitionEndBound){for(this.transitionEndBound=!0,b=!1,e=o.split(" "),f=[],c=0,d=e.length;d>c;c++)a=e[c],f.push(this.el.addEventListener(a,function(){return b?!0:(b=!0,setTimeout(function(){return g.render(),b=!1,z(g.el,"odometerdone")},0),!0)},!1));return f}},a.prototype.resetFormat=function(){var a,b,d,e,f,g,i,j;if(a=null!=(i=this.options.format)?i:c,a||(a="d"),d=h.exec(a),!d)throw new Error("Odometer: Unparsable digit format");return j=d.slice(1,4),g=j[0],f=j[1],b=j[2],e=(null!=b?b.length:void 0)||0,this.format={repeating:g,radix:f,precision:e}},a.prototype.render=function(a){var b,c,d,e,f,g,h;for(null==a&&(a=this.value),this.stopWatchingMutations(),this.resetFormat(),this.inside.innerHTML="",f=this.options.theme,b=this.el.className.split(" "),e=[],g=0,h=b.length;h>g;g++)c=b[g],c.length&&((d=/^odometer-theme-(.+)$/.exec(c))?f=d[1]:/^odometer(-|$)/.test(c)||e.push(c));return e.push("odometer"),p||e.push("odometer-no-transitions"),f?e.push("odometer-theme-"+f):e.push("odometer-auto-theme"),this.el.className=e.join(" "),this.ribbons={},this.formatDigits(a),this.startWatchingMutations()},a.prototype.formatDigits=function(a){var b,c,d,e,f,g,h,i,j,k;if(this.digits=[],this.options.formatFunction)for(d=this.options.formatFunction(a),j=d.split("").reverse(),f=0,h=j.length;h>f;f++)c=j[f],c.match(/0-9/)?(b=this.renderDigit(),b.querySelector(".odometer-value").innerHTML=c,this.digits.push(b),this.insertDigit(b)):this.addSpacer(c);else for(e=!this.format.precision||!t(a)||!1,k=a.toString().split("").reverse(),g=0,i=k.length;i>g;g++)b=k[g],"."===b&&(e=!0),this.addDigit(b,e)},a.prototype.update=function(a){var b,c=this;return a=this.cleanValue(a),(b=a-this.value)?(v(this.el,"odometer-animating-up odometer-animating-down odometer-animating"),b>0?r(this.el,"odometer-animating-up"):r(this.el,"odometer-animating-down"),this.stopWatchingMutations(),this.animate(a),this.startWatchingMutations(),setTimeout(function(){return c.el.offsetHeight,r(c.el,"odometer-animating")},0),this.value=a):void 0},a.prototype.renderDigit=function(){return s(d)},a.prototype.insertDigit=function(a,b){return null!=b?this.inside.insertBefore(a,b):this.inside.children.length?this.inside.insertBefore(a,this.inside.children[0]):this.inside.appendChild(a)},a.prototype.addSpacer=function(a,b,c){var d;return d=s(g),d.innerHTML=a,c&&r(d,c),this.insertDigit(d,b)},a.prototype.addDigit=function(a,b){var c,d,e,f;if(null==b&&(b=!0),"-"===a)return this.addSpacer(a,null,"odometer-negation-mark");if("."===a)return this.addSpacer(null!=(f=this.format.radix)?f:".",null,"odometer-radix-mark");if(b)for(e=!1;;){if(!this.format.repeating.length){if(e)throw new Error("Bad odometer format without digits");this.resetFormat(),e=!0}if(c=this.format.repeating[this.format.repeating.length-1],this.format.repeating=this.format.repeating.substring(0,this.format.repeating.length-1),"d"===c)break;this.addSpacer(c)}return d=this.renderDigit(),d.querySelector(".odometer-value").innerHTML=a,this.digits.push(d),this.insertDigit(d)},a.prototype.animate=function(a){return p&&"count"!==this.options.animation?this.animateSlide(a):this.animateCount(a)},a.prototype.animateCount=function(a){var c,d,e,f,g,h=this;if(d=+a-this.value)return f=e=u(),c=this.value,(g=function(){var i,j,k;return u()-f>h.options.duration?(h.value=a,h.render(),void z(h.el,"odometerdone")):(i=u()-e,i>b&&(e=u(),k=i/h.options.duration,j=d*k,c+=j,h.render(Math.round(c))),null!=w?w(g):setTimeout(g,b))})()},a.prototype.getDigitCount=function(){var a,b,c,d,e,f;for(d=1<=arguments.length?G.call(arguments,0):[],a=e=0,f=d.length;f>e;a=++e)c=d[a],d[a]=Math.abs(c);return b=Math.max.apply(Math,d),Math.ceil(Math.log(b+1)/Math.log(10))},a.prototype.getFractionalDigitCount=function(){var a,b,c,d,e,f,g;for(e=1<=arguments.length?G.call(arguments,0):[],b=/^\-?\d*\.(\d*?)0*$/,a=f=0,g=e.length;g>f;a=++f)d=e[a],e[a]=d.toString(),c=b.exec(e[a]),null==c?e[a]=0:e[a]=c[1].length;return Math.max.apply(Math,e)},a.prototype.resetDigits=function(){return this.digits=[],this.ribbons=[],this.inside.innerHTML="",this.resetFormat()},a.prototype.animateSlide=function(a){var b,c,d,f,g,h,i,j,k,l,m,n,o,p,q,s,t,u,v,w,x,y,z,B,C,D,E;if(s=this.value,j=this.getFractionalDigitCount(s,a),j&&(a*=Math.pow(10,j),s*=Math.pow(10,j)),d=a-s){for(this.bindTransitionEnd(),f=this.getDigitCount(s,a),g=[],b=0,m=v=0;f>=0?f>v:v>f;m=f>=0?++v:--v){if(t=A(s/Math.pow(10,f-m-1)),i=A(a/Math.pow(10,f-m-1)),h=i-t,Math.abs(h)>this.MAX_VALUES){for(l=[],n=h/(this.MAX_VALUES+this.MAX_VALUES*b*e),c=t;h>0&&i>c||0>h&&c>i;)l.push(Math.round(c)),c+=n;l[l.length-1]!==i&&l.push(i),b++}else l=function(){E=[];for(var a=t;i>=t?i>=a:a>=i;i>=t?a++:a--)E.push(a);return E}.apply(this);for(m=w=0,y=l.length;y>w;m=++w)k=l[m],l[m]=Math.abs(k%10);g.push(l)}for(this.resetDigits(),D=g.reverse(),m=x=0,z=D.length;z>x;m=++x)for(l=D[m],this.digits[m]||this.addDigit(" ",m>=j),null==(u=this.ribbons)[m]&&(u[m]=this.digits[m].querySelector(".odometer-ribbon-inner")),this.ribbons[m].innerHTML="",0>d&&(l=l.reverse()),o=C=0,B=l.length;B>C;o=++C)k=l[o],q=document.createElement("div"),q.className="odometer-value",q.innerHTML=k,this.ribbons[m].appendChild(q),o===l.length-1&&r(q,"odometer-last-value"),0===o&&r(q,"odometer-first-value");return 0>t&&this.addDigit("-"),p=this.inside.querySelector(".odometer-radix-mark"),null!=p&&p.parent.removeChild(p),j?this.addSpacer(this.format.radix,this.digits[j-1],"odometer-radix-mark"):void 0}},a}(),m.options=null!=(E=window.odometerOptions)?E:{},setTimeout(function(){var a,b,c,d,e;if(window.odometerOptions){d=window.odometerOptions,e=[];for(a in d)b=d[a],e.push(null!=(c=m.options)[a]?(c=m.options)[a]:c[a]=b);return e}},0),m.init=function(){var a,b,c,d,e,f;if(null!=document.querySelectorAll){for(b=document.querySelectorAll(m.options.selector||".odometer"),f=[],c=0,d=b.length;d>c;c++)a=b[c],f.push(a.odometer=new m({el:a,value:null!=(e=a.innerText)?e:a.textContent}));return f}},null!=(null!=(F=document.documentElement)?F.doScroll:void 0)&&null!=document.createEventObject?(D=document.onreadystatechange,document.onreadystatechange=function(){return"complete"===document.readyState&&m.options.auto!==!1&&m.init(),null!=D?D.apply(this,arguments):void 0}):document.addEventListener("DOMContentLoaded",function(){return m.options.auto!==!1?m.init():void 0},!1),"function"==typeof define&&define.amd?define([],function(){return m}):"undefined"!=typeof exports&&null!==exports?module.exports=m:window.Odometer=m}).call(this); diff --git a/tracker-server/public/stylesheets/demo.css b/tracker-server/public/stylesheets/demo.css new file mode 100644 index 0000000..9e9e481 --- /dev/null +++ b/tracker-server/public/stylesheets/demo.css @@ -0,0 +1,45 @@ +#demo-graph { + width: 1000px; + height: 600px; + display: block; + border: 2px solid black; + margin: 30px auto auto; +} + +.odometer { + font-size: 50px; + margin: auto; +} + +.odometer_wrapper { + text-align: center; +} + +body { + text-align: center; +} + +.numbers_table { + margin: auto; +} + +.numbers_table tr.labels { + font-size: 30px; + font-weight: bold; +} + +.numbers_table tr td { + padding: 10px 20px; +} + +h1 { + font-size: 45px; +} + +html { + background: rgba(0, 0, 0, 0.1) url(/images/background.png) no-repeat fixed center center; + -webkit-background-size: cover; + -moz-background-size: cover; + -o-background-size: cover; + background-size: cover; +} diff --git a/tracker-server/public/stylesheets/odometer-theme-car.css b/tracker-server/public/stylesheets/odometer-theme-car.css new file mode 100644 index 0000000..bdebfec --- /dev/null +++ b/tracker-server/public/stylesheets/odometer-theme-car.css @@ -0,0 +1,132 @@ +@import url("//fonts.googleapis.com/css?family=Arimo"); +.odometer.odometer-auto-theme, .odometer.odometer-theme-car { + display: inline-block; + vertical-align: middle; + *vertical-align: auto; + *zoom: 1; + *display: inline; + position: relative; +} +.odometer.odometer-auto-theme .odometer-digit, .odometer.odometer-theme-car .odometer-digit { + display: inline-block; + vertical-align: middle; + *vertical-align: auto; + *zoom: 1; + *display: inline; + position: relative; +} +.odometer.odometer-auto-theme .odometer-digit .odometer-digit-spacer, .odometer.odometer-theme-car .odometer-digit .odometer-digit-spacer { + display: inline-block; + vertical-align: middle; + *vertical-align: auto; + *zoom: 1; + *display: inline; + visibility: hidden; +} +.odometer.odometer-auto-theme .odometer-digit .odometer-digit-inner, .odometer.odometer-theme-car .odometer-digit .odometer-digit-inner { + text-align: left; + display: block; + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + overflow: hidden; +} +.odometer.odometer-auto-theme .odometer-digit .odometer-ribbon, .odometer.odometer-theme-car .odometer-digit .odometer-ribbon { + display: block; +} +.odometer.odometer-auto-theme .odometer-digit .odometer-ribbon-inner, .odometer.odometer-theme-car .odometer-digit .odometer-ribbon-inner { + display: block; + -webkit-backface-visibility: hidden; +} +.odometer.odometer-auto-theme .odometer-digit .odometer-value, .odometer.odometer-theme-car .odometer-digit .odometer-value { + display: block; + -webkit-transform: translateZ(0); +} +.odometer.odometer-auto-theme .odometer-digit .odometer-value.odometer-last-value, .odometer.odometer-theme-car .odometer-digit .odometer-value.odometer-last-value { + position: absolute; +} +.odometer.odometer-auto-theme.odometer-animating-up .odometer-ribbon-inner, .odometer.odometer-theme-car.odometer-animating-up .odometer-ribbon-inner { + -webkit-transition: -webkit-transform 0.5s; + -moz-transition: -moz-transform 0.5s; + -ms-transition: -ms-transform 0.5s; + -o-transition: -o-transform 0.5s; + transition: transform 0.5s; +} +.odometer.odometer-auto-theme.odometer-animating-up.odometer-animating .odometer-ribbon-inner, .odometer.odometer-theme-car.odometer-animating-up.odometer-animating .odometer-ribbon-inner { + -webkit-transform: translateY(-100%); + -moz-transform: translateY(-100%); + -ms-transform: translateY(-100%); + -o-transform: translateY(-100%); + transform: translateY(-100%); +} +.odometer.odometer-auto-theme.odometer-animating-down .odometer-ribbon-inner, .odometer.odometer-theme-car.odometer-animating-down .odometer-ribbon-inner { + -webkit-transform: translateY(-100%); + -moz-transform: translateY(-100%); + -ms-transform: translateY(-100%); + -o-transform: translateY(-100%); + transform: translateY(-100%); +} +.odometer.odometer-auto-theme.odometer-animating-down.odometer-animating .odometer-ribbon-inner, .odometer.odometer-theme-car.odometer-animating-down.odometer-animating .odometer-ribbon-inner { + -webkit-transition: -webkit-transform 0.5s; + -moz-transition: -moz-transform 0.5s; + -ms-transition: -ms-transform 0.5s; + -o-transition: -o-transform 0.5s; + transition: transform 0.5s; + -webkit-transform: translateY(0); + -moz-transform: translateY(0); + -ms-transform: translateY(0); + -o-transform: translateY(0); + transform: translateY(0); +} + +.odometer.odometer-auto-theme, .odometer.odometer-theme-car { + -moz-border-radius: 0.34em; + -webkit-border-radius: 0.34em; + border-radius: 0.34em; + font-family: "Arimo", monospace; + padding: 0.15em; + background: #000; + color: #eee0d3; +} +.odometer.odometer-auto-theme .odometer-digit, .odometer.odometer-theme-car .odometer-digit { + -moz-box-shadow: inset 0 0 0.3em rgba(0, 0, 0, 0.8); + -webkit-box-shadow: inset 0 0 0.3em rgba(0, 0, 0, 0.8); + box-shadow: inset 0 0 0.3em rgba(0, 0, 0, 0.8); + background-image: url(''); + background-size: 100%; + background-image: -webkit-gradient(linear, 50% 0%, 50% 100%, color-stop(0%, #333333), color-stop(40%, #333333), color-stop(60%, #101010), color-stop(80%, #333333), color-stop(100%, #333333)); + background-image: -moz-linear-gradient(top, #333333 0%, #333333 40%, #101010 60%, #333333 80%, #333333 100%); + background-image: -webkit-linear-gradient(top, #333333 0%, #333333 40%, #101010 60%, #333333 80%, #333333 100%); + background-image: linear-gradient(to bottom, #333333 0%, #333333 40%, #101010 60%, #333333 80%, #333333 100%); + padding: 0 0.15em; +} +.odometer.odometer-auto-theme .odometer-digit:first-child, .odometer.odometer-theme-car .odometer-digit:first-child { + -moz-border-radius: 0.2em 0 0 0.2em; + -webkit-border-radius: 0.2em; + border-radius: 0.2em 0 0 0.2em; +} +.odometer.odometer-auto-theme .odometer-digit:last-child, .odometer.odometer-theme-car .odometer-digit:last-child { + -moz-border-radius: 0 0.2em 0.2em 0; + -webkit-border-radius: 0; + border-radius: 0 0.2em 0.2em 0; + background-image: url(''); + background-size: 100%; + background-image: -webkit-gradient(linear, 50% 0%, 50% 100%, color-stop(0%, #eee0d3), color-stop(40%, #eee0d3), color-stop(60%, #bbaa9a), color-stop(80%, #eee0d3), color-stop(100%, #eee0d3)); + background-image: -moz-linear-gradient(top, #eee0d3 0%, #eee0d3 40%, #bbaa9a 60%, #eee0d3 80%, #eee0d3 100%); + background-image: -webkit-linear-gradient(top, #eee0d3 0%, #eee0d3 40%, #bbaa9a 60%, #eee0d3 80%, #eee0d3 100%); + background-image: linear-gradient(to bottom, #eee0d3 0%, #eee0d3 40%, #bbaa9a 60%, #eee0d3 80%, #eee0d3 100%); + background-color: #eee0d3; + color: #000; +} +.odometer.odometer-auto-theme .odometer-digit .odometer-digit-inner, .odometer.odometer-theme-car .odometer-digit .odometer-digit-inner { + left: 0.15em; +} +.odometer.odometer-auto-theme.odometer-animating-up .odometer-ribbon-inner, .odometer.odometer-auto-theme.odometer-animating-down.odometer-animating .odometer-ribbon-inner, .odometer.odometer-theme-car.odometer-animating-up .odometer-ribbon-inner, .odometer.odometer-theme-car.odometer-animating-down.odometer-animating .odometer-ribbon-inner { + -webkit-transition-timing-function: linear; + -moz-transition-timing-function: linear; + -ms-transition-timing-function: linear; + -o-transition-timing-function: linear; + transition-timing-function: linear; +} diff --git a/tracker-server/routes/index.js b/tracker-server/routes/index.js index 68401ff..600bcbc 100644 --- a/tracker-server/routes/index.js +++ b/tracker-server/routes/index.js @@ -2,6 +2,9 @@ import express from 'express'; const router = express.Router(); import app from '../app'; import NodeList from '../model/NodeList'; +const sseMW = require('./../helpers/sse'); +import Transaction from '../model/Transaction'; +import TransactionList from '../model/TransactionList'; /** * Gets all nodes. @@ -38,10 +41,23 @@ router.post('/register-node', (req, res) => { res.json({success: false, err: 'Specify id, address, port and publicKey'}); } else { const id = app.nodeList.registerNode(req.body.id, req.body.address, req.body.port, req.body.publicKey); + updateSseClients(); res.json({success: true, id: id}); } }); +router.post('/register-transaction', (req, res) => { + if(!isPresent(req.body.from) || !isPresent(req.body.to) || !isPresent(req.body.amount)|| !isPresent(req.body.remainder)|| !isPresent(req.body.numberOfChains) || !isPresent(req.body.numberOfBlocks)) { + res.status(403); + res.json({success: false, err: 'Specify from, to, amount, remainder, remainder, numberOfChains and numberOfBlocks'}); + } else { + const tx = new Transaction(req.body.from, req.body.to, req.body.amount, req.body.remainder, req.body.numberOfChains, req.body.numberOfBlocks); + app.transactionList.addTransaction(tx); + updateSseClients(); + res.json({success: true}); + } +}); + /** * Update the running status of a node. */ @@ -50,7 +66,7 @@ router.post('/set-node-status', (req, res) => { res.status(403); res.json({success: false, err: 'Specify node ID and running status'}); } else { - app.nodeList.setNodeStatus(req.body.id, req.body.running) + app.nodeList.setNodeStatus(req.body.id, req.body.running); res.json({success: true, id: req.body.id}); } }); @@ -73,11 +89,16 @@ router.get('/node', (req, res) => { } }); +router.get('/demo', (req, res) => { + res.render('demo'); +}); + /** * Reset the nodelist on the tracker server. */ router.post('/reset', (req, res) => { app.nodeList = new NodeList(); + app.transactionList = new TransactionList(); res.json({success: true}); }); @@ -92,4 +113,37 @@ function isPresent(arg) { return !!(arg || arg === 0 || arg === "" || arg === false); } +/////////////////////////// TOPN stuff ////////////////////////// + +const sseClients = new sseMW.Topic(); +// initial registration of SSE Client Connection +router.get('/topn/updates', function(req,res){ + const sseConnection = res.sseConnection; + sseConnection.setup(); + sseClients.add(sseConnection); +} ); + +/** + * send message to all registered SSE clients + */ +function updateSseClients() { + const nodes = app.nodeList.getGraphNodes(); + const edges = app.transactionList.getGraphEdges(); + sseClients.forEach(sseConnection => sseConnection.send( + {nodes: nodes, edges: edges, numbers: app.transactionList.getNumbers()})); +} + +/** + * send a heartbeat signal to all SSE clients, once every interval seconds (or every 3 seconds if no interval is specified) + * @param interval - interval in seconds + */ +function initHeartbeat(interval) { + setInterval(() => { + const msg = {"label":"The latest", "time":new Date()}; + updateSseClients( JSON.stringify(msg)); + }, interval?interval*1000:3000); +} +// initialize heartbeat at 10 second interval +initHeartbeat(10); + export default router; diff --git a/tracker-server/views/demo.ejs b/tracker-server/views/demo.ejs new file mode 100644 index 0000000..83cbe99 --- /dev/null +++ b/tracker-server/views/demo.ejs @@ -0,0 +1,44 @@ + + + + + + + + + + + + + + + +
+
+ +
+ + + +

Scale-Out Distributed Ledger

+ + + + + + + + + + + +
Total #transactionsAverage #chains per proofTotal #blocks per proof
+ +
+ + + + +
\ No newline at end of file