Monitor Incoming Payments with WebSocket

This tutorial shows how to monitor for incoming payments using the WebSocket rippled API. Since all XAG Ledger transactions are public, anyone can monitor incoming payments to any address.

WebSocket follows a model where the client and server establish one connection, then send messages both ways through the same connection, which remains open until explicitly closed (or until the connection fails). This is in contrast to the HTTP-based API model (including JSON-RPC and RESTful APIs), where the client opens and closes a new connection for each request.¹

Tip: The examples in this page use JavaScript so that the examples can run natively in a web browser. If you are developing in JavaScript, you can also use the RippleAPI library for JavaScript to simplify some tasks. This tutorial shows how to monitor for transactions without using RippleAPI so that you can translate the steps to other programming languages that don't have RippleAPI.

Prerequisites

  • The examples in this page use JavaScript and the WebSocket protocol, which are available in all major modern browsers. If you have some JavaScript knowledge and expertise in another programming language with a WebSocket client, you can follow along while adapting the instructions to the language of your choice.
  • You need a stable internet connection and access to a rippled server. The embedded examples connect to Ripple's pool of public servers. If you run your own rippled server, you can also connect to that server locally.
  • To properly handle XAG values without rounding errors, you need access to a number type that can do math on 64-bit unsigned integers. The examples in this tutorial use big.js . If you are working with issued currencies, you need even more precision. For more information, see Currency Precision.

1. Connect to the XAG Ledger

The first step of monitoring for incoming payments is to connect to the XAG Ledger, specifically a rippled server.

The following JavaScript code connects to one of Xrpgen's public server clusters. It then logs a message to the console, sends a request using the ping method and sets up a handler to log to the console again when it receives any message from the server side.

const socket = new WebSocket('wss://g1.xrpgen.com')
socket.addEventListener('open', (event) => {
  // This callback runs when the connection is open
  console.log("Connected!")
  const command = {
    "id": "on_open_ping_1",
    "command": "ping"
  }
  socket.send(JSON.stringify(command))
})
socket.addEventListener('message', (event) => {
  console.log('Got message from server:', event.data)
})
socket.addEventListener('close', (event) => {
  // Use this event to detect when you have become disconnected
  // and respond appropriately.
  console.log('Disconnected...')
})

