Design

Promise-based client

To prevent a "callback hell" structure where code gets nested and messy; making it harder to understand. A Promise<?> based design is used as async/await implementation are very mature today, it brings better quality of life to modern development.

import {JsonRpcClient} from '@defichain/jellyfish-api-jsonrpc'

const client = new JsonRpcClient('http://foo:bar@localhost:8554')
const {blocks} = await client.mining.getMiningInfo()

IEEE-754 arbitrary precision

Due to the dynamic nature of the JavaScript language, it forces all number to be interpolated as IEEE-754 which can cause precision to be lost. JellyfishSDK/jellyfish/issues/18

it('lost precision converting DFI 😥', () => {
  const n = 1200000000.00000001
  const a = JSON.parse(JSON.stringify(n)) * 1.0e8
  expect(a.toString()).toStrictEqual("120000000000000001")
});

JellyfishJSON

api-core implements JellyfishJSON that allows parsing of JSON with 'lossless', 'bignumber' and 'number' numeric precision.

  • 'lossless' uses LosslessJSON that parses numeric values as LosslessNumber. With LosslessNumber, one can perform regular numeric operations, and it will throw an error when this would result in losing information.
  • 'bignumber' parse all numeric values as 'BigNumber' using bignumber.js library.
  • 'number' parse all numeric values as 'Number' and precision will be loss if it exceeds IEEE-754 standard.
  • 'PrecisionPath' provides path based precision mapping, specifying 'bignumber' will automatically map all Number in that path as 'bignumber'. Otherwise, it will default to number, This applies deeply.

As not all numbers parsed are significant in all context, (e.g. mining.getMiningInfo()), this allows jellyfish library users to use the number for non precision sensitive operation (e.g. networkhashps) and BigNumber for precision sensitive operations.

ApiClient

As jellyfish is written in TypeScript, all RPC exchanges with a node are typed. BigNumber precision is used for all wallet or transaction related operations. While IEEE-754 number is used for all other arbitrary operations.

export class Wallet {
  async getBalance (minimumConfirmation: number = 0, includeWatchOnly: boolean = false): Promise<BigNumber> {
    return await this.client.call('getbalance', ['*', minimumConfirmation, includeWatchOnly], 'bignumber')
  }
}
export interface MiningInfo {
  blocks: number
  currentblockweight?: number
  //...
}

export class Mining {
  async getNetworkHashPerSecond (nblocks: number = 120, height: number = -1): Promise<number> {
    return await this.client.call('getnetworkhashps', [nblocks, height], 'number')
  }

  async getMiningInfo (): Promise<MiningInfo> {
    return await this.client.call('getmininginfo', [], 'number')
  }
}

Protocol agnostic core

ApiClient in api-core is a protocol agnostic DeFiChain client implementation with APIs separated into their category. The protocol-agnostic core enables independent communication protocols, allowing vendor-agnostic middleware adaptable to any needs.

export abstract class ApiClient {
  /**
   * A promise based procedure call handling
   *
   * @param method Name of the RPC method
   * @param params Array of params as RPC payload
   * @param precision
   * Numeric precision to parse RPC payload as 'lossless', 'bignumber' or 'number'.
   *
   * 'lossless' uses LosslessJSON that parses numeric values as LosslessNumber. With LosslessNumber, one can perform
   * regular numeric operations, and it will throw an error when this would result in losing information.
   *
   * 'bignumber' parse all numeric values as 'BigNumber' using bignumber.js library.
   *
   * 'number' parse all numeric values as 'Number' and precision will be loss if it exceeds IEEE-754 standard.
   *
   * @throws ApiError
   * @throws RpcApiError
   * @throws ClientApiError
   */
  abstract call<T> (method: string, params: any[], precision: Precision): Promise<T>
}