🤖REST API v1 (deprecated)

This page lists all available endpoints of the deprecated Boltz API v1.

API v1 is maintained for existing integrations, we highly recommend using API v2 for all new integrations.

Basics

Response and request encoding

All the responses to all calls are encoded as JSON objects. If endpoints require the client to provide any kind of arguments these also have to be encoded as JSON and sent in the body of a POST request. Make sure to set the Content-Type header of your POST requests to application/json.

Error handling

If a call fails for some reason, the returned HTTP status code will indicate that. Additionally, an object will be returned that includes the reason why the call failed:

{
  "error": "message explaining why the call failed"
}

Backend Version

Returns the version of Boltz Backend serving the API. A good call to get started and see if Boltz API requests and responses are working.

URLResponse

GET /version

JSON object

Status Codes:

  • 200 OK

Response object:

  • version: The deployed version of Boltz Backend.

Examples:

GET /version

{
  "version": "3.2.1-7c38088"
}

Supported Pairs

In order to create a swap, one first has to know which pairs are supported and what kind of rates, limits and fees are applied when creating a new swap. The following call returns this information.

URLResponse

GET /getpairs

JSON object

Status Codes:

  • 200 OK

Response object:

  • info: Contains information about special configuration parameters of the Boltz Backend deployment. Currently there is only one:

    • prepay.minerfee: If the array contains this value, Boltz requires a small invoice for the miner fee to be paid before the actual hold invoice of a Reverse Swap is revealed.

  • warnings: An array of strings that indicate that some feature of Boltz might me disabled or restricted. An example is:

    • reverse.swaps.disabled: Means that all reverse swaps (from Lightning to the chain) are disabled.

  • pairs: An object containing the supported pairs. The keys of the values are the IDs of the pairs (BTC/BTC is a special case with mainchain bitcoin as base asset and Lightning bitcoin as quote asset) and the values itself contain information about the pair:

    • hash: SHA256 hash of the JSON encoded data in the pair object.

    • rate: The exchange rate of the pair.

    • limits: A JSON Object containing the minimal and maximal amount of the pair's swap. The numbers are denominated 10 ** -8 of the quote asset.

      • maximalZeroConf: The maximal amounts that will be accepted without chain confirmations by Boltz. 0 indicates that Boltz will not accept 0-conf. See 0-conf for more info.

    • fees: A JSON object that contains different kinds of fees:

      • percentage: The percentage of the "send amount" that is charged by Boltz as "Boltz Fee" for swaps from quote to base asset (e.g. Lightning -> Bitcoin).

      • percentageSwapIn: The percentage of the "send amount" that is charged by Boltz as "Boltz Fee" for swaps from base to quote asset (e.g. Bitcoin -> Lightning).

      • minerFees: The network fees charged for locking up and claiming funds onchain. These values are absolute, denominated in 10 ** -8 of the quote asset.

Examples:

GET /getpairs

Response:

{
  "info": [],
  "warnings": [],
  "pairs": {
    "BTC/BTC": {
      "hash": "05d0f522ef08dd66fa0f87f167cc1380eaf7e5227e698941ecb44876e0736cb8",
      "rate": 1,
      "limits": {
        "maximal": 10000000,
        "minimal": 50000,
        "maximalZeroConf": {
          "baseAsset": 0,
          "quoteAsset": 0
        }
      },
      "fees": {
        "percentage": 0.5,
        "percentageSwapIn": 0.1,
        "minerFees": {
          "baseAsset": {
            "normal": 1360,
            "reverse": {
              "claim": 1104,
              "lockup": 1224
            }
          },
          "quoteAsset": {
            "normal": 1360,
            "reverse": {
              "claim": 1104,
              "lockup": 1224
            }
          }
        }
      }
    },
    "L-BTC/BTC": {
      "hash": "769215ae0f8cb14d250374a77de530ac2887c927dc08f8efce74f3634df03171",
      "rate": 1,
      "limits": {
        "maximal": 10000000,
        "minimal": 10000,
        "maximalZeroConf": {
          "baseAsset": 0,
          "quoteAsset": 0
        }
      },
      "fees": {
        "percentage": 0.25,
        "percentageSwapIn": 0.1,
        "minerFees": {
          "baseAsset": {
            "normal": 147,
            "reverse": {
              "claim": 152,
              "lockup": 276
            }
          },
          "quoteAsset": {
            "normal": 1360,
            "reverse": {
              "claim": 1104,
              "lockup": 1224
            }
          }
        }
      }
    }
  }
}

Creating Normal Submarine Swaps

This section walks you through creating Normal Submarine Swaps (Chain -> Lightning). They differ slightly depending on the kind of bitcoin that are swapped, more information can be found below. Please note that Boltz works with 10 ** -8 decimals internally and all amounts in the API endpoints follow this denomination. All requests to create Normal Submarine Swaps have the following common values in the API request:

  • type: The type of swap to create. For Normal Submarine Swaps this is submarine .

  • pairId: The pair of which the swap should be created, query available pairs via /getpairs.

  • orderSide: Possible values are buy & sell. Currently, we recommend using sell across all pairs of swap type submarine. The value buy for Normal Submarine Swaps of e.g. the L-BTC/BTC pair signifies a swap from Bitcoin mainchain to Liquid Lightning. Currently, this is not supported and the backend will return "error": "L-BTC has no lightning support".

If you already know the amount to be swapped, you should also set invoice.

  • invoice: The Lightning invoice of the user that should be paid.

