This document is supposed to help retrieving indexed & tracked TAP token data for further processing. Also make sure to read the TAP protocol specs: https://github.com/BennyTheDev/tap-protocol-specs.
Please note that this confirmed data and not mempool data. If you intend to create a marketplace, please utilize the Bitcoin mempool to make frontrunning techniques less likely, especially when it comes to token transfers.
This only holds true for TAP's external functions like "token-deploy", "token-mint" and "token-transfer" as they derive directly from BRC-20 and are a clone of that standard. Internal functions like "token-send" are mempool-safe and do not require mempool observation.
After public release, you may want to self-host Trac. In this case, the described endpoint below will need to get changed into your own endpoint location.
The examples below use websockets instead of classic RESTful API endpoints as these usually serve large amounts of data better. With the release of Trac, RESTful APIs will be available but should only be used for smaller sets of data.
- Some Javascript knowledge (Node or Browser)
- Socket.io 4.6.1
HTML/JS
<script src="https://cdnjs.cloudflare.com/ajax/libs/socket.io/4.6.1/socket.io.min.js" integrity="sha512-AI5A3zIoeRSEEX9z3Vyir8NqSMC1pY7r5h2cE+9J6FLsoEmSSGLFaqMQw8SWvoONXogkfFrkQiJfLeHLz3+HOg==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
Node
npm install socket.io-client
Code Anatomy
// node only!
const { io } = require("socket.io-client");
// connect to public TAP endpoint with Trac
const trac = io("https://tap.trac.network", {
autoConnect : true,
reconnection: true,
reconnectionDelay: 500,
reconnectionDelayMax : 500,
randomizationFactor : 0
});
trac.connect();
// default response event for all endpoint calls.
// this event handles all incoming results for requests performed using "emit".
//
// You may use the response to render client updates or store in your local database cache.
//
// Example response object:
// {
// "error": "",
// "func": ORIGINALLY-CALLED-FUNCTION-NAME,
// "args": ARRAY-WITH-ARGUMENTS,
// "call_id": OPTIONAL-CUSTOM-VALUE,
// "result": RETURNED-MIXED-TYPE-VALUE
// }
trac.on('response', async function(msg){
console.log(msg);
});
// default error event for internal Trac errors, if any
trac.on('error', async function(msg){
console.log(msg);
});
// example getter to get a deployed token.
// the results for this call will be triggered by the 'response' event above.
// this structure is exactly the same for all available getters of this endpoint.
trac.emit('get',
{
func : 'deployment', // the endpoints function to call
args : ['tap'], // the arguments for the function (in this case only 1 argument)
call_id : '' // a mixed-type custom id that is passed through in the 'response' event above to identify for which call the response has been and how to proceed.
});
You may want to do to things like retrieving list lengths + list items in one operation. Since the endpoint is using a streaming client and strategy, it is recommended to utilize chained getter calls.
Get account tokens example:
async function tokens(address)
{
trac.emit('get',
{
func : 'accountTokensLength',
args : [address],
call_id : { cmd : 'tokens_accountTokens', address : address }
});
}
async function getAccountTokens(address, offset = 0, max = 500, call_id = null)
{
trac.emit('get',
{
func : 'accountTokens',
args : [address, offset, max],
call_id : call_id === null ? '' : call_id
});
}
tokens(address);
Processing the response of above calls:
trac.on('response', async function(msg){
if(typeof msg.call_id === 'object')
{
switch(msg.call_id.cmd)
{
case 'tokens_accountTokens':
getAccountTokens(
msg.call_id.address,
0,
500,
{ cmd : 'tokens_end', address : msg.call_id.address }
);
break;
case 'tokens_end':
console.log(msg);
// render results...
break;
}
}
)};
As you can see, 'tokens_accountTokens' being passed as custom object in the call_id, is sent with the request. In the response event, the exact call_id will be passed through and you may use the results to trigger the next call in the chain until you received all desired data sets.
The results for the calls of the below getters should be used in the "response" event above, explained in Code Anatomy.
Parallel calls are encouraged. Especially if large amounts of data are requested, it is recommended to request them in parallel and smaller chunks.
/**
* Expects a ticker string argument.
* Returns a deployment object, containing the deployed token information.
* Returns null if the ticker doesn't exist.
*
* Example returned object:
*
*{
* tick: 'bitmap',
* max: '21000000000000000000000000',
* lim: '1000000000000000000000',
* dec: 18,
* blck: 802027,
* tx: '1e69ce0d3b46a0e2556b01afc87cd596be89c341dfc3e50cd81580a522a77d4a',
* ins: '1e69ce0d3b46a0e2556b01afc87cd596be89c341dfc3e50cd81580a522a77d4ai0',
* num: 22023299,
* ts: 1691376381,
* addr: 'bc1p6wynl030xsvr48r6kgw99jdtll7jd9n3e8f60j04hmgnrwwysruqwrf9pm',
* crsd: false
*}
*/
trac.emit('get',
{
func : 'deployment',
args : [ticker],
call_id : ''
});
/**
* Returns the amount of deployed tokens.
*/
trac.emit('get',
{
func : 'deploymentsLength',
args : [],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 deployment objects, containing the deployed token information.
*/
trac.emit('get',
{
func : 'deployments',
args : [offset, max],
call_id : ''
});
/**
* Expects a ticker string argument.
* Returns a string representing the amount of minted tokens in a Big Integer format.
* Returns null if the ticker doesn't exist.
*
* In general, all token balances, transferables, limits and token amounts for this endpoint return in the said format.
*
* You may use JS' BigInt() function to perform calculations.
*
* To render the tokens left in a readable format, you need to retrieve the "dec" (=decimals) attribute from a deployment object (see above).
* Then pass the decimal points to a function that renders a readable format.
*
* Example function:
*
* function formatBigIntString(string, decimals) {
*
* let pos = string.length - decimals;
*
* if(decimals == 0) {
* // nothing
* }else
* if(pos > 0){
* string = string.substring(0, pos) + "." + string.substring(pos, string.length);
* }else{
* string = '0.' + ( "0".repeat( decimals - string.length ) ) + string;
* }
*
* return string;
* }
*
*/
trac.emit('get',
{
func : 'mintTokensLeft',
args : [ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the address balance for that ticker as string in Big Integer format.
* Returns null if there are no balances yet for the address/ticker combination.
*
*/
trac.emit('get',
{
func : 'balance',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the address transferable amount for that ticker as string in Big Integer format.
* Returns null if there are no balances yet for the address/ticker combination.
*
* Note: there is no "available" gettter. Please calculate the available amounts like this:
*
* available = balance - transferable
*
*/
trac.emit('get',
{
func : 'transferable',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string ticker.
* Returns the amount of holders for the given ticker.
*
* Note: this getter will also include holders for the given ticker, that once owned but no longer hold.
*/
trac.emit('get',
{
func : 'holdersLength',
args : [ticker],
call_id : ''
});
/**
* Expects a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 holder addresses per call.
*
* Note: this getter will also return holders that had but now own 0 tokens of the given ticker.
*/
trac.emit('get',
{
func : 'holders',
args : [ticker, offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address.
* Returns the amount of tokens owned by the address.
*
* Note: this getter will also include tokens the address no longer owns.
*/
trac.emit('get',
{
func : 'accountTokensLength',
args : [ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 string tickers.
*
* Note: this getter will also return tickers the address once owned but no longer holds.
*/
trac.emit('get',
{
func : 'accountTokens',
args : [address, offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount of mints ever performed by the address.
*
* Note: the result includes mints that failed.
*/
trac.emit('get',
{
func : 'accountMintListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 mint objects.
*
* Note: the result includes mints that failed.
* you can use this list to render an account mint history.
*
* Example mint object:
*
* {
* addr: 'bc1pcjy9dud7hmtzyfd3hrulznxfjrcx6m0kmps36dgp8tut7mvddulqdrxzyq',
* blck: 802098,
* amt: '0',
* bal: '0',
* tx: '8d8ce31acac3ed85eb5df09bc700ddb14359042fb2015fa20beac1baf7384227',
* ins: '8d8ce31acac3ed85eb5df09bc700ddb14359042fb2015fa20beac1baf7384227i0',
* num: -77585,
* ts: 1691413973,
* fail: true,
* dmtblck: null
* }
*
* NOTE: dmtblock will be null for regular token-mint functions and populated with a blocknumber as integer for dmt-mint functions.
*/
trac.emit('get',
{
func : 'accountMintList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Expects a string ticker.
* Returns the amount of mints ever performed for the given ticker.
*
* Note: the result includes mints that failed.
*/
trac.emit('get',
{
func : 'tickerMintListLength',
args : [ticker],
call_id : ''
});
/**
* Expects a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 mint objects.
*
* Note: the result includes mints that failed.
* you can use this list to render a ticker mint history.
*/
trac.emit('get',
{
func : 'tickerMintList',
args : [ticker, offset, max],
call_id : ''
});
/**
* Returns the amount of mints ever performed.
*
* Note: the result includes mints that failed.
*/
trac.emit('get',
{
func : 'mintListLength',
args : [],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 mint objects.
*
* Note: the result includes mints that failed.
* you can use this to stream the entire mint history, across all tickers and addresses.
*/
trac.emit('get',
{
func : 'mintList',
args : [offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount transfer-inscribes ever performed by the address.
*
* Note: the result includes transfers that failed.
*/
trac.emit('get',
{
func : 'accountTransferListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 transfer-inscribe objects.
*
* Note: the result includes transfers that failed.
* you can use this list to render an account transfer-inscribe history.
*
*/
trac.emit('get',
{
func : 'accountTransferList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Expects a string ticker.
* Returns the amount of inscribe-transfer inscriptions ever performed for the given ticker.
*
* Note: the result includes mints that failed.
*/
trac.emit('get',
{
func : 'tickerTransferListLength',
args : [ticker],
call_id : ''
});
/**
* Expects a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 inscribe-transfer objects.
*
* Note: the result includes transfer-inscribes that failed.
* you can use this list to render a ticker transfer-inscribe history.
*/
trac.emit('get',
{
func : 'tickerTransferList',
args : [ticker, offset, max],
call_id : ''
});
/**
* Returns the amount of mints ever performed.
*
* Note: the result includes transfer-inscribes that failed.
*/
trac.emit('get',
{
func : 'transferListLength',
args : [],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 transfer-inscribe objects.
*
* Note: the result includes transfer-inscribes that failed.
* you can use this to stream the entire transfer-inscribe history, across all tickers and addresses.
*/
trac.emit('get',
{
func : 'transferList',
args : [offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount sends ever performed by the address.
*
* Note: the result includes sends that failed.
*/
trac.emit('get',
{
func : 'accountSentListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 send objects.
*
* Note: the result includes sends that failed.
* you can use this list to render an account send history.
*
*/
trac.emit('get',
{
func : 'accountSentList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Expects a string ticker.
* Returns the amount of sends ever performed for the given ticker.
*
* Note: the result includes sends that failed.
*/
trac.emit('get',
{
func : 'tickerSentListLength',
args : [ticker],
call_id : ''
});
/**
* Expects a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 send objects.
*
* Note: the result includes sends that failed.
* you can use this list to render a ticker send history.
*/
trac.emit('get',
{
func : 'tickerSentList',
args : [ticker, offset, max],
call_id : ''
});
/**
* Returns the amount of sends ever performed.
*
* Note: the result includes sends that failed.
*/
trac.emit('get',
{
func : 'sentListLength',
args : [],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 send objects.
*
* Note: the result includes sends that failed.
* you can use this to stream the entire send history, across all tickers and addresses.
*/
trac.emit('get',
{
func : 'sentList',
args : [offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount of tokens ever received by the address.
*
* Note: the result includes receives that failed.
*/
trac.emit('get',
{
func : 'accountReceiveListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 receive objects.
*
* Note: the result includes receives that failed.
* you can use this list to render an account receive history.
*
*/
trac.emit('get',
{
func : 'accountReceiveList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Expects an inscription id (not number)
* Returns an accumulation object for internal functions, including the inscribed json object.
* Returns null if the accumulator object doesn't exist.
*/
trac.emit('get',
{
func : 'accumulator',
args : [inscription id],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount of accumulated items, which represent incoming internal function prior tapping.
*
* Note: the result includes accumulated items that may have been tapped already.
*/
trac.emit('get',
{
func : 'accountAccumulatorListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 accumulator objects.
*
* Note: the result includes accumulated items that may have been tapped already.
* to check if an accumulated item has been tapped, check for its presence using the getter "accumulator" above.
* if a returned accumulator from the list is null, it means it has been tapped.
* you can use this list to render an account accumulation history and present tappable internal functions (=awaiting confirmation).
*
*/
trac.emit('get',
{
func : 'accountAccumulatorList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Returns the amount of accumulated items across all addresses, which represent incoming internal function prior tapping.
*
* Note: the result includes accumulated items that may have been tapped already.
*/
trac.emit('get',
{
func : 'accumulatorListLength',
args : [],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns up to 500 send objects.
*
* * Note: the result includes accumulated items that may have been tapped already.
* to check if an accumulated item has been tapped, check for its presence using the getter "accumulator" above.
* if a returned accumulator from the list is null, it means it has been tapped.
* you can use this list to render a "mempool-like" accumulation history and present tappable internal functions (=awaiting confirmation).
*/
trac.emit('get',
{
func : 'accumulatorList',
args : [offset, max],
call_id : ''
});
/**
* Returns an object with trade information.
* Returns null if the trade never existed or got filled.
*
* NOTE: To get a history of trades, please use one of the list getters below.
* You might need this getter in combination with trade histories to verify if the trade has ever existed.
*
*/
trac.emit('get',
{
func : 'trade',
args : [inscription id],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of trades for the given account and ticker.
*
*/
trac.emit('get',
{
func : 'accountTradesList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'accountTradesListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of trades for the given ticker.
*
*/
trac.emit('get',
{
func : 'tickerTradesList',
args : [ticker, offset, max],
call_id : ''
});
/**
* Expects a string ticker.
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'tickerTradesListLength',
args : [ticker],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of trades.
*
*/
trac.emit('get',
{
func : 'tradesList',
args : [offset, max],
call_id : ''
});
/**
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'tradesListLength',
args : [],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of received trades for the given account and ticker.
*
*/
trac.emit('get',
{
func : 'accountReceiveTradesFilledList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'accountReceiveTradesFilledListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string Bitcoin address, a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of filled trades for the given account and ticker.
*
*/
trac.emit('get',
{
func : 'accountTradesFilledList',
args : [address, ticker, offset, max],
call_id : ''
});
/**
* Expects a string Bitcoin address and a string ticker.
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'accountTradesFilledListLength',
args : [address, ticker],
call_id : ''
});
/**
* Expects a string ticker, an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of filled trades for the given ticker.
*
*/
trac.emit('get',
{
func : 'tickerTradesFilledList',
args : [ticker, offset, max],
call_id : ''
});
/**
* Expects a string ticker.
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'tickerTradesFilledListLength',
args : [ticker],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of filled trades.
*
*/
trac.emit('get',
{
func : 'tradesFilledList',
args : [offset, max],
call_id : ''
});
/**
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'tradesFilledListLength',
args : [],
call_id : ''
});
/**
* Accepts a string hash from a signed token-auth.
* Returns true if a hash has ever been used.
* Returns false if not.
*/
trac.emit('get',
{
func : 'authHashExists',
args : [signature_hash],
call_id : ''
});
/**
* Accept an inscription_id (not number) of a token-auth.
* Returns true if the given token-auth has been cancelled.
* Returns false if not.
*/
trac.emit('get',
{
func : 'authCancelled',
args : [inscription_id],
call_id : ''
});
/**
* Expects a string address, an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of token-auth's assigned to the address.
*
*/
trac.emit('get',
{
func : 'accountAuthList',
args : [address, offset, max],
call_id : ''
});
/**
* Expects a string address.
* Returns the amount of token-auth's assigned to the address.
*/
trac.emit('get',
{
func : 'accountAuthListLength',
args : [address],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of token_auths.
*
*/
trac.emit('get',
{
func : 'authList',
args : [offset, max],
call_id : ''
});
/**
* Returns the amount of token_auths.
*/
trac.emit('get',
{
func : 'authListLength',
args : [],
call_id : ''
});
/**
* Expects a string address, an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of token-auth redeems's assigned to the address.
*
*/
trac.emit('get',
{
func : 'accountRedeemList',
args : [address, offset, max],
call_id : ''
});
/**
* Expects a string address.
* Returns the amount of token-auth redeems's assigned to the address.
*/
trac.emit('get',
{
func : 'accountRedeemListLength',
args : [address],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of token_auths redeems.
*
*/
trac.emit('get',
{
func : 'redeemList',
args : [offset, max],
call_id : ''
});
/**
* Returns the amount of token_auths redeems.
*/
trac.emit('get',
{
func : 'redeemListLength',
args : [],
call_id : ''
});
/**
* Expects an integer offset (min 0) to start from and an integer max (max 500).
* Returns a list of existing DMT (Digital Matter Theory) elements.
*
*/
trac.emit('get',
{
func : 'dmtElementsList',
args : [offset, max],
call_id : ''
});
/**
* Returns the amount of history entries.
*/
trac.emit('get',
{
func : 'dmtElementsListLength',
args : [],
call_id : ''
});
/**
* Expects a string inscription id (not number!).
* Returns the allocated amount of a specific transfer inscription as string (BigInt encoded). If '0', then the transfer inscription has been spent.
* Returns null if the transfer inscription does not exist.
* NOTE: This function assumes that the caller knows the decimals of the token associated with the transfer inscription.
* Knowing this is important to calculate the actual amount, based on the decimal points for the targeted token.
*/
trac.emit('get',
{
func : 'transferAmountByInscription',
args : [inscription_id],
call_id : ''
});