forked from DefiLlama/DefiLlama-Adapters
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
138 lines (125 loc) · 3.78 KB
/
index.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
import { gql } from "graphql-request";
import { getPagedGql } from "../utils/gql";
import BigNumber from "bignumber.js";
import { Liq } from "../utils/types";
const subgraphUrl = "https://api.thegraph.com/subgraphs/name/traderjoe-xyz/lending";
const accountsQuery = gql`
query accounts($lastId: ID, $pageSize: Int) {
accounts(first: $pageSize, where: { hasBorrowed: true, id_gt: $lastId }) {
id
health
totalBorrowValueInUSD
totalCollateralValueInUSD
tokens {
id
symbol
market {
name
symbol
collateralFactor
underlyingPriceUSD
exchangeRate
reserveFactor
underlyingDecimals
underlyingAddress
}
borrowBalanceUnderlying
supplyBalanceUnderlying
enteredMarket
}
}
_meta {
block {
number
}
}
}
`;
type Account = {
id: string;
health: string;
totalBorrowValueInUSD: string;
totalCollateralValueInUSD: string;
tokens: Token[];
};
type Token = {
id: string;
symbol: string;
market: Market;
borrowBalanceUnderlying: string;
supplyBalanceUnderlying: string;
enteredMarket: boolean;
};
type Market = {
name: string;
symbol: string;
collateralFactor: string;
underlyingPriceUSD: string;
exchangeRate: string;
reserveFactor: string;
underlyingDecimals: number;
underlyingAddress: string;
};
const EXPLORER_BASE_URL = "https://snowtrace.io/address/";
const positions = async () => {
const accounts = (await getPagedGql(subgraphUrl, accountsQuery, "accounts")) as Account[];
// all positions across all users
const positions = accounts.flatMap((account) => {
const { totalBorrowValueInUSD, totalCollateralValueInUSD } = account;
const debts = account.tokens
.filter((token) => !(Number(token.borrowBalanceUnderlying) === 0 && Number(token.supplyBalanceUnderlying) === 0))
.map((token) => {
const decimals = token.market.underlyingDecimals;
const price = Number(token.market.underlyingPriceUSD);
const collateralFactor = Number(token.market.collateralFactor); // equivalent to liqThreshold in aave
let debt = new BigNumber(token.borrowBalanceUnderlying);
if (token.enteredMarket) {
const factoredSupply = new BigNumber(token.supplyBalanceUnderlying).times(collateralFactor);
debt = debt.minus(factoredSupply);
}
debt = debt.times(price);
return {
debt,
price,
token: token.market.underlyingAddress,
totalBal: token.supplyBalanceUnderlying,
decimals,
};
});
const liquidablePositions = debts
.filter(({ debt }) => debt.lt(0))
.map((pos) => {
const usdPosNetCollateral = pos.debt.negated();
const otherCollateral = new BigNumber(totalCollateralValueInUSD).minus(usdPosNetCollateral);
const diffDebt = new BigNumber(totalBorrowValueInUSD).minus(otherCollateral);
if (diffDebt.gt(0)) {
const amountCollateral = usdPosNetCollateral.div(pos.price);
const liqPrice = diffDebt.div(amountCollateral);
return {
owner: account.id,
liqPrice: Number(liqPrice.toFixed(6)),
collateral: "avax:" + pos.token,
collateralAmount: new BigNumber(pos.totalBal).times(10 ** pos.decimals).toFixed(0),
extra: {
url: EXPLORER_BASE_URL + account.id,
},
} as Liq;
} else {
return {
owner: "",
liqPrice: 0,
collateral: "",
collateralAmount: "",
};
}
})
.filter((t) => !!t.owner);
return liquidablePositions;
});
return positions;
};
module.exports = {
avalanche: {
liquidations: positions,
},
};