If the amount is not known yet, a preimage hash has be specified. The invoice that is provided later during the lifecycle of the Submarine Swap has to have the same preimage hash as the one specified here.

  • preimageHash: Hash of a preimage that will be used for the invoice that is set later on.

We recommend verifying that pair data fetched previously (like minerFees) is still valid by additionally passing the pairHash argument in this call.

  • pairHash: hash string of the pair object of /getpairs.

Members of our partner program may set this optional referral parameter to get a percentage of the fees earned from referred swaps as kickback.

  • referralId: Partner referral ID.

URLResponse

POST /createswap

JSON object

Status Codes:

  • 201 Created

  • 400 Bad Request: The swap could not be created. Check the error string in the JSON object of the body of the response for more information.

Response objects:

  • id: Id of the newly created swap.

  • timeoutBlockHeight: Base asset block height at which the swap will expire.

  • address: Address in which the bitcoin will be locked up. For the Bitcoin mainchain, this is a SegWit P2SHP2WSH address (P2WSH nested in a P2SH) for the sake of compatibility, for Liquid a P2WSH address and for EVM chains the address of the corresponding swap contract.

If a Lightning invoice is set in this call, one will also find the following values in the response:

  • acceptZeroConf: Whether Boltz will accept 0-conf for this swap.

  • expectedAmount: The amount that Boltz expects to be locked on the chain.

Normal Swaps: UTXO Chains

For UTXO chains, /createswap requests have to contain one additional parameter:

  • refundPublicKey: public key of a keypair that will allow the user to refund the locked up bitcoin once the time lock is expired. This keypair has to be generated and stored by the client integrating Boltz API.

Responses also contain one additional value:

  • redeemScript: redeem script from which the address is derived. The redeem script should be used, to verify that Boltz did provide the correct address.

In case the address is for the Liquid Network, it will be blinded by a key that is also in the response:

  • blindingKey: hex encoded private key with which the address was blinded

If the invoice has been set, you will also get this value:

Examples:

POST /createswap

Request body:

{
  "type": "submarine",
  "pairId": "BTC/BTC",
  "orderSide": "sell",
  "refundPublicKey": "AAAAE2VjZHNhLXNoYTItbmlzdHAyNTYAAAAIbmlzdHAyNTYAAABBBGBkraVi05Eyn6slRQV5h+fX6rtpudOq0LqPEnbnbxRshMdhS56vKWawUNLkLZZ4hKsTdbJZvgTtO/rDc2WI/Gw",
  "invoice": "lntb1m1pjv0dt5pp5y9dl3z50c5g6p26a86g432zdzvdlx6a565hk55a2ellz3t9f84jsdqqcqzzsxqyz5vqsp5a4k0f59u62na3fngv24nv5xjuxyf6qjnnj806se373h4gt9fmejq9qyyssqpeh42yy72pqzfwdfehvuru9s735vrgg324lxdp9gg8w6m379w8ajd3sxyy6f0qqfqa6vhk5k4pqfz6nys3u5xf68wcjyjygykn7za6cqf6flce"
}

Response:

{
  "id": "E63LC4",
  "bip21": "bitcoin:2NBBYeBZgY64nJKiibnGokwrBBPjoQeMzyx?amount=0.0010054&label=Send%20to%20BTC%20lightning",
  "address": "2NBBYeBZgY64nJKiibnGokwrBBPjoQeMzyx",
  "redeemScript": "a9148f8d01a3e1a794024fa78bd9c81d5ae9bb1c56d287632102bba4fbfe50ea8caf880cb367f7b0083c7e91c2bc2808c817823bd36385e3a376670319b125b17503aaaae268ac",
  "acceptZeroConf": false,
  "expectedAmount": 100540,
  "timeoutBlockHeight": 2470169
}

Normal Swaps: EVM Chains

Currently, Boltz only supports RSK Testnet.

Swaps from account-based EVM chains like RSK do not require a new address for every swap. /createswap takes the details of the swap (like lightning invoice and pair) and Boltz waits until the user locked e.g. RBTC in the contract. The addresses of those contracts can be queried with /getcontracts and the address of the contract that needs to be used for the swap is also returned in the response of this request.

For EVM chains, the request does not require any additional values, but the response returns one additional value:

  • claimAddress: The EVM chain destination address of Boltz. It is specified in thelock function of the swap contract.

Examples:

POST /createswap

Request body:

{
  "type": "submarine",
  "pairId": "RBTC/BTC",
  "orderSide": "sell",
  "invoice": "lntb700u1pj5gft9pp5he32emhme8nrl6u7yyaa7s8svcl3kcwlwz99yzsck09rh4tr9fzqdqqcqzzsxqyz5vqsp5jztrjz3qx8tekwgwqlp7up2c0fvyvpdnd54flk6f9hgytv43ca7q9qyyssqk3l4zr9xs80x0ymg3jczd0t6crwpq5klv9uu6xzgn8fvsr9rraqhluaxstcnxg4ka0frerzrxh8ghgf2ey44hvgdhjke4zcrw8ukugspe9jrvy"
}

Response:

{
  "id": "wXKtEu",
  "address": "0x165F8E654b3Fe310A854805323718D51977ad95F",
  "claimAddress": "0x4217BD283e9Dc9A2cE3d5D20fAE34AA0902C28db",
  "acceptZeroConf": false,
  "expectedAmount": 70513,
  "timeoutBlockHeight": 4455644
}

Swap Rates