The above example opens a secure connection (wss://) to one of Xrpgen's public API servers. To connect to a locally-running rippled server with the default configuration instead, open an unsecured connection (ws://) on port 6006 locally, using the following first line:

const socket = new WebSocket('ws://localhost:6006')

Tip: By default, connecting to a local rippled server gives you access to the full set of admin methods and admin-only data in some responses such as server_info, in addition to the public methods that are available when you connect to public servers over the internet.

Example:

Connection status: Not connected
Console:
(Log is empty)

2. Dispatch Incoming Messages to Handlers

Since WebSocket connections can have several messages going each way and there is not a strict 1:1 correlation between requests and responses, you need to identify what to do with each incoming message. A good model for coding this is to set up a "dispatcher" function that reads incoming messages and relays each message to the correct code path for handling it. To help dispatch messages appropriately, the rippled server provides a type field on every WebSocket message:

  • For any message that is a direct response to a request from the client side, the type is the string response. In this case, the server also provides the following:

    • An id field that matches the id provided in the request this is a response for. (This is important because responses may arrive out of order.)

    • A status field that indicates whether the API successfully processed your request. The string value success indicates a successful response. The string value error indicates an error.

      Warning: When submitting transactions, a status of success at the top level of the WebSocket message does not mean that the transaction itself succeeded. It only indicates that the server understood your request. For looking up a transaction's actual outcome, see Look Up Transaction Results.

  • For follow-up messages from subscriptions, the type indicates the type of follow-up message it is, such as the notification of a new transaction, ledger, or validation; or a follow-up to an ongoing pathfinding request. Your client only receives these messages if it subscribes to them.

Tip: The RippleAPI library for JavaScript handles this step by default. All asynchronous API requests use Promises to provide the response, and you can listen to streams using the .on(event, callback) method.

The following JavaScript code defines a helper function to make API requests into convenient asynchronous Promises , and sets up an interface to map other types of messages to global handlers:

const AWAITING = {}
const handleResponse = function(data) {
  if (!data.hasOwnProperty("id")) {
    console.error("Got response event without ID:", data)
    return
  }
  if (AWAITING.hasOwnProperty(data.id)) {
    AWAITING[data.id].resolve(data)
  } else {
    console.error("Response to un-awaited request w/ ID " + data.id)
  }
}

let autoid_n = 0
function api_request(options) {
  if (!options.hasOwnProperty("id")) {
    options.id = "autoid_" + (autoid_n++)
  }

  let resolveHolder;
  AWAITING[options.id] = new Promise((resolve, reject) => {
    // Save the resolve func to be called by the handleResponse function later
    resolveHolder = resolve
    try {
      // Use the socket opened in the previous example...
      socket.send(JSON.stringify(options))
    } catch(error) {
      reject(error)
    }
  })
  AWAITING[options.id].resolve = resolveHolder;
  return AWAITING[options.id]
}

const WS_HANDLERS = {
  "response": handleResponse
  // Fill this out with your handlers in the following format:
  // "type": function(event) { /* handle event of this type */ }
}
socket.addEventListener('message', (event) => {
  const parsed_data = JSON.parse(event.data)
  if (WS_HANDLERS.hasOwnProperty(parsed_data.type)) {
    // Call the mapped handler
    WS_HANDLERS[parsed_data.type](parsed_data)
  } else {
    console.log("Unhandled message from server", event)
  }
})

// Demonstrate api_request functionality
async function pingpong() {
  console.log("Ping...")
  const response = await api_request({command: "ping"})
  console.log("Pong!", response)
}
pingpong()
Responses
(Log is empty)

3. Subscribe to the Account

To get a live notification whenever a transaction affects your account, you can subscribe to the account with the subscribe method. In fact, it doesn't have to be your own account: since all transactions are public, you can subscribe to any account or even a combination of accounts.

After you subscribe to one or more accounts, the server sends a message with "type": "transaction" on each validated transaction that affects any of the specified accounts in some way. To confirm this, look for "validated": true in the transaction messages.

The following code sample subscribes to the Test Net Faucet's sending address. It logs a message on each such transaction by adding a handler to the dispatcher from the previous step.

async function do_subscribe() {
  const sub_response = await api_request({
    command:"subscribe",
    accounts: ["rUCzEr6jrEyMpjhs4wSdQdz4g8Y382NxfM"]
  })
  if (sub_response.status === "success") {
    console.log("Successfully subscribed!")
  } else {
    console.error("Error subscribing: ", sub_response)
  }
}
do_subscribe()

const log_tx = function(tx) {
  console.log(tx.transaction.TransactionType + " transaction sent by " +
              tx.transaction.Account +
              "\n  Result: " + tx.meta.TransactionResult +
              " in ledger " + tx.ledger_index +
              "\n  Validated? " + tx.validated)
}
WS_HANDLERS["transaction"] = log_tx

For the following example, try opening the Transaction Sender in a different window or even on a different device and sending transactions to the address you subscribed to:

Transactions
(Log is empty)

4. Read Incoming Payments

When you subscribe to an account, you get messages for all transactions to or from the account, as well as transactions that affect the account indirectly, such as trading its issued currencies. If your goal is to recognize when the account has received incoming payments, you must filter the transactions stream and process the payments based on the amount they actually delivered. Look for the following information:

  • The validated field indicates that the transaction's outcome is final. This should always be the case when you subscribe to accounts, but if you also subscribe to accounts_proposed or the transactions_proposed stream then the server sends similar messages on the same connection for unconfirmed transactions. As a precaution, it's best to always check the validated field.
  • The meta.TransactionResult field is the transaction result. If the result is not tesSUCCESS, the transaction failed and cannot have delivered any value.
  • The transaction.Account field is the sender of the transaction. If you are only looking for transactions sent by others, you can ignore any transactions where this field matches your account's address. (Keep in mind, it is possible to make a cross-currency payment to yourself.)
  • The transaction.TransactionType field is the type of transaction. The transaction types that can possibly deliver currency to an account are as follows:

    • Payment transactions can deliver XAG or issued currencies. Filter these by the transaction.Destination field, which contains the address of the recipient, and always use the meta.delivered_amount to see how much the payment actually delivered. XAG amounts are formatted as strings.

      Warning: If you use the transaction.Amount field instead, you may be vulnerable to the partial payments exploit. Malicious users can use this exploit to trick you into allowing the malicious user to trade or withdraw more money than they paid you.

    • OfferCreate transactions can deliver XAG or issued currencies by consuming offers your account has previously placed in the XAG Ledger's decentralized exchange. If you never place offers, you cannot receive money this way. Look at the metadata to see what currency the account received, if any, and how much.

  • The meta field contains transaction metadata, including exactly how much of which currency or currencies was delivered where. See Look Up transaction Results for more information on how to understand transaction metadata.

The following sample code looks at transaction metadata of all the above transaction types to report how much XAG an account received:

function CountXAGDifference(affected_nodes, address) {
  // Helper to find an account in an AffectedNodes array and see how much
  // its balance changed, if at all. Fortunately, each account appears at most
  // once in the AffectedNodes array, so we can return as soon as we find it.

  // Note: this reports the net balance change. If the address is the sender,
  // the transaction cost is deducted and combined with XAG sent/received

  for (let i=0; i<affected_nodes.length; i++) {
    if ((affected_nodes[i].hasOwnProperty("ModifiedNode"))) {
      // modifies an existing ledger entry
      let ledger_entry = affected_nodes[i].ModifiedNode
      if (ledger_entry.LedgerEntryType === "AccountRoot" &&
          ledger_entry.FinalFields.Account === address) {
        if (!ledger_entry.PreviousFields.hasOwnProperty("Balance")) {
          console.log("XAG balance did not change.")
        }
        // Balance is in PreviousFields, so it changed. Time for
        // high-precision math!
        const old_balance = new Big(ledger_entry.PreviousFields.Balance)
        const new_balance = new Big(ledger_entry.FinalFields.Balance)
        const diff_in_drops = new_balance.minus(old_balance)
        const XAG_amount = diff_in_drops.div(1e6)
        if (XAG_amount.gte(0)) {
          console.log("Received " + XAG_amount.toString() + " XAG.")
          return
        } else {
          console.log("Spent " + XAG_amount.abs().toString() + " XAG.")
          return
        }
      }
    } else if ((affected_nodes[i].hasOwnProperty("CreatedNode"))) {
      // created a ledger entry. maybe the account just got funded?
      let ledger_entry = affected_nodes[i].CreatedNode
      if (ledger_entry.LedgerEntryType === "AccountRoot" &&
          ledger_entry.NewFields.Account === address) {
        const balance_drops = new Big(ledger_entry.NewFields.Balance)
        const XAG_amount = balance_drops.div(1e6)
        console.log("Received " + XAG_amount.toString() + " XAG (account funded).")
        return
      }
    } // accounts cannot be deleted at this time, so we ignore DeletedNode
  }

  console.log("Did not find address in affected nodes.")
  return
}

function CountXAGReceived(tx, address) {
  if (tx.meta.TransactionResult !== "tesSUCCESS") {
    console.log("Transaction failed.")
    return
  }
  if (tx.transaction.TransactionType === "Payment") {
    if (tx.transaction.Destination !== address) {
      console.log("Not the destination of this payment.")
      return
    }
    if (typeof tx.meta.delivered_amount === "string") {
      const amount_in_drops = new Big(tx.meta.delivered_amount)
      const XAG_amount = amount_in_drops.div(1e6)
      console.log("Received " + XAG_amount.toString() + " XAG.")
      return
    } else {
      console.log("Received non-XAG currency.")
      return
    }
  } else if (["PaymentChannelClaim", "PaymentChannelFund", "OfferCreate",
          "CheckCash", "EscrowFinish"].includes(
          tx.transaction.TransactionType)) {
    CountXAGDifference(tx.meta.AffectedNodes, address)
  } else {
    console.log("Not a currency-delivering transaction type (" +
                tx.transaction.TransactionType + ").")
  }
}
Transactions
(Log is empty)

Next Steps

Footnotes

1. In practice, when calling an HTTP-based API multiple times, the client and server may reuse the same connection for several requests and responses. This practice is called HTTP persistent connection, or keep-alive . From a development standpoint, the code to use an HTTP-based API is the same regardless of whether the underlying connection is new or reused.