Skip to content

Commit

Permalink
common: check escrow balance before rav redeem
Browse files Browse the repository at this point in the history
Signed-off-by: Gustavo Inacio <gustavo@semiotic.ai>
  • Loading branch information
gusinacio committed Oct 10, 2024
1 parent a253dbf commit 0cb1056
Show file tree
Hide file tree
Showing 2 changed files with 90 additions and 1 deletion.
64 changes: 64 additions & 0 deletions packages/indexer-common/src/allocations/escrow-accounts.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
import { Address, toAddress } from '@graphprotocol/common-ts'
import { TAPSubgraph } from '../tap-subgraph'
import gql from 'graphql-tag'

type U256 = bigint

type EscrowAccountResponse = {
escrowAccounts: {
balance: string
sender: {
id: string
}
}[]
}

export class EscrowAccounts {
constructor(private sendersBalances: Map<Address, U256>) {}

getBalanceForSender(sender: Address): U256 {
const balance = this.sendersBalances.get(sender)
if (balance === undefined) {
throw new Error(`No balance found for sender: ${sender}`)
}
return balance
}

subtractSenderBalance(sender: Address, ravValue: U256) {
const balance = this.getBalanceForSender(sender)
const newBalance = balance - ravValue
this.sendersBalances.set(sender, newBalance)
}

static fromResponse(response: EscrowAccountResponse): EscrowAccounts {
const sendersBalances = new Map<Address, U256>()
response.escrowAccounts.forEach((account) => {
sendersBalances.set(toAddress(account.sender.id), BigInt(account.balance))
})

return new EscrowAccounts(sendersBalances)
}
}

export const getEscrowAccounts = async (
tapSubgraph: TAPSubgraph,
indexer: Address,
): Promise<EscrowAccounts> => {
const result = await tapSubgraph.query<EscrowAccountResponse>(
gql`
query EscrowAccountQuery($indexer: ID!) {
escrowAccounts(where: { receiver_: { id: $indexer } }) {
balance
sender {
id
}
}
}
`,
{ indexer },
)
if (!result.data) {
throw `There was an error while querying Tap Subgraph. Errors: ${result.error}`
}
return EscrowAccounts.fromResponse(result.data)
}
27 changes: 26 additions & 1 deletion packages/indexer-common/src/allocations/tap-collector.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ import pReduce from 'p-reduce'
import { TAPSubgraph } from '../tap-subgraph'
import { NetworkSubgraph, QueryResult } from '../network-subgraph'
import gql from 'graphql-tag'
import { getEscrowAccounts } from './escrow-accounts'

// every 15 minutes
const RAV_CHECK_INTERVAL_MS = 900_000
Expand Down Expand Up @@ -109,6 +110,7 @@ export class TapCollector {
declare tapSubgraph: TAPSubgraph
declare networkSubgraph: NetworkSubgraph
declare finalityTime: number
declare indexerAddress: Address

// eslint-disable-next-line @typescript-eslint/no-empty-function -- Private constructor to prevent direct instantiation
private constructor() {}
Expand Down Expand Up @@ -138,10 +140,11 @@ export class TapCollector {
collector.tapSubgraph = tapSubgraph
collector.networkSubgraph = networkSubgraph

const { voucherRedemptionThreshold, finalityTime } =
const { voucherRedemptionThreshold, finalityTime, address } =
networkSpecification.indexerOptions
collector.ravRedemptionThreshold = voucherRedemptionThreshold
collector.finalityTime = finalityTime
collector.indexerAddress = address

collector.logger.info(`RAV processing is initiated`)
collector.startRAVProcessing()
Expand Down Expand Up @@ -531,10 +534,28 @@ export class TapCollector {
logger.info(`Redeem last RAVs on chain individually`, {
signedRavs,
})
const escrowAccounts = await getEscrowAccounts(this.tapSubgraph, this.indexerAddress)

// Redeem RAV one-by-one as no plual version available
for (const { rav: signedRav, allocation, sender } of signedRavs) {
const { rav } = signedRav

// verify escrow balances
const ravValue = BigInt(rav.valueAggregate.toString())
const senderBalance = escrowAccounts.getBalanceForSender(sender)
if (senderBalance < ravValue) {
this.logger.warn(
'RAV was not sent to the blockchain \
because its value aggregate is lower than escrow balance.',
{
rav,
sender,
senderBalance,
},
)
continue
}

const stopTimer = this.metrics.ravsRedeemDuration.startTimer({
allocation: rav.allocationId,
})
Expand Down Expand Up @@ -565,6 +586,10 @@ export class TapCollector {
this.metrics.ravRedeemsInvalid.inc({ allocation: rav.allocationId })
return
}
// subtract from the escrow account
// THIS IS A MUT OPERATION
escrowAccounts.subtractSenderBalance(sender, ravValue)

this.metrics.ravCollectedFees.set(
{ allocation: rav.allocationId },
parseFloat(rav.valueAggregate.toString()),
Expand Down

0 comments on commit 0cb1056

Please sign in to comment.