In case the amount to be swapped is not known when creating a Normal Submarine Swap, the invoice can be set afterwards; even if the chain bitcoin were sent already. In this case, you want to first use this endpoint to figure out what the exact amount of the invoice should be based on the already sent bitcoin. Send a POST request with a JSON encoded body with this value:

  • id: Id of the Submarine Swap.

URLResponse

POST /swaprates

JSON object

Status Codes:

  • 200 OK

  • 400 Bad Request: The invoice amount could not be calculated. Check the error string in the JSON object of the body of the response for more information. A common case is where the user did not lock up chain bitcoin yet, which is a requirement in order to calculate an invoice amount: "error": "no coins were locked up yet".

Response object:

  • invoiceAmount: Amount of the invoice that should be set with /setinvoice.

Examples:

Request body:

{
  "id": "BY8asG"
}

Response:

{
  "invoiceAmount": 15713393
}

Setting an Invoice

In case the amount to be swapped is not known when creating a Normal Submarine Swap, the invoice can be set afterwards; even if the chain bitcoin were sent already. Please keep in mind that the invoice has to have the same preimage hash that was specified when creating the swap. Although the invoice can be changed after setting it initially, this endpoint will only work if Boltz did not try to pay the initial invoice yet. Requests to this endpoint have to be POST and should have the following values in its JSON encoded body:

  • id: Id of the swap for which the invoice should be set.

  • invoice: Invoice of the user that should be paid.

URLResponse

POST /setinvoice

JSON object

Status Codes:

  • 200 OK

  • 400 Bad Request: The invoice could not be set. Check the error string in the JSON object of the body of the response for more information.

Response objects:

What is returned when the invoice is set depends on the status of the Normal Submarine Swap. If no funds were sent (status swap.created) the endpoint will return a JSON object with these values:

  • acceptZeroConf: Whether Boltz will accept 0-conf for this swap.

  • expectedAmount: The amount that Boltz expects you to lock in the chain HTLC.

  • bip21: A BIP21 payment request for the expectedAmount of bitcoin and the address (only set when swapping from UTXO chains).

If chain bitcoin were sent already (status transaction.mempool or transaction.confirmed) the endpoint will return an empty JSON object, signifying success.

In case this endpoint is called again after an invoice was set and Boltz already tried to pay the invoice, the following response objects are returned:

  • error: Error message explaining that Boltz tried to pay the invoice already and that it cannot be changed anymore.

  • invoice: The invoice that was set and that will be used for the swap.

Examples:

If no bitcoin were sent yet:

POST /setinvoice

Request body:

{
  "id": "UwHIPg",
  "invoice": "lnbcrt1m1p08epqjpp5yvv222x7te9asyzuhmjym083lwqpp5vlem09ewfeufyrp6f76w2sdql2djkuepqw3hjqnz5gvsxzerywfjhxuccqzpgsp5u6kxgf9daf64ptvl2ht74m6duc2neywx3ecvwxs07vf7egw5dy5s9qy9qsqakms0e7ww46q9cq2fa2ymrcx6nucfknjalkm5w4ywvjpfxdp5ya82drvdvxhqzzt2ysysh5u7rellzjse37fng3vsqafuwwz3kv4ykcquy8k29"
}

Response:

{
  "acceptZeroConf": false,
  "expectedAmount": 1359564,
  "bip21": "bitcoin:QNaGS7WM31xANXQCbmrhXfnxUjxiGFpFwM?amount=0.01359564&label=Submarine%20Swap%20to%20BTC"
}

If bitcoin were sent already:

POST /setinvoice

Request body:

{
  "id": "UwHIPg",
  "invoice": "lnbcrt1m1p08epqjpp5yvv222x7te9asyzuhmjym083lwqpp5vlem09ewfeufyrp6f76w2sdql2djkuepqw3hjqnz5gvsxzerywfjhxuccqzpgsp5u6kxgf9daf64ptvl2ht74m6duc2neywx3ecvwxs07vf7egw5dy5s9qy9qsqakms0e7ww46q9cq2fa2ymrcx6nucfknjalkm5w4ywvjpfxdp5ya82drvdvxhqzzt2ysysh5u7rellzjse37fng3vsqafuwwz3kv4ykcquy8k29"
}

Response:

{}

If the invoice was previously successfully set and Boltz tried to pay it already:

POST /setinvoice

Request body:

{
  "id": "UwHIPg",
  "invoice": "lnbcrt100u1p0gv8hjpp5j0vs0te6wykahrp3aammm46m73n2afzk6a87ezfp3p58qpcpu4wqdqqcqzpgsp5j6wue634lac577xnupy8auvq7n9062vshvvc6xszq4jt5q9phhzq9qy9qsqhs7zrs98tu669xz7w0gqy96g5pvs9p6lssmyseg7a92kpjlzramk8khyzkd8x4nl2zasekmwt45z6pe78rk032lkmshjdnesw2vukwgqtglt89"
}

Response:

{
  "error": "lightning payment in progress already",
  "invoice": "lnbcrt1m1p08epqjpp5yvv222x7te9asyzuhmjym083lwqpp5vlem09ewfeufyrp6f76w2sdql2djkuepqw3hjqnz5gvsxzerywfjhxuccqzpgsp5u6kxgf9daf64ptvl2ht74m6duc2neywx3ecvwxs07vf7egw5dy5s9qy9qsqakms0e7ww46q9cq2fa2ymrcx6nucfknjalkm5w4ywvjpfxdp5ya82drvdvxhqzzt2ysysh5u7rellzjse37fng3vsqafuwwz3kv4ykcquy8k29"
}

Creating Reverse Submarine Swaps

