Files
DefiLlama-Adapters/projects/polymarket.js
2022-03-21 19:45:48 +01:00

121 lines
5.0 KiB
JavaScript

const sdk = require("@defillama/sdk");
const { request, gql } = require("graphql-request");
const { BigNumber } = require("bignumber.js");
const utils = require('./helper/utils');
// Can retrieve https://strapi-matic.poly.market/markets?active=true&_sort=end_date:desc
// Old graphql https://api.thegraph.com/subgraphs/name/tokenunion/polymarket-matic
// New graphql https://api.thegraph.com/subgraphs/name/polymarket/matic-markets-5
// Useful resources : https://polymarketwhales.info/markets and https://polymarket.com/
// # block: { number: $block }
const graphUrl = 'https://api.thegraph.com/subgraphs/name/polymarket/matic-markets-5'
const graphQuery = gql`
query GET_POLYMARKET($skip: Int, $block: Int) {
conditions(
first: 1000
skip: $skip
block: { number: $block }
) {
id
oracle
questionId
outcomeSlotCount
resolutionTimestamp
fixedProductMarketMakers(
orderBy: creationTimestamp
)
{
id
creationTimestamp
collateralToken {
id name symbol decimals
}
liquidityParameter
scaledLiquidityParameter
collateralVolume
scaledCollateralVolume
}
}
}
`;
async function getMarketsLiquidity_graphql(timestamp, block, chainBlocks) {
let scaledLiquidityParameterSum = BigNumber(0)
let skip = 0
// Continue querying graph while end not reached. Hard cap: skip = 5000
while (skip !== -1) {
const { conditions } = await request(
graphUrl,
graphQuery,
{skip, block}
);
skip += 1000
console.log(`${conditions && conditions.length} conditions found for skip: ${skip}`)
if (conditions && conditions.length > 0) {
conditions.forEach(condition => {
condition.fixedProductMarketMakers.forEach(fpmm => {
console.log(fpmm)
scaledLiquidityParameterSum = scaledLiquidityParameterSum.plus(BigNumber(fpmm.scaledLiquidityParameter))
})
})
}
else {
// Stop criteria: no conditions returned by graphql api
skip = -1
}
}
return scaledLiquidityParameterSum
}
// After a market resolves, then the winning market participants can withdraw their share based on the redemption rate and their contribution at the closing of the market. All participants do not do it immediately though, so volume of every market should be accounted for in TVL
// const polymarket_api_url = 'https://strapi-matic.poly.market/markets?_limit=-1&_sort=closed_time:desc' // &active=true
const conditionalTokensContract = '0x4D97DCd97eC945f40cF65F87097ACe5EA0476045'
const polygonUsdcContract = '0x2791bca1f2de4661ed88a30c99a7a9449aa84174'
const polymarket_api_url = 'https://strapi-matic.poly.market/markets?_limit=-1&_sort=volume:desc' // &active=true
async function getMarketsLiquidity_api() {
let markets = await utils.fetchURL(polymarket_api_url)
// Filter out unwanted fields
markets = markets.data.map(({ id, slug, volume, liquidity, created_at, closed }) => ({id, slug, volume, liquidity, created_at, closed}));
markets.forEach(market => {
market.volume = Number(market.volume) || 0
market.liquidity = Number(market.liquidity) || 0
if (market.liquidity < 0) market.liquidity = 0
});
const marketsLiquidity = markets.reduce((acc, market) => acc.plus(BigNumber(market.liquidity)), BigNumber(0));
return marketsLiquidity
}
async function polygon(timestamp, block, chainBlocks) {
// Get markets liquidity using API
//const marketsLiquidity = (await getMarketsLiquidity_api()).times(1e6)
//const marketsLiquidity = (await getMarketsLiquidity_graphql(timestamp, block, chainBlocks)).times(1e6)
// Also account for USDC held in the conditional tokens contract because a lot of positions are held outside of the LPs
// Corresponds to open positions that should be counted as TVL (since they still represent USDC locked into the conditional tokens contract)
let conditionalTokensUSDC = await sdk.api.erc20.balanceOf({
target: polygonUsdcContract,
owner: conditionalTokensContract,
block: chainBlocks['polygon'],
chain: 'polygon'
})
//conditionalTokensUSDC = BigNumber(conditionalTokensUSDC.output)
// Total open interest: the conditional tokens are held at 0x4D97DCd97eC945f40cF65F87097ACe5EA0476045 and then each market has it's own contract, the address of which is the id of the FixedProductMarketMaker
//const tvl = marketsLiquidity.plus(conditionalTokensUSDC).toFixed(0)
//console.log(`-----\n${marketsLiquidity.div(1e12).toFixed(8)}M of marketsLiquidity \n${conditionalTokensUSDC.div(1e12).toFixed(8)}M of conditionalTokensUSDC \nTVL: ${BigNumber(tvl).div(1e12).toFixed(2)}M\n`)
return {['polygon:' + polygonUsdcContract]: conditionalTokensUSDC.output};
}
module.exports = {
polygon: {
tvl: polygon
},
methodology: `TVL is the total quantity of USDC held in the conditional tokens contract as well as USDC collateral submitted to every polymarket' markets ever opened - once the markets resolve, participants can withdraw theire share given the redeption rate and their input stake, but they do not all do it.`
}