Skip to content
This repository has been archived by the owner on Aug 1, 2019. It is now read-only.

Latest commit

 

History

History
103 lines (90 loc) · 6.4 KB

DeveloperGuide.md

File metadata and controls

103 lines (90 loc) · 6.4 KB

Overview

This software implements a node instance of the location based network. Multiple such nodes form and maintain a geographically organized network that allows searching for nodes and their hosted services (e.g. profile server) by location.

Creating this implementation we had the following important design goals:

  • scale to a huge number of nodes (potentially millions)
  • survive both adding/losing a large number of nodes and the appearance of a national network block (e.g. the Chinese Firewall)

Consequently we made these decisions:

  • there are no distinguished supernodes, all nodes are equal
  • the full list of nodes may be too large, thus for nodes it should be enough to store only a part of the whole network.
  • each node has its own unique local network map that is redundant with the map of other nodes
  • nodes should know the geographically closer part of the network better, as the distance increases it knows less.

Consequently, nodes implement an algorithm that helps them pick which nodes they need to know about and which nodes they don’t. Each node keeps its own Network Map that consists of two parts with different goals and properties. The Neighborhood Map aims to maintain a comprehensive list of the closest nodes. The World Map aims to provide a rough coverage of the rest of the world outside the neighborhood. To prevent too much data and limit node density, the World Map picks only a single node from each area, expecting that the picked node knows its neighbourhood well thus we can ask it for further area details whenever necessary. Covered areas are defined with circular “bubbles” drawn around single nodes that must not overlap. The World Map uses a changing bubble size to have a denser local and sparser remote node density. So the greater the area, the bigger bubbles are, i.e. less nodes are included. Area bubbles are defined autonomously and are specific to the Network Map of each node. There is no node count limit for the World Map, the increasing size of bubbles serves as a limit for node counts. On the Neighborhood Map the bubbles may overlap, so it has a maximal count of neighboring nodes.

Project architecture

Directories, files and sources are organized as follows:

  • doc contains project documentation like this file
  • extlib contains external header-only libraries included with our sources for deterministic builds and easier deployment
  • generated contains the ProtoBuf sources generated from the protocol definitions for the location-based network
  • package contains utility scripts for packaging already compiled sources into binary releases.
    • libprotobuf3 contains scripts to create packages to backport protobuf version 3 to older distributions where it's not yet available
    • locnet contains scripts to pack our own binaries
  • src contains the source files for this software
    • config.h/cpp defines an interface collecting all configuration settings of the software and provides an implementation that parses options from the command line or a config file.
    • basic collects the essential typedefs, data holder structures and some utility classes used as basic building blocks in other sources
    • spatialdb defines an interface of a node database allowing search by location and gives an implementation with the SpatiaLite (an SQLite extension) engine
    • locnet provides interfaces for all different client types (local services, other network nodes and end user clients) and implements the application business logic for all interfaces in a single Node class on top of a spatial database
    • network implements asynchronous communication on top of TCP. Contains an abstraction for a Reactor task queue, a connection class to read/write byte buffers asynchronously and a generic Tcp server that accepts connections asynchronously.
    • messaging provides utilities to convert between basic application data holder classes and ProtoBuf messages. Additionally it defines an interface to dispatch requests to a peer and return its responses (i.e. hiding the complexity of communication) and defines implementations used for serving remote requests and a proxy for using services of a potentially remote node without exposing any communication details.
    • server assembles the network and messaging together. Defines a client and server classes for dispatching requests and pairing them up with responses that may come out of order. It also implements a message dispatcher that transparently sends requests to a remote peer using a network session.
    • main instantiates, links and runs all these building blocks that make up the software.
  • test contains sources to test our software

Note that basic.hpp and locnet.hpp contain types and interfaces that are redundant with the classes in IopLocNet.pb.h (generated from the ProtoBuf definitions). This redundancy is intentional: prevents direct dependance of the core application logic from generated sources used in another layer (messaging) and also allows extending the sources with additional messaging frameworks if necessary.

Components and workflow

To see how these building block fit together, let's check the workflow how clients are served.

Function main() creates TcpServer instances that bind to a TCP socket and start an asynchronous acceptor. Note that it is internally translated to an asynchronous task and added to the task queue of asio::io_service. Function io_service::run() (currently started as a separate thread from the Tcp server) consumes and executes tasks from this queue, thus our registered task accepts client connections and invokes our callback TcpServer::AsyncAcceptHandler() registered for this event.

Our implementation of this callback in DispatchingTcpServer instantiates a session and reads requests in a loop. When a request is read, a request dispatcher is invoked to deliver the message and return a response. Our configured request dispatcher translates the request from the format of our ProtoBuf protocol definition into our internal representation and invokes the appropriate method of our application logic implemented in Node. The node usually consults its SpatialDatabase to fetch or modify its map and returns the result to the dispatcher. The dispatcher then translates it back to ProtoBuf format and send it back to the client.