Skip to content

A simple WebSocket API that provides real-time information about the Kosu network.

Notifications You must be signed in to change notification settings

ParadigmFoundation/network-explorer

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

37 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Example: network-explorer

The network summary API provides an example of a simple WebSocket subscription for various statistics about the Kosu network and contract system. It also provides a set of simple REST methods to assist the creation of a network summary front-end.

It is purpose-built for explore.kosu.io, and likely has limited utility outside of that purpose, but it also stands to serve as an example of using kosu.js to connect with the Kosu network.

Develop

Configure

To run for development purposes, set the following environment variables (defaults shown):

# tcp port to bind ws server to
PORT="4242"

# host to bind to (defaults to hostname)
HOST=$(hostname)

# order stream JSONRPC url to source data from (don's use the default)
ETHEREUM_JSONRPC_URL="https://ropsten.infura.io"

# kosu node serving JSONRPC API/WebSocket
KOSU_JSONRPC_URL="ws://localhost:14342"

# redis config
REDIS_HOST="localhost"
REDIS_PORT="6379"

Clone and build

Get started with the server (must have docker and yarn):

  1. Clone: git clone [email protected]:ParadigmFoundation/kosu-monorepo
  2. Install deps: yarn (not npm i)
  3. Go to package: cd pacakges/network-explorer
  4. Set environment vars described above (or place and source a .env file)
  5. Build with yarn build
  6. Start with yarn start (will start a detached redis docker container)

Specification

The primary usage of this tool is the WebSocket API that push various blockchain and network data over a client <> server connection for each connected party. The schematic of the output object is shown and described below.

Usage

WebSocket API usage subject to change.

The network-summary-api server is designed to be as simple as possible. The application exposes a WebSocket server on a port defined by the PORT environment variable.

When a client connects to the server, the server immediately starts pushing API data to the client with a format described here (after sending a one-tme connection message with a subscriptionId). This is referred to as the subscription at points in this document.

The server also allows two types of request/response methods for accessing users token balances and rate limits. These requests (described here) should be sent over the same WebSocket connection as the subscription, and should use a unique client-generated id string. The client-provided id will be returned with successful request responses (along with a 0 OK code) so the client listen for the correct response message, and filter it from the stream of subscription messages.

In the future, the subscription may need to be initiated with a client request.

Connection message

Upon successful connection to the server, the server pushes a one-time "connection" message, with the subscription ID included in all future subscription messages. Keep in mind this subscription ID should be used to filter subscription messages from request-response messages.

// => sent to client (immediately upon successful connect)
{
    "message": "18706ca5-a7d7-4781-b027-acf6c52c2cc6"
}

Request format

The server accepts two query methods: balance and limit. These methods can be used by the browser client to request data about a users account via their Ethereum address (loaded via MetaMask on the front-end).

Use the same socket connection as the subscription to send messages of the following format:

/* === BALANCE REQUEST/RESPONSE EXAMPLE ===*/

// <= sent to server
{
    "id": "any-client-provided-string",
    "method": "balance",
    "param": "0x8b366a3d4e46ac5406f12766ad33e6482ce4f081"
}

// => sent to client
{
    "id": "any-client-provided-string",
    "code": 0,
    "data": "108691111111111111111"
}

/* === LIMIT REQUEST/RESPONSE EXAMPLE ===*/

// <= sent to server
{
    "id": "myId",
    "method": "limit",
    "param": "0x8b366a3d4e46ac5406f12766ad33e6482ce4f081"
}

// => sent to client
{
    "id": "myId",
    "code": 0,
    "data": "74891"
}

/* === BAD REQUEST/RESPONSE EXAMPLE ===*/

// <= sent to server (malformed request)
{
    "id": "anyId",
    "method": "limit"
}

// => sent to client (error response example, no id)
{
    "id": null,
    "code": 1,
    "data": "missing required parameters"
}

For the user's coinbase (Ethereum address, pulled via MetaMask injection), two requests will need to be made, exactly like the balance and limit examples above.

The balance method returns the number of Kosu tokens a user has (unlocked, un-staked) in their wallet (Ethereum address). It will return a value in the smallest unit of KOSU (a wei = 1*10^-18 KOSU) so a conversion will need to be made (multiply by 1*10^18). The Web3 library has utilities to make this conversion simple.

Browser example

Very simple example that demonstrates connection and parsing various messages. Primarily useful to demonstrate boolean conditions for various parsed message structures.

