2020-11-20 13:12:02 +01:00
|
|
|
import {
|
|
|
|
BigDecimal,
|
|
|
|
BigInt,
|
|
|
|
Bytes,
|
|
|
|
dataSource,
|
2020-12-08 13:32:15 +01:00
|
|
|
Address,
|
2020-12-11 11:37:09 +01:00
|
|
|
ethereum,
|
|
|
|
log
|
2020-11-20 13:12:02 +01:00
|
|
|
} from '@graphprotocol/graph-ts'
|
2020-12-08 13:32:15 +01:00
|
|
|
|
2020-11-20 13:12:02 +01:00
|
|
|
import {
|
2020-12-08 13:32:15 +01:00
|
|
|
Pool as PoolEntity,
|
2020-11-26 12:10:45 +01:00
|
|
|
User,
|
|
|
|
PoolToken,
|
|
|
|
PoolShare,
|
|
|
|
PoolTransaction,
|
2020-12-02 11:07:09 +01:00
|
|
|
PoolFactory,
|
|
|
|
Datatoken,
|
|
|
|
TokenBalance,
|
|
|
|
TokenTransaction,
|
2021-07-30 09:36:44 +02:00
|
|
|
PoolTransactionTokenValues,
|
|
|
|
Global
|
2021-02-09 11:13:31 +01:00
|
|
|
} from './@types/schema'
|
2020-12-11 11:37:09 +01:00
|
|
|
|
2021-02-09 11:13:31 +01:00
|
|
|
import { Pool } from './@types/templates/Pool/Pool'
|
2021-07-29 10:55:48 +02:00
|
|
|
import { ERC20 } from './@types/templates/Pool/ERC20'
|
|
|
|
import { ERC20SymbolBytes } from './@types/templates/Pool/ERC20SymbolBytes'
|
|
|
|
import { ERC20NameBytes } from './@types/templates/Pool/ERC20NameBytes'
|
2020-11-20 13:12:02 +01:00
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export const ZERO_BD = BigDecimal.fromString('0.0')
|
|
|
|
export const MINUS_1_BD = BigDecimal.fromString('-1.0')
|
|
|
|
export const ONE_BD = BigDecimal.fromString('1.0')
|
2020-11-20 13:12:02 +01:00
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export const ONE_BASE_18 = BigInt.fromI32(10).pow(18 as u8)
|
|
|
|
export const BONE = BigDecimal.fromString('1000000000000000000')
|
2020-12-08 13:32:15 +01:00
|
|
|
|
2021-01-18 15:57:55 +01:00
|
|
|
export const ENABLE_DEBUG = true
|
2020-12-03 10:05:34 +01:00
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
const network = dataSource.network()
|
2020-11-20 13:12:02 +01:00
|
|
|
|
2021-07-30 09:36:44 +02:00
|
|
|
export function getOceanAddress(): string {
|
2021-03-01 13:41:36 +01:00
|
|
|
// switch is not working for some reason
|
|
|
|
if (network == 'ropsten') return '0x5e8dcb2afa23844bcc311b00ad1a0c30025aade9'
|
|
|
|
if (network == 'rinkeby') return '0x8967bcf84170c91b0d24d4302c2376283b0b3a07'
|
|
|
|
if (network == 'polygon') return '0x282d8efce846a88b159800bd4130ad77443fa1a1'
|
2021-04-16 13:47:03 +02:00
|
|
|
if (network == 'moonbeamalpha')
|
2021-07-07 11:06:38 +02:00
|
|
|
return '0xf6410bf5d773c7a41ebff972f38e7463fa242477'
|
2021-06-10 12:21:50 +02:00
|
|
|
if (network == 'gaiaxtestnet')
|
|
|
|
return '0x80e63f73cac60c1662f27d2dfd2ea834acddbaa8'
|
2021-09-16 19:53:21 +02:00
|
|
|
if (network == 'catenaxtestnet')
|
|
|
|
return '0x80e63f73cac60c1662f27d2dfd2ea834acddbaa8'
|
2021-06-10 12:21:50 +02:00
|
|
|
if (network == 'mumbai') return '0xd8992ed72c445c35cb4a2be468568ed1079357c8'
|
|
|
|
if (network == 'bsc') return '0xdce07662ca8ebc241316a15b611c89711414dd1a'
|
2021-10-05 14:05:52 +02:00
|
|
|
if (network == 'celoalfajores')
|
|
|
|
return '0xd8992ed72c445c35cb4a2be468568ed1079357c8'
|
|
|
|
if (network == 'energyweb')
|
|
|
|
return '0x593122aae80a6fc3183b2ac0c4ab3336debee528'
|
|
|
|
if (network == 'moonriver')
|
|
|
|
return '0x99c409e5f62e4bd2ac142f17cafb6810b8f0baae'
|
2021-01-20 09:35:53 +01:00
|
|
|
return '0x967da4048cd07ab37855c090aaf366e4ce1b9f48'
|
|
|
|
}
|
2020-11-20 13:12:02 +01:00
|
|
|
|
2021-02-09 11:13:31 +01:00
|
|
|
export const OCEAN: string = getOceanAddress()
|
|
|
|
|
2021-07-30 09:36:44 +02:00
|
|
|
export function getGlobalStats(): Global | null {
|
|
|
|
let gStats: Global | null = Global.load('1')
|
|
|
|
if (gStats == null) {
|
|
|
|
gStats = new Global('1')
|
|
|
|
gStats.totalOceanLiquidity = ZERO_BD
|
|
|
|
gStats.totalSwapVolume = ZERO_BD
|
|
|
|
gStats.totalValueLocked = ZERO_BD
|
|
|
|
gStats.totalOrderVolume = ZERO_BD
|
|
|
|
gStats.orderCount = BigInt.fromI32(0)
|
|
|
|
gStats.poolCount = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
return gStats
|
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function _debuglog(
|
|
|
|
message: string,
|
|
|
|
event: ethereum.Event,
|
|
|
|
args: Array<string>
|
|
|
|
): void {
|
2020-12-03 10:05:34 +01:00
|
|
|
if (event != null) {
|
|
|
|
args.push(event.transaction.hash.toHex())
|
|
|
|
args.push(event.address.toHex())
|
|
|
|
}
|
2020-12-11 11:37:09 +01:00
|
|
|
for (let i = 0; i < args.length; i++) {
|
2020-12-03 10:05:34 +01:00
|
|
|
message = message.concat(' {}')
|
|
|
|
}
|
2021-06-25 11:00:34 +02:00
|
|
|
log.debug('@@@@@@ ' + message, args)
|
2020-12-08 17:57:09 +01:00
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function debuglog(
|
|
|
|
message: string,
|
|
|
|
event: ethereum.Event,
|
|
|
|
args: Array<string>
|
|
|
|
): void {
|
2020-12-08 17:57:09 +01:00
|
|
|
if (!ENABLE_DEBUG) return
|
|
|
|
_debuglog(message, event, args)
|
2020-12-03 10:05:34 +01:00
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function hexToDecimal(hexString: string, decimals: i32): BigDecimal {
|
|
|
|
const bytes = Bytes.fromHexString(hexString.toString()).reverse() as Bytes
|
|
|
|
const bi = BigInt.fromUnsignedBytes(bytes)
|
|
|
|
const scale = BigInt.fromI32(10)
|
|
|
|
.pow(decimals as u8)
|
|
|
|
.toBigDecimal()
|
2020-11-20 13:12:02 +01:00
|
|
|
return bi.divDecimal(scale)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function bigIntToDecimal(amount: BigInt, decimals: i32): BigDecimal {
|
2020-12-11 11:37:09 +01:00
|
|
|
const scale = BigInt.fromI32(10)
|
|
|
|
.pow(decimals as u8)
|
|
|
|
.toBigDecimal()
|
2020-11-20 13:12:02 +01:00
|
|
|
return amount.toBigDecimal().div(scale)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function tokenToDecimal(amount: BigDecimal, decimals: i32): BigDecimal {
|
2020-12-11 11:37:09 +01:00
|
|
|
const scale = BigInt.fromI32(10)
|
|
|
|
.pow(decimals as u8)
|
|
|
|
.toBigDecimal()
|
2020-11-20 13:12:02 +01:00
|
|
|
return amount.div(scale)
|
|
|
|
}
|
|
|
|
|
2020-12-08 13:32:15 +01:00
|
|
|
export function decimalToBigInt(value: BigDecimal): BigInt {
|
|
|
|
value.truncate(18)
|
2020-12-11 11:37:09 +01:00
|
|
|
const scale = BigInt.fromI32(10).pow((value.exp.toI32() + 18) as u8)
|
2020-12-08 13:32:15 +01:00
|
|
|
return value.digits.times(scale)
|
|
|
|
}
|
|
|
|
|
2021-07-29 10:55:48 +02:00
|
|
|
export function isNullEthValue(value: string): boolean {
|
|
|
|
return (
|
|
|
|
value ==
|
|
|
|
'0x0000000000000000000000000000000000000000000000000000000000000001'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTokenSymbol(tokenAddress: Address): string {
|
|
|
|
const contract = ERC20.bind(tokenAddress)
|
|
|
|
const contractSymbolBytes = ERC20SymbolBytes.bind(tokenAddress)
|
|
|
|
|
|
|
|
// try types string and bytes32 for symbol
|
|
|
|
let symbolValue = 'unknown'
|
|
|
|
const symbolResult = contract.try_symbol()
|
|
|
|
if (symbolResult.reverted) {
|
|
|
|
const symbolResultBytes = contractSymbolBytes.try_symbol()
|
|
|
|
if (!symbolResultBytes.reverted) {
|
|
|
|
// for broken pairs that have no symbol function exposed
|
|
|
|
if (!isNullEthValue(symbolResultBytes.value.toHexString())) {
|
|
|
|
symbolValue = symbolResultBytes.value.toString()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
symbolValue = symbolResult.value
|
|
|
|
}
|
|
|
|
|
|
|
|
return symbolValue
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTokenName(tokenAddress: Address): string {
|
|
|
|
const contract = ERC20.bind(tokenAddress)
|
|
|
|
const contractNameBytes = ERC20NameBytes.bind(tokenAddress)
|
|
|
|
|
|
|
|
// try types string and bytes32 for name
|
|
|
|
let nameValue = 'unknown'
|
|
|
|
const nameResult = contract.try_name()
|
|
|
|
if (nameResult.reverted) {
|
|
|
|
const nameResultBytes = contractNameBytes.try_name()
|
|
|
|
if (!nameResultBytes.reverted) {
|
|
|
|
// for broken exchanges that have no name function exposed
|
|
|
|
if (!isNullEthValue(nameResultBytes.value.toHexString())) {
|
|
|
|
nameValue = nameResultBytes.value.toString()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
nameValue = nameResult.value
|
|
|
|
}
|
|
|
|
|
|
|
|
return nameValue
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTokenDecimals(tokenAddress: Address): i32 {
|
|
|
|
const contract = ERC20.bind(tokenAddress)
|
|
|
|
let decimals = 18
|
|
|
|
const decimalCall = contract.try_decimals()
|
|
|
|
if (!decimalCall.reverted) {
|
|
|
|
decimals = decimalCall.value
|
|
|
|
}
|
|
|
|
return decimals
|
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function updatePoolTokenBalance(
|
|
|
|
poolToken: PoolToken,
|
|
|
|
balance: BigDecimal,
|
|
|
|
source: string
|
|
|
|
): void {
|
|
|
|
debuglog(
|
|
|
|
'########## updating poolToken balance (source, oldBalance, newBalance, poolId) ',
|
|
|
|
null,
|
|
|
|
[source, poolToken.balance.toString(), balance.toString(), poolToken.poolId]
|
|
|
|
)
|
2021-02-09 11:13:31 +01:00
|
|
|
if (balance < ZERO_BD || poolToken.balance < ZERO_BD) {
|
|
|
|
log.warning(
|
|
|
|
'EEEEEEEEEEEEEEEEE poolToken.balance < Zero: pool={}, poolToken={}, oldBalance={}, newBalance={}',
|
2021-02-23 16:54:15 +01:00
|
|
|
[
|
|
|
|
poolToken.poolId,
|
2021-07-29 10:55:48 +02:00
|
|
|
poolToken.address.toString(),
|
2021-02-23 16:54:15 +01:00
|
|
|
poolToken.balance.toString(),
|
|
|
|
balance.toString()
|
|
|
|
]
|
|
|
|
)
|
2021-02-09 11:13:31 +01:00
|
|
|
}
|
|
|
|
|
2020-12-03 10:05:34 +01:00
|
|
|
poolToken.balance = balance
|
|
|
|
}
|
|
|
|
|
2021-07-30 09:36:44 +02:00
|
|
|
export function updatePoolSwapVolume(
|
|
|
|
pool: Pool,
|
|
|
|
swapAmount: BigDecimal
|
|
|
|
// source: string
|
|
|
|
): void {
|
|
|
|
debuglog(
|
|
|
|
'########## updating poolToken balance (source, oldBalance, newBalance, poolId) ',
|
|
|
|
null,
|
|
|
|
[source, pool.totalSwapVolume.toString(), swapAmount.toString(), pool.id]
|
|
|
|
)
|
|
|
|
if (swapAmount < ZERO_BD || pool.totalSwapVolume < ZERO_BD) {
|
|
|
|
log.warning(
|
|
|
|
'EEEEEEEEEEEEEEEEE poolToken.balance < Zero: pool={}, poolToken={}, oldBalance={}, newBalance={}',
|
|
|
|
[
|
|
|
|
pool.id,
|
|
|
|
poolToken.tokenAddress.toString(),
|
|
|
|
poolToken.balance.toString(),
|
|
|
|
swapAmount.toString()
|
|
|
|
]
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
poolToken.swapBalanceOcean = poolToken.swapBalanceOcean.plus(swapAmount)
|
|
|
|
}
|
2020-12-11 11:37:09 +01:00
|
|
|
export function createUserEntity(address: string): void {
|
|
|
|
if (User.load(address) == null) {
|
|
|
|
const user = new User(address)
|
2021-10-05 12:51:47 +02:00
|
|
|
user.nrSales = 0
|
2020-12-11 11:37:09 +01:00
|
|
|
user.save()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function createPoolShareEntity(
|
|
|
|
id: string,
|
|
|
|
pool: string,
|
|
|
|
user: string
|
|
|
|
): void {
|
|
|
|
const poolShare = new PoolShare(id)
|
2020-11-20 13:12:02 +01:00
|
|
|
|
|
|
|
createUserEntity(user)
|
|
|
|
|
|
|
|
poolShare.userAddress = user
|
|
|
|
poolShare.poolId = pool
|
|
|
|
poolShare.balance = ZERO_BD
|
|
|
|
poolShare.save()
|
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function createPoolTokenEntity(
|
|
|
|
id: string,
|
|
|
|
pool: string,
|
2021-07-29 10:55:48 +02:00
|
|
|
address: Address
|
2020-12-11 11:37:09 +01:00
|
|
|
): void {
|
2021-07-29 10:55:48 +02:00
|
|
|
const datatoken = Datatoken.load(address.toHexString())
|
2020-11-20 13:12:02 +01:00
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
const poolToken = new PoolToken(id)
|
2020-11-20 13:12:02 +01:00
|
|
|
poolToken.poolId = pool
|
2021-07-29 10:55:48 +02:00
|
|
|
poolToken.isDatatoken = !!datatoken
|
2020-12-11 11:37:09 +01:00
|
|
|
poolToken.tokenId = datatoken ? datatoken.id : ''
|
2021-07-29 10:55:48 +02:00
|
|
|
poolToken.address = address.toHexString()
|
2020-11-20 13:12:02 +01:00
|
|
|
poolToken.balance = ZERO_BD
|
|
|
|
poolToken.denormWeight = ZERO_BD
|
2021-07-29 10:55:48 +02:00
|
|
|
poolToken.symbol = getTokenSymbol(address)
|
|
|
|
poolToken.name = getTokenName(address)
|
|
|
|
poolToken.decimals = getTokenDecimals(address)
|
2020-11-20 13:12:02 +01:00
|
|
|
poolToken.save()
|
|
|
|
}
|
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
export function updatePoolTransactionToken(
|
2020-12-11 11:37:09 +01:00
|
|
|
poolTx: string,
|
|
|
|
poolTokenId: string,
|
|
|
|
amount: BigDecimal,
|
|
|
|
balance: BigDecimal,
|
|
|
|
feeValue: BigDecimal
|
2020-12-02 11:07:09 +01:00
|
|
|
): void {
|
2021-03-01 13:41:36 +01:00
|
|
|
log.warning('WWWWWWWWWW ---- started update ptx with id {}', [poolTx])
|
2021-04-16 10:07:01 +02:00
|
|
|
log.warning('updatePoolTransactionToken({}, {} , {} , {} , {}}', [
|
|
|
|
poolTx,
|
|
|
|
poolTokenId,
|
|
|
|
amount.toString(),
|
|
|
|
balance.toString(),
|
|
|
|
feeValue.toString()
|
|
|
|
])
|
2020-12-11 11:37:09 +01:00
|
|
|
const ptx = PoolTransaction.load(poolTx)
|
|
|
|
const poolToken = PoolToken.load(poolTokenId)
|
|
|
|
const pool = PoolEntity.load(poolToken.poolId)
|
2021-04-16 10:07:01 +02:00
|
|
|
if (!ptx) {
|
|
|
|
log.error('Cannot load PoolTransaction {}', [poolTx])
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (!poolToken) {
|
|
|
|
log.error('Cannot load PoolToken {}', [poolTokenId])
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (!pool) {
|
|
|
|
log.error('Cannot load PoolEntity {}', [poolToken.poolId])
|
|
|
|
return
|
|
|
|
}
|
2021-09-03 13:40:08 +02:00
|
|
|
const ptxTokenValuesId = poolTx.concat('-').concat(poolTokenId)
|
2020-12-02 11:07:09 +01:00
|
|
|
let ptxTokenValues = PoolTransactionTokenValues.load(ptxTokenValuesId)
|
|
|
|
if (ptxTokenValues == null) {
|
|
|
|
ptxTokenValues = new PoolTransactionTokenValues(ptxTokenValuesId)
|
2021-04-16 10:07:01 +02:00
|
|
|
log.warning('created PoolTransactionTokenValues for {}', [ptxTokenValuesId])
|
2020-12-02 11:07:09 +01:00
|
|
|
}
|
2021-09-03 13:40:08 +02:00
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
ptxTokenValues.txId = poolTx
|
|
|
|
ptxTokenValues.poolToken = poolTokenId
|
2020-12-02 17:20:29 +01:00
|
|
|
ptxTokenValues.poolAddress = poolToken.poolId
|
|
|
|
ptxTokenValues.userAddress = ptx.userAddress
|
2021-07-29 10:55:48 +02:00
|
|
|
ptxTokenValues.tokenAddress = PoolToken.load(poolTokenId).address
|
2020-12-02 17:20:29 +01:00
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
ptxTokenValues.value = amount
|
2020-12-02 17:20:29 +01:00
|
|
|
ptxTokenValues.tokenReserve = balance
|
2020-12-02 11:07:09 +01:00
|
|
|
ptxTokenValues.feeValue = feeValue
|
|
|
|
if (amount.lt(ZERO_BD)) {
|
|
|
|
ptxTokenValues.type = 'out'
|
|
|
|
} else {
|
|
|
|
ptxTokenValues.type = 'in'
|
2020-11-26 07:38:08 +01:00
|
|
|
}
|
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
ptxTokenValues.save()
|
2021-09-03 13:40:08 +02:00
|
|
|
log.warning('ptxTokenValues {} saved {}', [
|
|
|
|
ptxTokenValues.id,
|
|
|
|
ptxTokenValues.type
|
|
|
|
])
|
2021-01-19 12:16:03 +01:00
|
|
|
if (ptxTokenValues.tokenAddress == OCEAN) {
|
2021-02-09 11:13:31 +01:00
|
|
|
const factory = PoolFactory.load('1')
|
2021-07-30 09:36:44 +02:00
|
|
|
|
2021-05-13 08:17:54 +02:00
|
|
|
factory.totalOceanLiquidity = factory.totalOceanLiquidity
|
|
|
|
.plus(ptxTokenValues.tokenReserve)
|
|
|
|
.minus(pool.oceanReserve)
|
2021-07-30 09:36:44 +02:00
|
|
|
|
|
|
|
const gStats: Global | null = getGlobalStats()
|
|
|
|
gStats.totalOceanLiquidity = factory.totalOceanLiquidity
|
|
|
|
|
|
|
|
gStats.save()
|
2021-02-09 11:13:31 +01:00
|
|
|
if (factory.totalOceanLiquidity < ZERO_BD || pool.oceanReserve < ZERO_BD) {
|
|
|
|
log.warning(
|
|
|
|
'EEEEEEEEEEEEEEEEE totalOceanLiquidity or oceanReserve < Zero: pool={}, totOcnLiq={}, ocnRes={}',
|
2021-02-23 16:54:15 +01:00
|
|
|
[
|
|
|
|
pool.id,
|
|
|
|
factory.totalOceanLiquidity.toString(),
|
|
|
|
pool.oceanReserve.toString()
|
|
|
|
]
|
|
|
|
)
|
2021-02-09 11:13:31 +01:00
|
|
|
}
|
2020-12-08 17:57:09 +01:00
|
|
|
ptx.oceanReserve = ptxTokenValues.tokenReserve
|
2020-12-09 11:54:13 +01:00
|
|
|
pool.oceanReserve = ptxTokenValues.tokenReserve
|
2021-02-09 11:13:31 +01:00
|
|
|
factory.save()
|
2020-12-08 17:57:09 +01:00
|
|
|
} else {
|
|
|
|
ptx.datatokenReserve = ptxTokenValues.tokenReserve
|
2020-12-09 11:54:13 +01:00
|
|
|
pool.datatokenReserve = ptxTokenValues.tokenReserve
|
2020-12-08 17:57:09 +01:00
|
|
|
}
|
2021-03-01 13:41:36 +01:00
|
|
|
// debuglog('########## updatePoolTransactionToken: ', null, [
|
|
|
|
// BigInt.fromI32(ptx.block).toString(),
|
|
|
|
// BigInt.fromI32(ptx.timestamp).toString(),
|
|
|
|
// ptxTokenValues.type,
|
|
|
|
// ptxTokenValues.value.toString(),
|
|
|
|
// ptxTokenValues.tokenReserve.toString(),
|
|
|
|
// poolToken.poolId
|
|
|
|
// ])
|
2021-04-16 10:07:01 +02:00
|
|
|
log.warning('saving ptx {} ', [ptx.id.toString()])
|
2020-12-08 17:57:09 +01:00
|
|
|
ptx.save()
|
2021-04-16 10:07:01 +02:00
|
|
|
log.warning('saving pool {} ', [pool.id.toString()])
|
2020-12-09 11:54:13 +01:00
|
|
|
pool.save()
|
2020-12-02 11:07:09 +01:00
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function calcSpotPrice(
|
|
|
|
balanceIn: BigDecimal,
|
|
|
|
wIn: BigDecimal,
|
|
|
|
balanceOut: BigDecimal,
|
|
|
|
wOut: BigDecimal,
|
|
|
|
swapFee: BigDecimal
|
|
|
|
): BigDecimal {
|
|
|
|
if (balanceIn <= ZERO_BD || balanceOut <= ZERO_BD) return MINUS_1_BD
|
|
|
|
debuglog('################ calcSpotPrice', null, [
|
|
|
|
balanceIn.toString(),
|
|
|
|
wIn.toString(),
|
|
|
|
balanceOut.toString(),
|
|
|
|
wOut.toString(),
|
|
|
|
swapFee.toString()
|
|
|
|
])
|
|
|
|
|
|
|
|
const numer = balanceIn.div(wIn)
|
|
|
|
const denom = balanceOut.div(wOut)
|
|
|
|
if (denom <= ZERO_BD) return MINUS_1_BD
|
|
|
|
|
|
|
|
const ratio = numer.div(denom)
|
|
|
|
const scale = ONE_BD.div(ONE_BD.minus(swapFee))
|
|
|
|
const price = ratio.times(scale)
|
|
|
|
price.truncate(18)
|
|
|
|
debuglog('################ calcSpotPrice values:', null, [
|
|
|
|
numer.toString(),
|
|
|
|
denom.toString(),
|
|
|
|
ratio.toString(),
|
|
|
|
scale.toString(),
|
|
|
|
price.toString()
|
|
|
|
])
|
|
|
|
return price
|
|
|
|
}
|
|
|
|
|
|
|
|
export function createPoolTransaction(
|
|
|
|
event: ethereum.Event,
|
|
|
|
// eslint-disable-next-line camelcase
|
|
|
|
event_type: string,
|
|
|
|
userAddress: string
|
|
|
|
): void {
|
|
|
|
const poolId = event.address.toHex()
|
|
|
|
const pool = PoolEntity.load(poolId)
|
|
|
|
const ptx = event.transaction.hash.toHexString()
|
2020-12-02 11:07:09 +01:00
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
const ocnToken = PoolToken.load(poolId.concat('-').concat(OCEAN))
|
|
|
|
const dtToken = PoolToken.load(
|
|
|
|
poolId.concat('-').concat(pool.datatokenAddress)
|
|
|
|
)
|
2020-12-02 17:20:29 +01:00
|
|
|
if (ocnToken == null || dtToken == null) {
|
|
|
|
return
|
|
|
|
}
|
2020-12-02 11:07:09 +01:00
|
|
|
|
|
|
|
let poolTx = PoolTransaction.load(ptx)
|
2020-12-02 17:20:29 +01:00
|
|
|
if (poolTx != null) {
|
|
|
|
return
|
2020-11-20 13:12:02 +01:00
|
|
|
}
|
2020-12-02 17:20:29 +01:00
|
|
|
poolTx = new PoolTransaction(ptx)
|
2020-11-20 13:12:02 +01:00
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
poolTx.poolAddress = poolId
|
|
|
|
poolTx.userAddress = userAddress
|
2020-12-02 17:20:29 +01:00
|
|
|
poolTx.poolAddressStr = poolId
|
|
|
|
poolTx.userAddressStr = userAddress
|
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
poolTx.sharesTransferAmount = ZERO_BD
|
|
|
|
poolTx.sharesBalance = ZERO_BD
|
2020-12-03 10:05:34 +01:00
|
|
|
|
2020-12-09 11:54:13 +01:00
|
|
|
// pool.datatokenReserve = dtToken.balance
|
|
|
|
// pool.oceanReserve = ocnToken.balance
|
2020-12-08 17:57:09 +01:00
|
|
|
// Initial reserve values, will be updated in `updatePoolTransactionToken`
|
|
|
|
poolTx.datatokenReserve = dtToken.balance
|
|
|
|
poolTx.oceanReserve = ocnToken.balance
|
2021-05-13 08:17:54 +02:00
|
|
|
debuglog('poolTX reserves:(dt, ocean)', null, [
|
|
|
|
poolTx.datatokenReserve.toString(),
|
|
|
|
poolTx.oceanReserve.toString()
|
|
|
|
])
|
2020-12-08 13:32:15 +01:00
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
const p = Pool.bind(Address.fromString(poolId))
|
2020-12-09 11:54:13 +01:00
|
|
|
debuglog(
|
2021-05-13 08:17:54 +02:00
|
|
|
'createPoolTransaction args sent to calcInGivenOut (ocnBalance, ocnWeight, dtBalance, dtWeight, dtAmount, swapFee)',
|
2020-12-11 11:37:09 +01:00
|
|
|
null,
|
2020-12-09 11:54:13 +01:00
|
|
|
[
|
|
|
|
decimalToBigInt(ocnToken.balance).toString(),
|
|
|
|
decimalToBigInt(ocnToken.denormWeight).toString(),
|
|
|
|
decimalToBigInt(dtToken.balance).toString(),
|
|
|
|
decimalToBigInt(dtToken.denormWeight).toString(),
|
|
|
|
ONE_BASE_18.toString(),
|
2021-05-13 08:17:54 +02:00
|
|
|
decimalToBigInt(pool.swapFee).toString()
|
2020-12-09 11:54:13 +01:00
|
|
|
]
|
|
|
|
)
|
2021-05-13 08:17:54 +02:00
|
|
|
const priceResult = p.try_calcInGivenOut(
|
|
|
|
decimalToBigInt(ocnToken.balance),
|
|
|
|
decimalToBigInt(ocnToken.denormWeight),
|
|
|
|
decimalToBigInt(dtToken.balance),
|
|
|
|
decimalToBigInt(dtToken.denormWeight),
|
|
|
|
ONE_BASE_18,
|
|
|
|
decimalToBigInt(pool.swapFee)
|
|
|
|
)
|
|
|
|
debuglog('got results', null, [])
|
2020-12-11 11:37:09 +01:00
|
|
|
poolTx.consumePrice = priceResult.reverted
|
|
|
|
? MINUS_1_BD
|
|
|
|
: bigIntToDecimal(priceResult.value, 18)
|
2021-05-13 08:17:54 +02:00
|
|
|
debuglog('calcInGivenOut:', null, [
|
|
|
|
priceResult.reverted ? 'failed' : priceResult.value.toString()
|
|
|
|
])
|
|
|
|
|
|
|
|
const priceSpot = p.try_calcSpotPrice(
|
|
|
|
decimalToBigInt(ocnToken.balance),
|
|
|
|
decimalToBigInt(ocnToken.denormWeight),
|
|
|
|
decimalToBigInt(dtToken.balance),
|
|
|
|
decimalToBigInt(dtToken.denormWeight),
|
|
|
|
decimalToBigInt(pool.swapFee)
|
2020-12-09 11:54:13 +01:00
|
|
|
)
|
2021-05-13 08:17:54 +02:00
|
|
|
poolTx.spotPrice = priceSpot.reverted
|
|
|
|
? ZERO_BD
|
|
|
|
: bigIntToDecimal(priceSpot.value, 18)
|
|
|
|
debuglog('SpotPrice:', null, [
|
|
|
|
priceSpot.reverted ? 'failed' : priceSpot.value.toString()
|
|
|
|
])
|
2020-12-09 11:54:13 +01:00
|
|
|
|
|
|
|
pool.consumePrice = poolTx.consumePrice
|
|
|
|
pool.spotPrice = poolTx.spotPrice
|
2021-02-09 11:13:31 +01:00
|
|
|
const oldValueLocked = pool.valueLocked
|
|
|
|
const spotPrice = pool.spotPrice >= ZERO_BD ? pool.spotPrice : ZERO_BD
|
2021-05-13 08:17:54 +02:00
|
|
|
pool.valueLocked = poolTx.oceanReserve.plus(
|
|
|
|
poolTx.datatokenReserve.times(spotPrice)
|
|
|
|
)
|
2021-02-09 11:13:31 +01:00
|
|
|
const factory = PoolFactory.load('1')
|
|
|
|
if (oldValueLocked < ZERO_BD || pool.valueLocked < ZERO_BD) {
|
|
|
|
log.warning(
|
|
|
|
'EEEEEEEEEEEEEEEEE valueLocked < Zero: pool={}, oldVL={}, newVL={}, OCEAN={}, DT={}, spotPrice={}',
|
2021-02-23 16:54:15 +01:00
|
|
|
[
|
|
|
|
pool.id,
|
|
|
|
oldValueLocked.toString(),
|
|
|
|
pool.valueLocked.toString(),
|
|
|
|
poolTx.oceanReserve.toString(),
|
|
|
|
poolTx.datatokenReserve.toString(),
|
|
|
|
pool.spotPrice.toString()
|
|
|
|
]
|
|
|
|
)
|
2021-02-09 11:13:31 +01:00
|
|
|
}
|
2021-05-13 08:17:54 +02:00
|
|
|
factory.totalValueLocked = factory.totalValueLocked
|
|
|
|
.minus(oldValueLocked)
|
|
|
|
.plus(pool.valueLocked)
|
2020-12-09 11:54:13 +01:00
|
|
|
|
2021-07-30 09:36:44 +02:00
|
|
|
const gStats: Global | null = getGlobalStats()
|
|
|
|
|
|
|
|
gStats.totalValueLocked = factory.totalValueLocked
|
|
|
|
gStats.save()
|
|
|
|
|
2020-12-09 11:54:13 +01:00
|
|
|
pool.transactionCount = pool.transactionCount.plus(BigInt.fromI32(1))
|
2020-12-08 13:32:15 +01:00
|
|
|
|
2020-12-03 10:05:34 +01:00
|
|
|
pool.save()
|
2021-01-18 16:24:36 +01:00
|
|
|
factory.save()
|
2020-12-09 11:54:13 +01:00
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
debuglog(
|
|
|
|
'updated pool reserves (source, dtBalance, ocnBalance, dtReserve, ocnReserve): ',
|
2020-12-03 10:05:34 +01:00
|
|
|
event,
|
2020-12-11 11:37:09 +01:00
|
|
|
[
|
|
|
|
'createPoolTransaction',
|
|
|
|
dtToken.balance.toString(),
|
|
|
|
ocnToken.balance.toString(),
|
|
|
|
pool.datatokenReserve.toString(),
|
|
|
|
pool.oceanReserve.toString()
|
|
|
|
]
|
|
|
|
)
|
2020-12-02 11:07:09 +01:00
|
|
|
|
|
|
|
poolTx.tx = event.transaction.hash
|
2020-12-11 11:37:09 +01:00
|
|
|
// eslint-disable-next-line camelcase
|
2020-12-02 11:07:09 +01:00
|
|
|
poolTx.event = event_type
|
|
|
|
poolTx.block = event.block.number.toI32()
|
|
|
|
poolTx.timestamp = event.block.timestamp.toI32()
|
|
|
|
poolTx.gasUsed = event.transaction.gasUsed.toBigDecimal()
|
|
|
|
poolTx.gasPrice = event.transaction.gasPrice.toBigDecimal()
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
debuglog('####################### poolTransaction: ', event, [
|
2021-03-01 13:41:36 +01:00
|
|
|
ptx,
|
2020-12-11 11:37:09 +01:00
|
|
|
BigInt.fromI32(poolTx.block).toString(),
|
|
|
|
BigInt.fromI32(poolTx.timestamp).toString(),
|
|
|
|
pool.oceanReserve.toString()
|
|
|
|
])
|
2020-12-08 17:57:09 +01:00
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
poolTx.save()
|
|
|
|
}
|
2020-11-20 13:12:02 +01:00
|
|
|
|
|
|
|
export function decrPoolCount(finalized: boolean): void {
|
2020-12-11 11:37:09 +01:00
|
|
|
const factory = PoolFactory.load('1')
|
2020-11-20 13:12:02 +01:00
|
|
|
factory.poolCount -= 1
|
|
|
|
if (finalized) factory.finalizedPoolCount -= 1
|
|
|
|
factory.save()
|
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function saveTokenTransaction(
|
|
|
|
event: ethereum.Event,
|
|
|
|
eventName: string
|
|
|
|
): void {
|
|
|
|
const tx = event.transaction.hash
|
|
|
|
.toHexString()
|
|
|
|
.concat('-')
|
|
|
|
.concat(event.logIndex.toString())
|
|
|
|
const userAddress = event.transaction.from.toHex()
|
2020-11-26 12:10:45 +01:00
|
|
|
let transaction = TokenTransaction.load(tx)
|
|
|
|
if (transaction == null) {
|
|
|
|
transaction = new TokenTransaction(tx)
|
|
|
|
}
|
|
|
|
transaction.event = eventName
|
|
|
|
transaction.datatokenAddress = event.address.toHex()
|
|
|
|
transaction.userAddress = userAddress
|
|
|
|
transaction.gasUsed = event.transaction.gasUsed.toBigDecimal()
|
|
|
|
transaction.gasPrice = event.transaction.gasPrice.toBigDecimal()
|
|
|
|
transaction.tx = event.transaction.hash
|
|
|
|
transaction.timestamp = event.block.timestamp.toI32()
|
|
|
|
transaction.block = event.block.number.toI32()
|
|
|
|
transaction.save()
|
|
|
|
|
|
|
|
createUserEntity(userAddress)
|
|
|
|
}
|
|
|
|
|
2020-12-11 11:37:09 +01:00
|
|
|
export function updateTokenBalance(
|
|
|
|
id: string,
|
|
|
|
token: string,
|
|
|
|
user: string,
|
|
|
|
amount: BigDecimal
|
|
|
|
): void {
|
2020-12-02 11:07:09 +01:00
|
|
|
let tokenBalance = TokenBalance.load(id)
|
|
|
|
if (tokenBalance == null) {
|
|
|
|
tokenBalance = new TokenBalance(id)
|
|
|
|
createUserEntity(user)
|
|
|
|
tokenBalance.userAddress = user
|
|
|
|
tokenBalance.datatokenId = token
|
|
|
|
tokenBalance.balance = ZERO_BD
|
|
|
|
}
|
2020-11-26 07:38:08 +01:00
|
|
|
|
2020-12-02 11:07:09 +01:00
|
|
|
tokenBalance.balance = tokenBalance.balance.plus(amount)
|
2020-11-26 07:38:08 +01:00
|
|
|
tokenBalance.save()
|
|
|
|
}
|