This section walks you creating Reverse Submarine Swaps (Lightning -> Chain). Similar to Normal Submarine Swaps, requests and responses change slightly depending on the kind of bitcoin involved in the swap. Keep in mind, Boltz uses 10 ** -8 as denomination for responses on the API.

All requests to create Reverse Submarine Swaps have the following common values in the API request:

  • type: Type of the swap to create. For Reverse Submarine Swaps this is reversesubmarine.

  • pairId: The pair of which the swap should be created, query available pairs via /getpairs.

  • orderSide: Possible values are buy & sell. Currently, we recommend using buy across all pairs of swap type reversesubmarine. The value sell for reverse swaps of e.g. the L-BTC/BTC pair signifies a swap from Bitcoin mainchain to Liquid Lightning. Currently, this is not supported and the backend will return "error": "L-BTC has no lightning support"

  • preimageHash: The SHA256 hash of a preimage that was generated by the client. The size of that preimage has to be 32 bytes, otherwise claiming will fail.

There are two options how to set the amount of a reverse swap. The first option is to specify the amount of the invoice that Boltz will generate:

  • invoiceAmount: amount of the invoice that will be generated by Boltz

The second option is to specify the amount that will be locked in the chain HTLC. That amount is not what the user will finally receive because of transaction fees required to claim the HTLC. But those can be approximated easily in advance with third party tools like mempool.space.

  • onchainAmount: amount Boltz will lock in the chain HTLC

We recommend verifying that pair data fetched previously (like minerFees) is still valid by additionally passing the pairHash argument in this call.

  • pairHash: hash string in the pair object of /getpairs.

Members of our partner program may set this optional referral parameter to get a percentage of the fees earned from referred swaps as kickback.

  • referralId: Partner referral ID.

URLResponse

POST /createswap

JSON object

Status Codes:

  • 201 Created

  • 400 Bad Request: The swap could not be created. Check the error string in the JSON object of the body of the response for more information.

Response objects:

  • id: Id of the newly created swap.

  • lockupAddress: Address derived from the redeemScript or contract in which Boltz will lock up bitcoin.

  • invoice: Hold invoice that needs to be paid before Boltz locks up bitcoin.

  • timeoutBlockHeight: Base asset block height at which the swap will expire.

In case the invoice amount was specified, the amount that will be locked in the chain HTLC is also returned:

  • onchainAmount: Amount of chain bitcoin that will be locked by Boltz.

Boltz Backend finally features the so-called Prepay Miner Fee protocol that requires an invoice for network fees to be paid before the actual hold invoice of a Reverse Submarine Swap. If this protocol is enabled, the response object will also contain a minerFeeInvoice. Once the minerFeeInvoice is paid, Boltz will send the event minerfee.paid and when the actual hold invoice is paid, the chain bitcoin will be sent.

This protocol is a countermeasure against a specific attack vector and is currently _not_ enabled on Boltz Mainnet.

Reverse Swaps: UTXO Chains

For UTXO chains, /createswap requests have to contain one additional parameter:

  • claimPublicKey: Public key of a keypair that will allow the user to claim the locked up bitcoin with the preimage. This keypair has to be generated and stored by the Boltz API client.

Responses also contain one additional value:

  • redeemScript: Redeem script from which the lockup address is derived. The redeem script should be used, to verify that Boltz didn't try to cheat by creating an address without a HTLC.

In case the lockup address is on the Liquid Network, it will be blinded by a key that is returned in the response too:

  • blindingKey: Hex encoded private key with which the address was blinded.

Examples:

POST /createswap

Request body:

{
  "type": "reversesubmarine",
  "pairId": "L-BTC/BTC",
  "orderSide": "buy",
  "invoiceAmount": 1000000,
  "preimageHash": "51a05b15e66ecd12bf6b1b62a678e63add0185bc5f41d2cd013611f7a4b6703f",
  "claimPublicKey": "03b76c1fe14bab50e52a026f35287fda75b9304bcf311ee85b4d32482400a436f5"
}

Response:

{
  "id": "v3CfMa",
  "invoice": "lntb10m1pjvsy8ppp52xs9k90xdmx390mtrd32v78x8twsrpdutaqa9ngpxcgl0f9kwqlsdpz2djkuepqw3hjqnpdgf2yxgrpv3j8yetnwvcqz95xqrrsssp5hzcjq972f9cl8c6u3zechepm65hjceaqvlzye6kc23qkz5rhva6q9qyyssq4kkw9fwjq7n9cm4j3ajj2a92ka0zyeg3sxppfy932c62pnsqkw7nhvg9rrxztszw37wqtal4cchw2f4s09qe48pngsl9euv7wjlz93qqrfx60s",
  "blindingKey": "897034f717beb12a3c2b7ae8c08c5c4def7bc7cfb6efa3713c617f28d90d1419",
  "redeemScript": "8201208763a914be1abd8e8d7ef7e64a9c6e1e2f498f3a92e078a2882103b76c1fe14bab50e52a026f35287fda75b9304bcf311ee85b4d32482400a436f5677503dbf40eb175210330fd4cfd53b5c20886415c1b67d2daa87bce2761b9be009e9d1f9eec4419ba5968ac",
  "lockupAddress": "tlq1qqdd0v79wcqnpvujf5mfp88d5cz8rynk0awr6m84ca8pzn39kwagsh4z204s9d5sww3cxckd47wjxlqwl3u6tgdqfa877txqt9m8wgk22qwyp5yzxaf40",
  "timeoutBlockHeight": 980187,
  "onchainAmount": 995724
}