const ws = new WebSocket("wss://explore-api.kosu.io/");

// will be sent upon connect as `message`, and in each subscription as `id`
let subId;

ws.onopen = () => {
    ws.onmessage = msg => {
        const data = msg.data.toString();
        const parsed = JSON.parse(data);
        const { id, code, data, message } = parsed;

        /*
         in this case, this is the first message and the server is telling
         us our subscription id
        */
        if (message) {
            // store our subId for future use
            subId = message.toString();

            // in this case, `data` is subscription data (see below)
        } else if (subId && id === subId && data) {
            // handle subscription data here
            // in this case, the server has encountered an error processing a subscription
        } else if (subId && id === subId && !data) {
            // ignore this message and don't update displayed data
            // in this case `data` is an OK response to a client request
        } else if (subId && id !== subId && data && code === 0) {
            // handle server response message here
            // the `id` field will be the same `id` that was included in the request
            /*
         in this case, an error was encountered processing a client request
         and the method has failed.
        */
        } else if (subId && !id && data && code === 1) {
            // the `data` field may contain error information
            // unexpected unknown case
        } else {
            // ignore?
        }
    };
};

API Schema

Structure of the stringified object streamed to each connection, updated upon each new block.

For each field there is a number and corresponding description below that provides more information about each field.

The numbers in the code snippet below also match the annotated mockup showing where each field rendered on the front-end should source data.

Note: every value (except for parent objects) is sent as a string, so conversions must be made client-side. Keep in mind overflows/loss-of-precision when dealing with token balances.

// js-flavored JSON with annotations (see below)

{
    /*
     A unique `ID` string is provided by the server when a client connection is
     initiated. It can be used to filter subscription messages (like these) from
     request/response messages used to query a users address (from MetaMask) for
     their token balances and OS rate limits.
    */

    "id": "18706ca5-a7d7-4781-b027-acf6c52c2cc6",

    /*
     The data object contains all API data. Mostly, each field is updated every
     block, even if the data does not change. If the server encounters an error
     with a particular query, it will omit that field, so the client should
     keep a cache and not over-write displayed values with `null` or `undefined`
     even if that is the value returned by the API. This will be fixed in future
     server versions.
    */
    "data": {
        "token": {
            "totalSupply": "111111111",        // 1
            "price": "0"                       // 2
        },
        "bandwidth": {
            "totalLimit": "75000",             // 3
            "totalOrders": "1210351",          // 4
            "remainingLimit": "73125",         // 5
            "numberPosters": "1514",           // 6
            "secToNextPeriod": "60",           // 7
            "currentEthBlock": "5230840",      // 8
            "periodEndEthBlock": "5230844",    // 9
            "rebalancePeriodNumber": "27940"   // 10
        },
        "network": {
            "blockHeight": "1327413",                  // 11
            "lastBlock_time": "1551727994832",         // 12
            "avgBlock_interval": "1492",               // 13
            "numberValidators": "32",                  // 14
            "totalValidatorStake": "65314806500000",   // 15
            "totalPosterStake": "421806500030"         // 16
        },
        "transactions": [                   // 17
            // ...
            {
                "orderId": "...",          // 17 (a)
                "posterAddress": "0x....", // 17 (b)
                "makerAddress": "0x...",   // 17 (c)
                "orderType": "0x"          // 17 (d)
            }
            // ...
        ],
        "validators": [                     // 18
            // ...
            {
                "publicBey": "...",         // 18 (a)
                "stake": "1345600000000",   // 18 (b)
                "reward": "1200000000",     // 18 (c)
                "uptimePercent": "11",      // 18 (d)
                "firstVote": "45102",       // 18 (e)
                "lastVoted": "1327413",     // 18 (f)
                "totalVotes": "1282014",    // 18 (g)
                "power":"10"                // 18 (h)
            }
            // ...
        ]
    }
}

Annotations

