From 18c1d6909d153d71f442e625356a4de9c8636895 Mon Sep 17 00:00:00 2001
From: Fuxing Loh <4266087+fuxingloh@users.noreply.github.com>
Date: Fri, 4 Mar 2022 17:50:09 +0800
Subject: [PATCH] feature(controller): Resolve PoolSwap from/to from
`AccountHistory` (#824)
* dex price resolution
* deprecated no Data postfix interface
* fix import
* optimize poolpair.service.ts
* add new verbose option
---
.idea/dictionaries/fuxing.xml | 1 +
.../__tests__/api/poolpairs.test.ts | 100 +++++++++++++++--
.../whale-api-client/src/api/poolpairs.ts | 49 +++++++-
src/module.api/poolpair.controller.ts | 41 +++++--
src/module.api/poolpair.service.ts | 105 +++++++++++++++++-
5 files changed, 271 insertions(+), 25 deletions(-)
diff --git a/.idea/dictionaries/fuxing.xml b/.idea/dictionaries/fuxing.xml
index b831359a1..ab9d93f85 100644
--- a/.idea/dictionaries/fuxing.xml
+++ b/.idea/dictionaries/fuxing.xml
@@ -135,6 +135,7 @@
pooledtx
poolpair
poolpairs
+ poolswap
previousblockhash
prevout
prevouts
diff --git a/packages/whale-api-client/__tests__/api/poolpairs.test.ts b/packages/whale-api-client/__tests__/api/poolpairs.test.ts
index 9c0f4e18c..d5da08c53 100644
--- a/packages/whale-api-client/__tests__/api/poolpairs.test.ts
+++ b/packages/whale-api-client/__tests__/api/poolpairs.test.ts
@@ -3,7 +3,7 @@ import { StubWhaleApiClient } from '../stub.client'
import { StubService } from '../stub.service'
import { ApiPagedResponse, WhaleApiClient, WhaleApiException } from '../../src'
import { addPoolLiquidity, createPoolPair, createToken, getNewAddress, mintTokens, poolSwap } from '@defichain/testing'
-import { PoolPairData, PoolSwap, PoolSwapAggregated, PoolSwapAggregatedInterval } from '../../src/api/poolpairs'
+import { PoolPairData, PoolSwapAggregatedData, PoolSwapAggregatedInterval, PoolSwapData } from '../../src/api/poolpairs'
import { Testing } from '@defichain/jellyfish-testing'
let container: MasterNodeRegTestContainer
@@ -371,13 +371,95 @@ describe('poolswap', () => {
await container.generate(1)
await service.waitForIndexedHeight(height)
- const response: ApiPagedResponse = await client.poolpairs.listPoolSwaps('9')
- expect(response.length).toStrictEqual(2)
+ const response: ApiPagedResponse = await client.poolpairs.listPoolSwaps('9')
expect(response.hasNext).toStrictEqual(false)
- expect(response[0].fromAmount).toStrictEqual('50.00000000')
- expect(response[1].fromAmount).toStrictEqual('25.00000000')
- expect(response[0].fromTokenId).toStrictEqual(1)
- expect(response[1].fromTokenId).toStrictEqual(1)
+ expect([...response]).toStrictEqual([
+ {
+ id: expect.any(String),
+ txid: expect.stringMatching(/[0-f]{64}/),
+ txno: 1,
+ poolPairId: '9',
+ sort: expect.any(String),
+ fromAmount: '50.00000000',
+ fromTokenId: 1,
+ block: {
+ hash: expect.stringMatching(/[0-f]{64}/),
+ height: expect.any(Number),
+ time: expect.any(Number),
+ medianTime: expect.any(Number)
+ }
+ },
+ {
+ id: expect.any(String),
+ txid: expect.stringMatching(/[0-f]{64}/),
+ txno: 3,
+ poolPairId: '9',
+ sort: expect.any(String),
+ fromAmount: '25.00000000',
+ fromTokenId: 1,
+ block: {
+ hash: expect.stringMatching(/[0-f]{64}/),
+ height: expect.any(Number),
+ time: expect.any(Number),
+ medianTime: expect.any(Number)
+ }
+ }
+ ])
+
+ const verbose: ApiPagedResponse = await client.poolpairs.listPoolSwapsVerbose('9')
+ expect(verbose.hasNext).toStrictEqual(false)
+ expect([...verbose]).toStrictEqual([
+ {
+ id: expect.any(String),
+ txid: expect.stringMatching(/[0-f]{64}/),
+ txno: 1,
+ poolPairId: '9',
+ sort: expect.any(String),
+ fromAmount: '50.00000000',
+ fromTokenId: 1,
+ block: {
+ hash: expect.stringMatching(/[0-f]{64}/),
+ height: expect.any(Number),
+ time: expect.any(Number),
+ medianTime: expect.any(Number)
+ },
+ from: {
+ address: expect.any(String),
+ symbol: 'A',
+ amount: '50.00000000'
+ },
+ to: {
+ address: expect.any(String),
+ amount: '45.71428571',
+ symbol: 'DFI'
+ }
+ },
+ {
+ id: expect.any(String),
+ txid: expect.stringMatching(/[0-f]{64}/),
+ txno: 3,
+ poolPairId: '9',
+ sort: expect.any(String),
+ fromAmount: '25.00000000',
+ fromTokenId: 1,
+ block: {
+ hash: expect.stringMatching(/[0-f]{64}/),
+ height: expect.any(Number),
+ time: expect.any(Number),
+ medianTime: expect.any(Number)
+ },
+ from: {
+ address: expect.any(String),
+ symbol: 'A',
+ amount: '25.00000000'
+ },
+ to: {
+ address: expect.any(String),
+ amount: '39.99999999',
+ symbol: 'DFI'
+ }
+ }
+ ])
const poolPair: PoolPairData = await client.poolpairs.get('9')
expect(poolPair).toStrictEqual({
@@ -594,7 +676,7 @@ describe('poolswap aggregated', () => {
await service.waitForIndexedHeight(height)
}
- const dayAggregated: ApiPagedResponse = await client.poolpairs.listPoolSwapAggregates('10', PoolSwapAggregatedInterval.ONE_DAY, 10)
+ const dayAggregated: ApiPagedResponse = await client.poolpairs.listPoolSwapAggregates('10', PoolSwapAggregatedInterval.ONE_DAY, 10)
expect([...dayAggregated]).toStrictEqual([
{
aggregated: {
@@ -631,7 +713,7 @@ describe('poolswap aggregated', () => {
])
- const hourAggregated: ApiPagedResponse = await client.poolpairs.listPoolSwapAggregates('10', PoolSwapAggregatedInterval.ONE_HOUR, 3)
+ const hourAggregated: ApiPagedResponse = await client.poolpairs.listPoolSwapAggregates('10', PoolSwapAggregatedInterval.ONE_HOUR, 3)
expect([...hourAggregated]).toStrictEqual([
{
aggregated: {
diff --git a/packages/whale-api-client/src/api/poolpairs.ts b/packages/whale-api-client/src/api/poolpairs.ts
index 8de76fe40..b71a68679 100644
--- a/packages/whale-api-client/src/api/poolpairs.ts
+++ b/packages/whale-api-client/src/api/poolpairs.ts
@@ -35,12 +35,24 @@ export class PoolPairs {
* @param {string} id poolpair id
* @param {number} size of PoolSwap to query
* @param {string} next set of PoolSwap
- * @return {Promise>}
+ * @return {Promise>}
*/
- async listPoolSwaps (id: string, size: number = 30, next?: string): Promise> {
+ async listPoolSwaps (id: string, size: number = 30, next?: string): Promise> {
return await this.client.requestList('GET', `poolpairs/${id}/swaps`, size, next)
}
+ /**
+ * List pool swaps with from/to
+ *
+ * @param {string} id poolpair id
+ * @param {number} [size=10] of PoolSwap to query, max of 20 per page
+ * @param {string} next set of PoolSwap
+ * @return {Promise>}
+ */
+ async listPoolSwapsVerbose (id: string, size: number = 10, next?: string): Promise> {
+ return await this.client.requestList('GET', `poolpairs/${id}/swaps/verbose`, size, next)
+ }
+
/**
* List pool swap aggregates
*
@@ -48,9 +60,9 @@ export class PoolPairs {
* @param {PoolSwapAggregatedInterval} interval interval
* @param {number} size of PoolSwap to query
* @param {string} next set of PoolSwap
- * @return {Promise>}
+ * @return {Promise>}
*/
- async listPoolSwapAggregates (id: string, interval: PoolSwapAggregatedInterval, size: number = 30, next?: string): Promise> {
+ async listPoolSwapAggregates (id: string, interval: PoolSwapAggregatedInterval, size: number = 30, next?: string): Promise> {
return await this.client.requestList('GET', `poolpairs/${id}/swaps/aggregate/${interval as number}`, size, next)
}
}
@@ -103,7 +115,17 @@ export interface PoolPairData {
}
}
-export interface PoolSwap {
+/**
+ * @deprecated use PoolSwapData instead
+ */
+export type PoolSwap = PoolSwapData
+
+/**
+ * @deprecated use PoolSwapAggregatedData instead
+ */
+export type PoolSwapAggregated = PoolSwapAggregatedData
+
+export interface PoolSwapData {
id: string
sort: string
txid: string
@@ -113,6 +135,15 @@ export interface PoolSwap {
fromAmount: string
fromTokenId: number
+ /**
+ * To handle for optional value as Whale service might fail to resolve when indexing
+ */
+ from?: PoolSwapFromToData
+ /**
+ * To handle for optional value as Whale service might fail to resolve when indexing
+ */
+ to?: PoolSwapFromToData
+
block: {
hash: string
height: number
@@ -121,7 +152,13 @@ export interface PoolSwap {
}
}
-export interface PoolSwapAggregated {
+export interface PoolSwapFromToData {
+ address: string
+ amount: string
+ symbol: string
+}
+
+export interface PoolSwapAggregatedData {
id: string
key: string
bucket: number
diff --git a/src/module.api/poolpair.controller.ts b/src/module.api/poolpair.controller.ts
index 091e49024..e9f15386b 100644
--- a/src/module.api/poolpair.controller.ts
+++ b/src/module.api/poolpair.controller.ts
@@ -2,7 +2,7 @@ import { Controller, Get, NotFoundException, Param, ParseIntPipe, Query } from '
import { JsonRpcClient } from '@defichain/jellyfish-api-jsonrpc'
import { ApiPagedResponse } from '@src/module.api/_core/api.paged.response'
import { DeFiDCache } from '@src/module.api/cache/defid.cache'
-import { PoolPairData, PoolSwap, PoolSwapAggregated } from '@whale-api-client/api/poolpairs'
+import { PoolPairData, PoolSwapAggregatedData, PoolSwapData } from '@whale-api-client/api/poolpairs'
import { PaginationQuery } from '@src/module.api/_core/api.query'
import { PoolPairService } from './poolpair.service'
import BigNumber from 'bignumber.js'
@@ -83,9 +83,35 @@ export class PoolPairController {
async listPoolSwaps (
@Param('id', ParseIntPipe) id: string,
@Query() query: PaginationQuery
- ): Promise> {
- const result = await this.poolSwapMapper.query(id, query.size, query.next)
- return ApiPagedResponse.of(result, query.size, item => {
+ ): Promise> {
+ const items = await this.poolSwapMapper.query(id, query.size, query.next)
+ return ApiPagedResponse.of(items, query.size, item => {
+ return item.sort
+ })
+ }
+
+ /**
+ * @param {string} id poolpair id
+ * @param {PaginationQuery} query with size restricted to 20
+ * @param {number} query.size
+ * @param {string} [query.next]
+ * @return {Promise>}
+ */
+ @Get('/:id/swaps/verbose')
+ async listPoolSwapsVerbose (
+ @Param('id', ParseIntPipe) id: string,
+ @Query() query: PaginationQuery
+ ): Promise> {
+ query.size = query.size > 20 ? 20 : query.size
+ const items: PoolSwapData[] = await this.poolSwapMapper.query(id, query.size, query.next)
+
+ for (const swap of items) {
+ const fromTo = await this.poolPairService.findSwapFromTo(swap.block.height, swap.txid, swap.txno)
+ swap.from = fromTo?.from
+ swap.to = fromTo?.to
+ }
+
+ return ApiPagedResponse.of(items, query.size, item => {
return item.sort
})
}
@@ -106,10 +132,10 @@ export class PoolPairController {
@Param('id', ParseIntPipe) id: string,
@Param('interval', ParseIntPipe) interval: string,
@Query() query: PaginationQuery
- ): Promise> {
+ ): Promise> {
const lt = query.next === undefined ? undefined : parseInt(query.next)
const aggregates = await this.poolSwapAggregatedMapper.query(`${id}-${interval}`, query.size, lt)
- const mapped: Array> = aggregates.map(async value => {
+ const mapped: Array> = aggregates.map(async value => {
return {
...value,
aggregated: {
@@ -126,8 +152,7 @@ export class PoolPairController {
}
}
-function mapPoolPair (id: string, info: PoolPairInfo, totalLiquidityUsd?: BigNumber, apr?: PoolPairData['apr'],
- volume?: PoolPairData['volume']): PoolPairData {
+function mapPoolPair (id: string, info: PoolPairInfo, totalLiquidityUsd?: BigNumber, apr?: PoolPairData['apr'], volume?: PoolPairData['volume']): PoolPairData {
const [symbolA, symbolB] = info.symbol.split('-')
return {
diff --git a/src/module.api/poolpair.service.ts b/src/module.api/poolpair.service.ts
index 20cfb988e..cfe3a1048 100644
--- a/src/module.api/poolpair.service.ts
+++ b/src/module.api/poolpair.service.ts
@@ -1,21 +1,38 @@
-import { Injectable, NotFoundException } from '@nestjs/common'
+import { Inject, Injectable, NotFoundException } from '@nestjs/common'
import { JsonRpcClient } from '@defichain/jellyfish-api-jsonrpc'
import BigNumber from 'bignumber.js'
import { PoolPairInfo } from '@defichain/jellyfish-api-core/dist/category/poolpair'
import { SemaphoreCache } from '@src/module.api/cache/semaphore.cache'
-import { PoolPairData } from '@whale-api-client/api/poolpairs'
+import { PoolPairData, PoolSwapFromToData } from '@whale-api-client/api/poolpairs'
import { getBlockSubsidy } from '@src/module.api/subsidy'
import { BlockMapper } from '@src/module.model/block'
import { TokenMapper } from '@src/module.model/token'
import { PoolSwapAggregated, PoolSwapAggregatedMapper } from '@src/module.model/pool.swap.aggregated'
import { PoolSwapAggregatedInterval } from '@src/module.indexer/model/dftx/pool.swap.aggregated'
+import { TransactionVout, TransactionVoutMapper } from '@src/module.model/transaction.vout'
+import { SmartBuffer } from 'smart-buffer'
+import {
+ CCompositeSwap,
+ CompositeSwap,
+ CPoolSwap,
+ OP_DEFI_TX,
+ PoolSwap as PoolSwapDfTx,
+ toOPCodes
+} from '@defichain/jellyfish-transaction'
+import { fromScript } from '@defichain/jellyfish-address'
+import { NetworkName } from '@defichain/jellyfish-network'
+import { AccountHistory } from '@defichain/jellyfish-api-core/dist/category/account'
+import { DeFiDCache } from '@src/module.api/cache/defid.cache'
@Injectable()
export class PoolPairService {
constructor (
+ @Inject('NETWORK') protected readonly network: NetworkName,
protected readonly rpcClient: JsonRpcClient,
+ protected readonly deFiDCache: DeFiDCache,
protected readonly cache: SemaphoreCache,
protected readonly poolSwapAggregatedMapper: PoolSwapAggregatedMapper,
+ protected readonly voutMapper: TransactionVoutMapper,
protected readonly tokenMapper: TokenMapper,
protected readonly blockMapper: BlockMapper
) {
@@ -202,6 +219,40 @@ export class PoolPairService {
return value
}
+ public async findSwapFromTo (height: number, txid: string, txno: number): Promise<{ from?: PoolSwapFromToData, to?: PoolSwapFromToData } | undefined> {
+ const vouts = await this.voutMapper.query(txid, 1)
+ const dftx = findPoolSwapDfTx(vouts)
+ if (dftx === undefined) {
+ return undefined
+ }
+
+ const fromAddress = fromScript(dftx.fromScript, this.network)?.address
+ const fromToken = await this.deFiDCache.getTokenInfo(dftx.fromTokenId.toString())
+
+ const toAddress = fromScript(dftx.toScript, this.network)?.address
+ if (fromAddress === undefined || toAddress === undefined || fromToken === undefined) {
+ return undefined
+ }
+
+ const history = await this.getAccountHistory(toAddress, height, txno)
+ if (history === undefined) {
+ return undefined
+ }
+
+ return {
+ from: {
+ address: fromAddress,
+ symbol: fromToken.symbol,
+ amount: dftx.fromAmount.toFixed(8)
+ },
+ to: findPoolSwapFromTo(history, false)
+ }
+ }
+
+ private async getAccountHistory (address: string, height: number, txno: number): Promise {
+ return await this.rpcClient.account.getAccountHistory(address, height, txno)
+ }
+
private async getLoanTokenSplits (): Promise | undefined> {
return await this.cache.get>('LP_LOAN_TOKEN_SPLITS', async () => {
const result = await this.rpcClient.masternode.getGov('LP_LOAN_TOKEN_SPLITS')
@@ -324,3 +375,53 @@ export class PoolPairService {
}
}
}
+
+function findPoolSwapDfTx (vouts: TransactionVout[]): PoolSwapDfTx | undefined {
+ const hex = vouts[0].script.hex
+ const buffer = SmartBuffer.fromBuffer(Buffer.from(hex, 'hex'))
+ const stack = toOPCodes(buffer)
+ if (stack.length !== 2 || stack[1].type !== 'OP_DEFI_TX') {
+ return undefined
+ }
+
+ const dftx = (stack[1] as OP_DEFI_TX).tx
+ if (dftx === undefined) {
+ return undefined
+ }
+
+ switch (dftx.name) {
+ case CPoolSwap.OP_NAME:
+ return (dftx.data as PoolSwapDfTx)
+
+ case CCompositeSwap.OP_NAME:
+ return (dftx.data as CompositeSwap).poolSwap
+
+ default:
+ return undefined
+ }
+}
+
+function findPoolSwapFromTo (history: AccountHistory, from: boolean): PoolSwapFromToData | undefined {
+ for (const amount of history.amounts) {
+ const [value, symbol] = amount.split('@')
+ const isNegative = value.startsWith('-')
+
+ if (isNegative && from) {
+ return {
+ address: history.owner,
+ amount: new BigNumber(value).absoluteValue().toFixed(8),
+ symbol: symbol
+ }
+ }
+
+ if (!isNegative && !from) {
+ return {
+ address: history.owner,
+ amount: new BigNumber(value).absoluteValue().toFixed(8),
+ symbol: symbol
+ }
+ }
+ }
+
+ return undefined
+}