In case the Prepay Miner Fee protocol is enabled:

Request body:

{
  "type": "reversesubmarine",
  "pairId": "BTC/BTC",
  "orderSide": "buy",
  "claimPublicKey": "0391fbaf549578fd7c2cb26b216441825bd780d85dba1f3d706e2f206587e96266",
  "invoiceAmount": 100000,
  "preimageHash": "2215034def003b63b2717fccd4ce8259f4807a39318c14e2bdd42639ca989a45"
}

Response body:

{
  "id": "mVSKyF",
  "invoice": "lnbcrt996940n1p0dhjr3pp5yg2sxn00qqak8vn30lxdfn5zt86gq73exxxpfc4a6snrnj5cnfzsdql2djkuepqw3hjqsj5gvsxzerywfjhxuccqzy0sp5hjcvwl2glrq9n3vzm8072cdruz3hhz70edml8g0u76gryve6np4q9qy9qsqzw5w8ulxjgrg478hz4enjrw0a9tedl8s3n879xqh3mhn0pxrvajrz9qnnsr58twx4a30gk57d4fykm7x3v2vcamw7k4ny9fkpwl65vcpw8v5em",
  "redeemScript": "8201208763a914fb75ff5dc4272c2da33d744615905f54b62de41588210391fbaf549578fd7c2cb26b216441825bd780d85dba1f3d706e2f206587e962666775028e01b1752102c22801bd7dd3a6afb780671c1c983fcd91fa46826eadd82e325e7e13bb348a9768ac",
  "lockupAddress": "bcrt1qweryu6nk8gn5lj8ar5kjdy476wynheszg0lumu6jx83l2v6f435stlel03",
  "onchainAmount": 98694,
  "timeoutBlockHeight": 398,
  "minerFeeInvoice": "lnbcrt3060n1p0dhjr3pp5sk2u4rt0z8rrl6jj62d6szqvsdejj8kjcxa8tdt4dau5rtyskj6qdp4f45kuetjypnx2efqvehhygznwashqgr5dusyy4zrypskgerjv4ehxcqzpgsp5qtsm5vfy9yq8kjpthla67jagmcxnj529pm3edk94npf6fekq2sxq9qy9qsqmun0z8ed4kp9dhp7lthvzdrx3ngmjs32smx6l4hvyyktv92mf348aftgrwf44sl94ewywr3sw8dc4acy63yamxxpjtd4pkkr2uw2h5gpqc3d3y"
}

Reverse Swaps: EVM Chains

Currently, Boltz only supports RSK Testnet.

For EVM chains, /createswap requests have to contain one additional parameter:

  • claimAddress: address from which the bitcoin will be claimed

The response also has one more property:

  • refundAddress: the address of Boltz which is specified as refund address when it is locking up funds

Boltz features an optional "gasless" protocol that allows a user to pay an additional lightning invoice to pay for gas on EVM chains like RSK to claim a reverse swap. This is useful for users who do not not have e.g. RBTC on RSK yet. In the gasless protocol, using the example of RSK, Boltz sends just enough RBTC to the claimAddress in the swap process for the user to successfully claim the swap. To use this protocol, set the following property in the request body to true.

  • gasless: If the gasless protocol should be used for the reverse swap.

When the gasless protocol is used, the response will contain two more values. One is the amount of RBTC that will be sent to the claimAddress. The other one is an invoice to pay for the sent gasAssetAmount.

  • gasAssetAmount: Amount of e.g. RBTC that will be sent to the claimAddress to be used as gas to claim the swap.

  • gaslessInvoice: Invoice that pays for gasAssetAmount.

Only when both invoices (gaslessInvoice and invoice) are paid, Boltz will lock the chain bitcoin to proceed with the swap.

Examples:

POST /createswap

Request body:

{
  "type": "reversesubmarine",
  "pairId": "RBTC/BTC",
  "orderSide": "buy",
  "claimAddress": "0x88532974EC20559608681A53F4Ac8C34dd5e2804",
  "invoiceAmount": 100000,
  "preimageHash": "295b93a766959d607861ab7b7a6bf9e178e7c69c3cc4ca715065dfe9d6eea352"
}

Response body:

{
  "id": "NPT9VE",
  "invoice": "lntb1m1pj5g2sssp5p4nn4jk4w7p208c22eytkvclleghsfeug4cu2u9qpaqnflrtp3yqpp599de8fmxjkwkq7rp4dah56leu9uw035u8nzv5u2svh07n4hw5dfqdpq2djkuepqw3hjq5jz23pjqctyv3ex2umnxqyp2xqcqz959qxpqysgqca37dvrmpy294383zsszeqyny0fdqtkr8tllnvvj0g2w65lgcwaszah5kcs4hej9dsm9c3tj43tklnc0y6fqy5964h4xz0rylz4nqugpz0y90y",
  "refundAddress": "0x4217BD283e9Dc9A2cE3d5D20fAE34AA0902C28db",
  "lockupAddress": "0x165F8E654b3Fe310A854805323718D51977ad95F",
  "timeoutBlockHeight": 4454445,
  "onchainAmount": 99195
}

Swap Status

Before handling status events of this method, we recommended to read: [Swap Types & States](lifecycle.md)

To query the status of a swap one can use this endpoint which returns a JSON object containing the status of the swap. Possible states and status events are documented in the section Swap Types & States.

Requests querying the status of a swap have to be POST and contain a single value in its JSON encoded body:

  • id: Id of the swap of which the status is queried.

URLResponse

POST /swapstatus

JSON object

