Trade Programmatically
Reservoir's aggregated liquidity and easy-to-use APIs make it a useful tool for market makers in the NFT ecosystem who want to quickly and effectively provide liquidity across NFT markets. Reservoir provides the tools to understand the market, create robust trading strategies, and create and publish orders. Market makers can publish orders to the Reservoir ecosystem and other major NFT marketplaces using Reservoir's tooling.
Reservoir has an extremely comprehensive set of real-time and historical data for NFT collections, sales, and liquidity. You can see some of that data in use here: https://dune.com/reservoir0x/nft-marketplace-overview This data is particularly useful for creating models of NFT prices, and building robust market making strategies. All this data and more is available using our Data APIs. If you need more comprehensive data access, feel free to reach out.
Data
-
All historical sales
maker, taker, mp, price (Eth and USD), fees -
Most historical listings
-
Maker, price (Eth and USD), expiration, final state (executed expired)
-
Token and collection metadata
-
Token attributes: i.e. token #123 has ‘blue hair’
-
Collection attributes: i.e. total supply, holders, etc.
-
Real time floor price data including TWAPs
-
Some, but not all current bids
Reservoir SDK
Reservoir SDK allows you to interact with major NFT marketplaces easily. In order to showcase this SDK we walk you through an example of market making activity below.
Feel free to check out the full example code here.
Installing and Configuring the Reservoir SDK
> yarn add @reservoir0x/reservoir-sdk
The Reservoir SDK also requires ethers to be installed, make sure that's added to your package.json if it isn't already.
> yarn add ethers
To configure the SDK we first need to create a global instance of it:
import { createClient } from "@reservoir0x/reservoir-sdk"
const client = createClient({
apiBase: "https://api.reservoir.tools",
apiKey: "YOUR_API_KEY",
source: "YOUR.SOURCE"
});
export default client
More details about the Reservoir SDK are available here.
Bidding and Listing on Different Marketplaces
We are going to assume here that you have the price that you want to bid or list a token already computed. Where data can be useful for this calculation, see the Data APIs for real time data.
Bidding
In order to bid you must first set the bid order params. For a collection bid, you simply set the collection as follows:
Collection Bid Example
let bidPrice = x; //example bid price
let expirationTime = dayjs().add(20, "minutes").unix().toString();
let bidOrder = {
collection: contract,
orderKind: "seaport",
orderbook: "reservoir",
automatedRoyalties: true,
excludeFlaggedTokens: true,
weiPrice: ethers.utils.parseEther(bidPrice).toString(),
expirationTime,
};
To make a token bid, In order to bid on a single token, you pass token: tokenId into the params in addition to the collection contract.
Token Bid Example
let bidPrice = x; //example bid price
let expirationTime = dayjs().add(20, "minutes").unix().toString();
let bidOrder = {
collection: contract,
token: tokenId,
orderKind: "seaport",
orderbook: "reservoir",
automatedRoyalties: true,
excludeFlaggedTokens: true,
weiPrice: ethers.utils.parseEther(bidPrice).toString(),
expirationTime,
};
You can also place trait and arbitrary token set bids, depending on the orderbook you choose.
Here orderKind
and orderbook
set the marketplace you wish to interact with:
Marketplace | orderKind | orderbook | Marketplace Fee | Available Features |
---|---|---|---|---|
OpenSea | seaport | opensea | 2.5% | Collection, token bids |
LooksRare | looks-rare | looks-rare | 2% | Collection, token bids |
X2Y2 | x2y2 | x2y2 | 0.5% | Collection, token bids |
Reservoir | seaport | reservoir | 0% | Collection, trait, token arbitrary token set bids |
Then, you can simply use the the Reservoir SDK for bidding:
await reservoirClient.actions
.placeBid({
signer: signer,
onProgress: () => {},
bids: [bidOrder],
})
The place bid function takes an array of bid objects that match the params passed to the execute bid api
Listing
In order to list you must first set the listing params:
Example System Set-up
let listPrice = x //example list price
let expirationTime = dayjs().add(20, "minutes").unix().toString();
let listOrder = {
token: `${contract}:${token}`,
orderKind: "seaport",
orderbook: "reservoir",
weiPrice: ethers.utils.parseEther(listPrice).toString(),
expirationTime: listingTime,
};
await reservoirClient.actions
.listToken({
signer: signer,
onProgress: () => {},
listings: [listOrder],
})
Example Server-side Setup
To submit list and bid orders server-side using OpenZeppelin, you must first configure an OpenZeppelin defender relayer. Once this is done, sending transactions through your relay is as simple as creating a custom signer as follows:
const { DefenderRelaySigner, DefenderRelayProvider } = require('defender-relay-client/lib/ethers');
const { ethers } = require('ethers');
const credentials = { apiKey: YOUR_API_KEY, apiSecret: YOUR_API_SECRET };
const provider = new DefenderRelayProvider(credentials);
const signer = new DefenderRelaySigner(credentials, provider, { speed: 'fast' });
One could set up a programmatic trading system in a number of ways. But one simple set up is as follows:
- Design a trading strategy using Reservoir’s APIs to get all the data needed to set buys and sells or bids and asks appropriately
- Use Reservoir’s abstracted SDK for creating orders
- For on-chain actions use Ethers.js and connect to an OpenZeppelin defender relayer and Reservoir SDK takes care of everything that needs to happen behind the scenes (including approvals, making safety checks, etc.)
- Deploy a node server to execute a cron job, loop the job over a collections list and executes a market making strategy above
Updated almost 2 years ago