Note: All values are strings (double-quoted) in the JSON sent to the client with each new block. Convert to JS number type in the browser where necessary.

  1. Kosu token supply (data.token.totalSupply): the total number of Kosu tokens in circulation. Keep in mind this number will be in units of wei (smallest divisible unit) so for the actual "quantity" of full KOSU tokens, this value will need to be multiplied by 1 * 10^18.

  2. Kosu token price (data.token.price): the price of one Kosu in Ether, based on the real-time price from the Kosu bonding contract.

  3. Total limit (data.bandwidth.totalLimit): this value is the current network-agreed-upon value for the total number of order transactions that can be accepted per rebalance period. It is a consensus-critical parameter that is unlikely to change very often.

  4. Total orders (data.bandwidth.totalOrders): this incremental value simply counts the total number of order transactions accepted on the network since genesis. It is independent of any individual period.

  5. Remaining limit (data.bandwidth.remainingLimit): the number of remaining (unused) allocated orders for the current period. It will count down during each period as orders are accepted, and will reset to the total_limit upon each rebalance event.

  6. Number of posters (data.bandwidth.numberPosters): the total number of poster accounts; the number of addresses with KOSU registered to post order's to the OS network.

  7. Seconds to next period (data.bandwidth.secToNextPeriod): a (very) rough estimation of the number of seconds to the next rebalance period. Taken by counting the number of Ethereum blocks until the next period, multiplied by the average Ethereum block-time.

  8. Current Ethereum block (data.bandwidth.currentEthBlock): the best-known (highest) Ethereum block number. Updated each time an Kosu block is committed.

  9. Period end Ethereum block (data.bandwidth.periodEndEthBlock): the Ethereum block number at which the current rebalance period ends.

  10. Rebalance period number (data.bandwidth.rebalancePeriodNumber): an incremental counter that tracks the number of rebalance periods that have occurred. Displays the number of the currently active period (not the last completed).

  11. Block height (data.network.blockHeight): updates with each new committed block, the block_height increases by 1. It tracks the current height of the Kosu blockchain.

  12. Last block time (data.network.lastBlockTime): the UNIX timestamp (in milliseconds) of the most recently committed block.

  13. Average interval (data.network.avgBlockInterval): the average_block_interval tracks the arithmetic moving average (over a server-configurable number of blocks) of the interval in milliseconds between each block. Commonly referred to as the "block-time", it can be shown in seconds by dividing the number by 1000.

  14. Validator counter (data.network.numberValidators): simply tracks the number of validators in the active validator set. It will change only when a new validator is added, or a current one is removed via governance processes.

  15. Total validator stake (data.network.totalValidatorStake): the total number of KOSU (in wei) staked in the ValidatorRegistry contract by active validators. Also an important metric for determining network security and value.

  16. Total poster stake (data.network.totalPosterStake): the total number of KOSU (in wei) staked in the PosterRegistry contract by poster accounts.

  17. Transactions (data.transactions): an array of objects (defined below) with some data representing the most recent 20 order transactions from the network.

    a. Order ID (data.transactions[N].orderId): a hash of the Paradigm order object, used to identify orders.

    b. Poster address (data.transactions[N].posterAddress): the Ethereum address of the poster entity who signed and submitted the order.

    c. Maker address (data.transactions[N].makerAddress): the Ethereum address of the maker entity who originated the order message.

    d. Order type (data.transactions[N].orderType): if the subContract for a given order is known, the orderType field will be a short string of the name of the settlement logic (such as "0x", or "Dharma") and will be null if unknown.

  18. Validators (data.validators): an array of objects (defined below) for each active validator on the network.

    a. Public Key (data.validators[N].publicKey): the validators active tendermint public key, which corresponds to a current validating private key. Hashed in multiple ways to generate node_id, etc. Also used to query the ValidatorRegistry contract.

    b. Stake (data.validators[N].stake): a specific validators KOSU stake (in wei) currently held in the ValidatorRegistry contract.

    c. Reward (data.validators[N].reward): the number of Kosu tokens (in wei) the validator has specified to receive (or burn, if negative) for each reward period.

    d. Uptime percent (data.validators[N].uptimePercent): a number between 0 and 100 that represents that validators uptime: the percentage of time they have been online (voting on blocks) since they were added as a validator.

    e. First block (data.validators[N].firstVote): the height of the first Kosu block that a given validator voted on. Used as, or to calculate a validators "age" in the network.

    f. Last voted (data.validators[N].lastVoted): the height of OrderStream network at which a given validator voted (or proposed) a block.

    g. Total votes (data.validators[N].totalVotes): the total number of blocks a given validator has voted on.

    h. Vote power (data.validators[N].power): the vote power the validator has on the Tendermint chain. Also affects how often a given validator is selected as block proposer.

Notes

  • Values should be cached by the client (or stored in the DOM) and only updated when necessary
  • Error messages will not have an id field
  • Error messages send as responses to requests will have a code set to 0
  • Reach out to [email protected] with API difficulties

About

A simple WebSocket API that provides real-time information about the Kosu network.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published