Status Codes:

  • 200 OK

  • 404 Not Found: The swap with the provided id couldn't be found.

  • 400 Bad Request: The id argument wasn't provided.

Response object:

  • status: Status of the swap, e.g. transaction.mempool & transaction.claimed for successful Normal Submarine Swaps and transaction.mempool and transaction.confirmed for successful Reverse Submarine Swaps.

  • transaction: For Reverse Submarine Swaps, this field contains lockup transaction details in the statestransaction.mempool and transaction.confirmed:

    • id: Id of the lockup transaction.

    • hex: Hex encoded lockup transaction (only set for transactions on UTXO chains).

    • eta: If the status is transaction.mempool, this value is the estimated time of arrival (ETA) in blocks of when the transaction will be confirmed. Only set for transactions on UTXO chains.

  • zeroConfRejected: Set to true for Swaps with the status transaction.mempool and a lockup transaction that is not eligible for 0-conf.

  • failureReason: Set when it's necessary to further clarify the failure reason.

Examples:

POST /swapstatus

Request body:

{
  "id": "Asnj2Y"
}

Response:

{
  "status": "invoice.paid"
}

POST /swapstatus

Request body:

{
  "id": "ryUK9G"
}

Response:

{
  "status": "transaction.mempool",
  "transaction": {
    "id": "31fcddf287d985eef85211b75976cd903dba3008a8e13b597e1b54941278c29f",
    "hex": "01000000000101618cd5c50221577a1b98ae4a73f652917f9d2e343b9bc6a978239da78dfcbc630000000000ffffffff02b878010000000000220020ddc8dd3bcb45660e421fc3129bfdcb317446c27ce909369d4c8cb17bbd6d4951c718393b00000000160014ea9e0fc9432fc8b6831e94ac3974d46d1ba1c62f024830450221008b14ecce2eebb2ec7e56c53de4796603fbd22cfd269f3a5249446b79987dec36022059b68568a57ebfbd50cc792af3d514c507af20e795ff5f0bf2a02d2fc44be223012103a1ad2a3891018e856700a20a4dea6bea9f4bba58ab3ca7cbaefaa06e805770d100000000",
    "eta": 2
  }
}

POST /swapstatus

Request body:

{
  "id": "gnIthU"
}

Response:

{
  "status": "transaction.lockupFailed",
  "failureReason": "locked 1396075383 is less than expected 1396075384"
}

Swap Status Stream

To avoid querying the /swapstatus endpoint regularly to get the latest swap status, this endpoint streams swap status updates via Server-Side Events.

Requests to this endpoint have to provide the required swap id parameter via an URL parameter because all requests have to be of the method GET.

Every event in the Server-Side stream has data that is encoded exactly like the JSON object of the /swapstatus endpoint. Please refer to the examples below for a reference implementation in JavaScript in how to handle the stream.

URLResponse

GET /streamswapstatus

Server-Side event stream

Examples:

Server-Side event streams have to be handled differently from regular HTTP responses. Below is a sample implementation in JavaScript and also what a raw response of a Server-Side event stream looks like.

Sample implementation in JavaScript:

var stream = new EventSource(boltzApi + '/streamswapstatus?id=' + swapId);

source.onmessage = function (event) {
  const data = JSON.parse(event.data);

  console.log('Swap status update:' + data.status);
};

Raw response:

data: {"status":"transaction.mempool"}

data: {"status":"invoice.paid"}

Swap Timeouts

Boltz Swaps have different timeouts for each pair. This endpoint allows querying those timeouts denominated in blocks of the base and quote chain.

URLResponse

GET /timeouts

JSON object

Status Codes:

  • 200 OK

Response object:

  • timeouts: A JSON object with the pairs as keys and a JSON object with the timeouts as values.

Examples:

GET /timeouts

Response:

{
  "timeouts": {
    "BTC/BTC": {
      "base": {
        "reverse": 144,
        "swapMinimal": 144,
        "swapMaximal": 288
      },
      "quote": {
        "reverse": 144,
        "swapMinimal": 144,
        "swapMaximal": 288
      }
    },
    "L-BTC/BTC": {
      "base": {
        "reverse": 1440,
        "swapMinimal": 1440,
        "swapMaximal": 2880
      },
      "quote": {
        "reverse": 144,
        "swapMinimal": 144,
        "swapMaximal": 288
      }
    }
  }
}

Swap Contracts

Currently, Boltz only supports RSK Testnet.

To query the addresses of contracts used by Boltz for swaps on EVM chains like RSK, the following endpoint can be queried:

URLResponse

GET /getcontracts

JSON object

Status Codes:

  • 200 OK

Response object:

  • chain: JSON object that contains all relevant contract addresses of this EVM chain.

    • network: JSON object that contains information about the network.

      • chainId: Id of the EVM chain.

    • swapContracts: JSON object containing swap contract addresses as values.

    • tokens: JSON object with the ticker symbol of the supported token as key and its address as value.

Examples:

GET /getcontracts

Response:

{
  "rsk": {
    "network": {
      "chainId": 31
    },
    "swapContracts": {
      "EtherSwap": "0x165F8E654b3Fe310A854805323718D51977ad95F",
      "ERC20Swap": "0x5F51247606d29Df75Af11475A85F7072f6472345"
    },
    "tokens": {}
  }
}

Fee Estimations

Boltz provides an API endpoint that returns fee estimations for all supported chains. These fee estimations are not enforced by Boltz and merely represent a recommendation.

For UTXO chains like Bitcoin it is important to mention that if 0-conf is accepted by Boltz for a particular pair and to be used with Normal Submarine Swaps, the lockup transaction has to have at least 80% of the recommended sat/vbyte value. For more information refer to the 0-conf section.

URLResponse

GET /getfeeestimation

JSON object

Status Codes:

  • 200 OK

Response object:

This endpoint returns a JSON object of which each key is the symbol of a chain and each value the estimated fee for that chain denominated in sat/vbyte for UTXO chains like Bitcoin or GWEI for EVM chains like RSK.

Examples:

GET /getfeeestimation

Response:

{
  "BTC": 16,
  "L-BTC": 0.11
}

Raw Transactions

Boltz API also allows for querying raw transactions of all supported UTXO chains, irrespective of whether the transactions are still in the mempool or already included in a block. Note, that Boltz does not provide any kind of cryptographic proof that the transaction was included in a block. Also this call is primarily kept for backward compatibility with older integrations, it is not needed to construct transactions as the response of /swapstatus provides all necessary info.

Requests querying for transactions have to be POST and contain two arguments in its JSON encoded body:

  • currency: The chain to be queried for the transaction.

  • transactionId: The id of the transaction that should be queried.

URLResponse

POST /gettransaction

JSON object

Status Codes:

  • 200 OK

  • 400 Bad Request: An argument wasn't provided or the transaction couldn't be found.

Response object:

  • transactionHex: The requested transaction encoded in hex.

Examples:

POST /gettransaction

Request body:

{
  "currency": "BTC",
  "transactionId": "0e3e2357e806b6cdb1f70b54c3a3a17b6714ee1f0e68bebb44a74b1efd512098"
}

Response:

{
  "transactionHex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0704ffff001d0104ffffffff0100f2052a0100000043410496b538e853519c726a2c91e61ec11600ae1390813a627c66fb8be7947be63c52da7589379515d4e0a604f8141781e62294721166bf621e73a82cbf2342c858eeac00000000"
}

Lockup Transactions

The following endpoint can be used to query the user's lockup transaction of a Normal Submarine Swap on UTXO chains. The request has to be POST and contain the following argument in the JSON encoded body:

  • id: Id of the Submarine Swap.

URLResponse

POST /getswaptransaction

JSON object

Status Codes:

  • 200 OK

  • 400 Bad Request: An argument wasn't provided, or the swap couldn't be found.

Response object:

  • transactionHex: The lockup transaction of the Normal Submarine Swap encoded in hex.

  • timeoutBlockHeight: The block height at which the HTLC in the lockup transaction will time out.

If the HTLC has not timed out yet, there will be an additional value in the response:

  • timeoutEta: UNIX timestamp at which the HTLC is expected to time out.

Examples:

POST /getswaptransaction

Request body:

{
  "id": "KR8XaB"
}

Response:

{
  "transactionHex": "020000000001015bf3fe03071edae971276831963d45821ce6bc95c567fd7832bee3b1848254ba0000000000feffffff02b82a75e80000000017a914ca9b26af0865c2a59f0d6c41ede68f03264a52398700e1f5050000000017a914d2271732308baf018a1c2c751a3afb197f2a2e7e870247304402200206ae10cd66267caea1c6e0fca0275924a85fa377572f599304c1abb6a3a97302204335eed108151048de7ba56fd644352831e3d453b412547d8b762f4e96ee1e310121035b6440fe45a8bf7c1d9d238fa39a128ee78b58df73377bfeb1d6d752849714c800000000",
  "timeoutBlockHeight": 252,
  "timeoutEta": 1586353245
}

This call works for Normal Submarine Swaps only. If used for Reverse Submarine Swaps, the response will be:

{
  "error": "could not find swap with id: CR8XaB"
}

Broadcasting Transactions

This endpoint is used to broadcast transactions on UTXO chains. It is similar to /gettransaction but instead of getting the hex representation of existing transactions, this call broadcasts new transactions to the network. It is mainly intended to be used as an easy way to broadcast claim & refund transactions by Boltz API clients that don't have access to a full node. We encourage checking out alternatives like mempool.space's public API for Bitcoin or Liquid to reduce reliance on Boltz. The call returns the id of the broadcast transaction,which can be used to verify that the refund transaction was broadcast successfully using a third party service.

Requests broadcasting transactions have to be POST and contain two arguments in the JSON encoded body:

  • currency: Which network the transaction should be broadcast on.

  • transactionHex: The HEX encoded transaction.

URLResponse

POST /broadcasttransaction

JSON object

Status Codes:

  • 200 OK

  • 400 Bad Request: An argument wasn't provided or the node that was used to broadcast the transaction returned an error.

Response object:

  • transactionId: The id of the transaction that was broadcast.

Example:

POST /broadcasttransaction

Request body:

{
  "currency": "BTC",
  "transactionHex": "02000000000101f7ddc247aad2c1e91b2495cf6814aa183b46785cf21f44f43a3c3419c09d377201000000171600142a805e4cfaa6fb79360917a5b0b9c5fcb0dfe6e9ffffffff02938601000000000017a9141137c50104d0814e8663ded75b43ddaa7b9d192b87e8cc96000000000017a9148f6df517d00e650d8c5e6bfa0986b775b256609e870247304402202e8a8572ce3cb232a7b48483bdc280feba7f9cf8c163ac2df091e54dfcf90bb6022042ba2b3e4d89220b3b39b52a444dff2010b22144099aaa348d83301f2ae456be01210341720559e7375a409b03e814415a6c15fc142c5a9e78a83831ff6fe4706d352900000000"
}

Response:

{
  "transactionId": "52ff6682b0bff109e6c6d97de6b6d075f7241c9ac364e02de6315281e423d816"
}

There is one special case: when trying to broadcast a refund transaction for a swap that has not timed out yet, the backend will return some additional information in addition to the error in the JSON encoded response:

  • error: The reason why broadcasting failed. In this special case always: non-mandatory-script-verify-flag (Locktime requirement not satisfied) (code 64).

  • timeoutEta: UNIX timestamp at which the HTLC is expected to time out.

  • timeoutBlockHeight: Block height at which the HTLC in the lockup transaction will time out.

Example:

POST /broadcasttransaction

Request body:

{
  "currency": "BTC",
  "transactionHex": "0100000000010113a19f721bc15b17a63700a7bf0056b7640b2c80239577f7ee95618baa34958f0100000023220020e1088f54c6d46e861bf4c5590bdff35c6f277b10e4e2787e1a5df23f1e540a3dfdffffff01bcdff5050000000016001486590ca595782212ce4fe0a4be8855f55f7f288603483045022100c142b9616659ba1728c254d8e275a304dc31c9139f005f9a97938cb1606c370e0220686d4e6166a7adab916f9af823c2f173aa9bd7f47a581909bda95881e1c00e07010064a9146aad1375552e58e9d4281a331caf271d0d160e3c8763210396ed47336687c51bc7e2bd32d0fc7a377d33c888f02a0647a7f1156761614a0d6702c401b1752103ffb18860cbe08060bd93a17abe4b436c46d0ee5b43fd0c24ba5bd65d6f42beb568ac00000000"
}

Response:

{
  "timeoutEta": 1586374147,
  "timeoutBlockHeight": 452,
  "error": "non-mandatory-script-verify-flag (Locktime requirement not satisfied) (code 64)"
}

Authentication

Boltz API does not require any sort of authentication to perform swaps. However, some API endpoints like querying referral fees for members of our partner program, do.

To authenticate your API request, three request headers have to be set:

  • TS: current UNIX timestamp (can only deviate from server time by 1 minute at most)

  • API-KEY: your API key

  • API-HMAC: SHA256 HMAC encoded as HEX (lower case letters!) of the following values:

    • value of the TS header

    • method of the HTTP request (e.g. GET or POST)

    • request path, including the leading slash (e.g. /referrals/query)

    • if the request method is POST, the body of the request

TypeScript Node.js example:

import axios from 'axios';
import { createHmac } from 'crypto';

const path = '/referrals/query';

const ts = Math.round(new Date().getTime() / 1000);
const hmac = createHmac('sha256', argv.secret)
  .update(`${ts}GET${path}`)
  .digest('hex');

try {
  const res = await axios.get(
    `https://${argv.rest.host}:${argv.rest.port}${path}`,
    {
      headers: {
        TS: ts,
        'API-KEY': argv.key,
        'API-HMAC': hmac,
      },
    },
  );

  console.log(JSON.stringify(res.data, undefined, 2));
} catch (e) {
  const error = e as any;
  console.log(`${error.message}: ${JSON.stringify(error.response.data)}`);
}

Querying Referral Fees

Members of the Boltz partner program can request a referral key (hi@bol.tz) to get a percentage of the fees earned from referred swaps as kickback. To query for their referrals, they can send an authenticated request to this endpoint.

URLResponse

GET /referrals/query

JSON object

Status Codes:

  • 200 OK

  • 401 Unauthorized: Missing or invalid request authentication.

Response object:

The response of a valid request is grouped by year, month and referral key.

Examples:

GET /referrals/query

Response:

{
  "2021": {
    "9": {
      "cliTest": {
        "BTC": 60
      }
    }
  }
}

Lightning Node Info

This endpoint allows you to query info like public keys and URIs of the lightning nodes operated by Boltz.

URLResponse

GET /getnodes

JSON object

Status Codes:

  • 200 OK

Response object:

  • nodes: JSON with the symbol of the chain on which the Lightning node is running as key and the following objects:

    • nodeKey: Public key of the Lightning node.

    • uris: Array of the URIs on which the Lightning node is reachable.

Examples:

GET /getnodes

Response:

{
  "nodes": {
    "BTC": {
      "uris": [
        "026165850492521f4ac8abd9bd8088123446d126f648ca35e60f88177dc149ceb2@45.86.229.190:9735",
        "026165850492521f4ac8abd9bd8088123446d126f648ca35e60f88177dc149ceb2@d7kak4gpnbamm3b4ufq54aatgm3alhx3jwmu6kyy2bgjaauinkipz3id.onion:9735"
      ],
      "nodeKey": "026165850492521f4ac8abd9bd8088123446d126f648ca35e60f88177dc149ceb2"
    }
  }
}

Lightning Node Statistics

For display purposes on our website, basic statistics about our lightning nodes are exposed via the following endpoint:

URLResponse

GET /nodestats

JSON object

Status Codes:

  • 200 OK

Response object:

  • nodes: JSON with the symbol of the chain on which the Lightning node is running as key, and the following objects:

    • peers: Number of peers.

    • channels: Number of public channels.

    • oldestChannel: UNIX timestamp of the block in which the opening transaction of the oldest channel was included.

    • capacity: Sum of the capacity of all public channels.

Examples:

GET /nodestats

Response:

{
  "nodes": {
    "BTC": {
      "peers": 79,
      "channels": 103,
      "oldestChannel": 1590772669,
      "capacity": 369879555
    }
  }
}

Last updated