Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

improve(relayer): Batch query deposit fill status #1338

Merged
merged 27 commits into from
Mar 22, 2024
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
5150ff8
refactor(relayer): Factor out MDC computation
pxrl Mar 20, 2024
44bdb9a
refactor(relayer): Defer deposit MDCs check
pxrl Mar 20, 2024
c34ba19
Merge branch 'master' into pxrl/mdcRefactor
pxrl Mar 20, 2024
244127d
Merge branch 'pxrl/mdcRefactor' into pxrl/deferMDCCheck
pxrl Mar 20, 2024
bfef865
refactor(relayer): Relocate profitability & fill execution
pxrl Mar 20, 2024
489bc6f
simplify
pxrl Mar 20, 2024
55df690
Merge remote-tracking branch 'origin/pxrl/deferMDCCheck' into pxrl/re…
pxrl Mar 20, 2024
78eb55b
lint
pxrl Mar 20, 2024
4e618bb
Exit early
pxrl Mar 20, 2024
9e13803
fixes
pxrl Mar 20, 2024
1df1757
Merge remote-tracking branch 'origin/pxrl/deferMDCCheck' into pxrl/re…
pxrl Mar 20, 2024
40114d6
Merge remote-tracking branch 'origin/master' into pxrl/relayerRefacto…
pxrl Mar 21, 2024
0b3ccf8
Restore logging
pxrl Mar 21, 2024
89cf27b
Merge branch 'master' into pxrl/relayerRefactorEvaluate
pxrl Mar 21, 2024
5a246b9
Merge branch 'master' into pxrl/relayerRefactorEvaluate
pxrl Mar 21, 2024
3eb1877
improve(relayer): Batch query deposit fill status
pxrl Mar 21, 2024
4816ba4
Merge branch 'master' into pxrl/relayerRefactorEvaluate
pxrl Mar 21, 2024
fe7cbc2
Merge branch 'pxrl/relayerRefactorEvaluate' into pxrl/batchFillStatus
pxrl Mar 21, 2024
bf892e2
Improve test
pxrl Mar 21, 2024
05c7bdd
lint
pxrl Mar 21, 2024
8c7b76d
Tweak filter
pxrl Mar 21, 2024
da509bb
Restore comment
pxrl Mar 21, 2024
b72b550
Fix test
pxrl Mar 21, 2024
9d9225f
Remove redundant Object.values()
pxrl Mar 21, 2024
5d76888
Merge branch 'pxrl/relayerRefactorEvaluate' into pxrl/batchFillStatus
pxrl Mar 21, 2024
29e1bb8
Merge branch 'master' into pxrl/batchFillStatus
pxrl Mar 22, 2024
2a79c3a
Bump sdk & contracts
pxrl Mar 22, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
168 changes: 81 additions & 87 deletions src/relayer/Relayer.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { utils as sdkUtils } from "@across-protocol/sdk-v2";
import { utils as ethersUtils } from "ethers";
import { FillStatus, L1Token, V3Deposit, V3DepositWithBlock } from "../interfaces";
import { L1Token, V3Deposit, V3DepositWithBlock } from "../interfaces";
import {
BigNumber,
bnZero,
Expand Down Expand Up @@ -128,17 +128,6 @@ export class Relayer {
return false;
}

const destSpokePool = this.clients.spokePoolClients[destinationChainId].spokePool;
const fillStatus = await sdkUtils.relayFillStatus(destSpokePool, deposit, "latest", destinationChainId);
if (fillStatus === FillStatus.Filled) {
this.logger.debug({
at: "Relayer::getUnfilledDeposits",
message: "Skipping deposit that was already filled.",
deposit,
});
return false;
}

// Skip deposit with message if sending fills with messages is not supported.
if (!this.config.sendingMessageRelaysEnabled && !isMessageEmpty(resolveDepositMessage(deposit))) {
this.logger.warn({
Expand Down Expand Up @@ -244,13 +233,88 @@ export class Relayer {
mdcPerChain,
minDepositConfirmations,
});

return mdcPerChain;
}

// Iterate over all unfilled deposits. For each unfilled deposit, check that:
// a) it exceeds the minimum number of required block confirmations,
// b) the token balance client has enough tokens to fill it,
// c) the fill is profitable.
// If all hold true then complete the fill. Otherwise, if slow fills are enabled, request a slow fill.
async evaluateFill(deposit: V3DepositWithBlock, maxBlockNumber: number, sendSlowRelays: boolean): Promise<void> {
const { depositId, depositor, recipient, destinationChainId, originChainId, inputToken, outputAmount } = deposit;
const { hubPoolClient, profitClient, tokenClient } = this.clients;
const { slowDepositors } = this.config;

// If the deposit does not meet the minimum number of block confirmations, skip it.
if (deposit.blockNumber > maxBlockNumber) {
const chain = getNetworkName(originChainId);
this.logger.debug({
at: "Relayer",
message: `Skipping ${chain} deposit ${depositId} due to insufficient deposit confirmations.`,
depositId,
blockNumber: deposit.blockNumber,
maxBlockNumber,
transactionHash: deposit.transactionHash,
});
return;
}

// If depositor is on the slow deposit list, then send a zero fill to initiate a slow relay and return early.
if (slowDepositors?.includes(depositor)) {
if (sendSlowRelays) {
this.logger.debug({
at: "Relayer",
message: "Initiating slow fill for grey listed depositor",
depositor,
});
this.requestSlowFill(deposit);
}
// Regardless of whether we should send a slow fill or not for this depositor, exit early at this point
// so we don't fast fill an already slow filled deposit from the slow fill-only list.
return;
}

const l1Token = hubPoolClient.getL1TokenInfoForL2Token(inputToken, originChainId);
const selfRelay = [depositor, recipient].every((address) => address === this.relayerAddress);
if (tokenClient.hasBalanceForFill(deposit, outputAmount) && !selfRelay) {
const {
repaymentChainId,
realizedLpFeePct,
relayerFeePct,
gasLimit: _gasLimit,
gasCost,
} = await this.resolveRepaymentChain(deposit, l1Token);
if (isDefined(repaymentChainId)) {
const gasLimit = isMessageEmpty(resolveDepositMessage(deposit)) ? undefined : _gasLimit;
this.fillRelay(deposit, repaymentChainId, realizedLpFeePct, gasLimit);
} else {
profitClient.captureUnprofitableFill(deposit, realizedLpFeePct, relayerFeePct, gasCost);
}
} else if (selfRelay) {
const { realizedLpFeePct } = await hubPoolClient.computeRealizedLpFeePct({
...deposit,
paymentChainId: destinationChainId,
});

// A relayer can fill its own deposit without an ERC20 transfer. Only bypass profitability requirements if the
// relayer is both the depositor and the recipient, because a deposit on a cheap SpokePool chain could cause
// expensive fills on (for example) mainnet.
this.fillRelay(deposit, destinationChainId, realizedLpFeePct);
} else {
// TokenClient.getBalance returns that we don't have enough balance to submit the fast fill.
// At this point, capture the shortfall so that the inventory manager can rebalance the token inventory.
tokenClient.captureTokenShortfallForFill(deposit, outputAmount);
if (sendSlowRelays) {
this.requestSlowFill(deposit);
}
}
}

async checkForUnfilledDepositsAndFill(sendSlowRelays = true): Promise<void> {
// Fetch all unfilled deposits, order by total earnable fee.
const { config } = this;
const { hubPoolClient, profitClient, spokePoolClients, tokenClient, multiCallerClient } = this.clients;
const { profitClient, spokePoolClients, tokenClient, multiCallerClient } = this.clients;

// Flush any pre-existing enqueued transactions that might not have been executed.
multiCallerClient.clearTransactionQueue();
Expand All @@ -268,82 +332,12 @@ export class Relayer {
}

const mdcPerChain = this.computeRequiredDepositConfirmations(allUnfilledDeposits);

// Iterate over all unfilled deposits. For each unfilled deposit, check that:
// a) it exceeds the minimum number of required block confirmations,
// b) the token balance client has enough tokens to fill it,
// c) the fill is profitable.
// If all hold true then complete the fill. If there is insufficient balance to complete the fill and slow fills are
// enabled then request a slow fill instead.
const { slowDepositors } = config;
for (const deposit of allUnfilledDeposits) {
const { depositId, depositor, recipient, destinationChainId, originChainId, inputToken, outputAmount } = deposit;

// If the deposit does not meet the minimum number of block confirmations, skip it.
const { originChainId } = deposit;
const maxBlockNumber = spokePoolClients[originChainId].latestBlockSearched - mdcPerChain[originChainId];
if (deposit.blockNumber > maxBlockNumber) {
const chain = getNetworkName(originChainId);
this.logger.debug({
at: "Relayer#checkForUnfilledDepositsAndFill",
message: `Skipping ${chain} deposit ${depositId} due to insufficient deposit confirmations.`,
depositId,
blockNumber: deposit.blockNumber,
maxBlockNumber,
transactionHash: deposit.transactionHash,
});
continue;
}

// If depositor is on the slow deposit list, then send a zero fill to initiate a slow relay and return early.
if (slowDepositors?.includes(depositor)) {
if (sendSlowRelays) {
this.logger.debug({
at: "Relayer#checkForUnfilledDepositsAndFill",
message: "Initiating slow fill for grey listed depositor",
depositor,
});
this.requestSlowFill(deposit);
}
// Regardless of whether we should send a slow fill or not for this depositor, exit early at this point
// so we don't fast fill an already slow filled deposit from the slow fill-only list.
continue;
}

const l1Token = hubPoolClient.getL1TokenInfoForL2Token(inputToken, originChainId);
const selfRelay = [depositor, recipient].every((address) => address === this.relayerAddress);
if (tokenClient.hasBalanceForFill(deposit, outputAmount) && !selfRelay) {
const {
repaymentChainId,
realizedLpFeePct,
relayerFeePct,
gasLimit: _gasLimit,
gasCost,
} = await this.resolveRepaymentChain(deposit, l1Token);
if (isDefined(repaymentChainId)) {
const gasLimit = isMessageEmpty(resolveDepositMessage(deposit)) ? undefined : _gasLimit;
this.fillRelay(deposit, repaymentChainId, realizedLpFeePct, gasLimit);
} else {
profitClient.captureUnprofitableFill(deposit, realizedLpFeePct, relayerFeePct, gasCost);
}
} else if (selfRelay) {
const { realizedLpFeePct } = await hubPoolClient.computeRealizedLpFeePct({
...deposit,
paymentChainId: destinationChainId,
});

// A relayer can fill its own deposit without an ERC20 transfer. Only bypass profitability requirements if the
// relayer is both the depositor and the recipient, because a deposit on a cheap SpokePool chain could cause
// expensive fills on (for example) mainnet.
this.fillRelay(deposit, destinationChainId, realizedLpFeePct);
} else {
// TokenClient.getBalance returns that we don't have enough balance to submit the fast fill.
// At this point, capture the shortfall so that the inventory manager can rebalance the token inventory.
tokenClient.captureTokenShortfallForFill(deposit, outputAmount);
if (sendSlowRelays) {
this.requestSlowFill(deposit);
}
}
await this.evaluateFill(deposit, maxBlockNumber, sendSlowRelays);
}

// If during the execution run we had shortfalls or unprofitable fills then handel it by producing associated logs.
if (tokenClient.anyCapturedShortFallFills()) {
this.handleTokenShortfall();
Expand Down
61 changes: 35 additions & 26 deletions src/utils/FillUtils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import { HubPoolClient, SpokePoolClient } from "../clients";
import {
Fill,
FillsToRefund,
FillStatus,
FillWithBlock,
SpokePoolClientsByChain,
V2DepositWithBlock,
Expand Down Expand Up @@ -258,11 +259,12 @@ export function getFillsInRange(
// @todo Better alignment with the upstream UnfilledDeposit type.
export type RelayerUnfilledDeposit = {
deposit: V3DepositWithBlock;
fillStatus: number;
version: number;
invalidFills: Fill[];
};

// @description Returns an array of unfilled deposits over all spokePoolClients.
// @description Returns all unfilled deposits, indexed by destination chain.
// @param spokePoolClients Mapping of chainIds to SpokePoolClient objects.
// @param configStoreClient ConfigStoreClient instance.
// @param depositLookBack Deposit lookback (in seconds) since SpokePoolClient time as at last update.
Expand Down Expand Up @@ -291,34 +293,41 @@ export async function getUnfilledDeposits(
}

// Iterate over each chainId and check for unfilled deposits.
chainIds.forEach((destinationChainId) => {
const destinationClient = spokePoolClients[destinationChainId];
await sdkUtils.mapAsync(
chainIds,
async (destinationChainId) => {
const destinationClient = spokePoolClients[destinationChainId];

unfilledDeposits[destinationChainId] = chainIds
.filter((chainId) => chainId !== destinationChainId)
.map((originChainId) => {
const originClient = spokePoolClients[originChainId];
const earliestBlockNumber = originFromBlocks[originChainId];
// For each destination chain, query each _other_ SpokePool for deposits within the lookback.
const deposits = chainIds
.filter((chainId) => chainId !== destinationChainId)
.map((originChainId) => {
const originClient = spokePoolClients[originChainId];
const earliestBlockNumber = originFromBlocks[originChainId];
const { deploymentBlock, latestBlockSearched } = originClient;

// Basic sanity check...
assert(
earliestBlockNumber >= originClient.deploymentBlock && earliestBlockNumber <= originClient.latestBlockSearched
);
// Basic sanity check...
assert(earliestBlockNumber >= deploymentBlock && earliestBlockNumber <= latestBlockSearched);

// Find all unfilled deposits for the current loops originChain -> destinationChain. Note that this also
// validates that the deposit is filled "correctly" for the given deposit information. This includes validation
// of the all deposit -> relay props, the realizedLpFeePct and the origin->destination token mapping.
return originClient
.getDepositsForDestinationChain(destinationChainId)
.filter((deposit) => deposit.blockNumber >= earliestBlockNumber)
.filter(sdkUtils.isV3Deposit<V3DepositWithBlock, V2DepositWithBlock>) // @todo: Remove after v2 deprecated.
.map((deposit) => {
const version = hubPoolClient.configStoreClient.getConfigStoreVersionForTimestamp(deposit.quoteTimestamp);
return { ...destinationClient.getValidUnfilledAmountForDeposit(deposit), deposit, version };
})
.filter(({ unfilledAmount }) => unfilledAmount.gt(bnZero));
})
.flat();
// Find all unfilled deposits for the current loops originChain -> destinationChain.
return originClient
.getDepositsForDestinationChain(destinationChainId)
.filter((deposit) => deposit.blockNumber >= earliestBlockNumber)
.filter(sdkUtils.isV3Deposit<V3DepositWithBlock, V2DepositWithBlock>); // @todo: Remove after v2 deprecated.
})
.flat();

// Resolve the latest fill status for each deposit and filter out any that are now filled.
const { spokePool } = destinationClient;
const fillStatus = await sdkUtils.fillStatusArray(spokePool, deposits);
pxrl marked this conversation as resolved.
Show resolved Hide resolved
unfilledDeposits[destinationChainId] = deposits
.map((deposit, idx) => ({ deposit, fillStatus: fillStatus[idx] }))
.filter((_, idx) => fillStatus[idx] !== FillStatus.Filled)
.map(({ deposit, fillStatus }) => {
const version = hubPoolClient.configStoreClient.getConfigStoreVersionForTimestamp(deposit.quoteTimestamp);
const { invalidFills } = destinationClient.getValidUnfilledAmountForDeposit(deposit);
return { deposit, version, fillStatus, invalidFills };
});
});

return unfilledDeposits;
Expand Down
31 changes: 30 additions & 1 deletion test/Relayer.BasicFill.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { clients, constants, utils as sdkUtils } from "@across-protocol/sdk-v2";
import { AcrossApiClient, ConfigStoreClient, MultiCallerClient, TokenClient } from "../src/clients";
import { V2FillWithBlock, V3FillWithBlock } from "../src/interfaces";
import { CONFIG_STORE_VERSION } from "../src/common";
import { bnOne } from "../src/utils";
import { bnOne, getUnfilledDeposits } from "../src/utils";
import { Relayer } from "../src/relayer/Relayer";
import { RelayerConfig } from "../src/relayer/RelayerConfig"; // Tested
import {
Expand All @@ -28,6 +28,7 @@ import {
enableRoutesOnHubPool,
ethers,
expect,
fillV3Relay,
getLastBlockTime,
getV3RelayHash,
lastSpyLogIncludes,
Expand Down Expand Up @@ -235,6 +236,34 @@ describe("Relayer: Check for Unfilled Deposits and Fill", async function () {
expect(multiCallerClient.transactionCount()).to.equal(0); // no new transactions were enqueued.
});

it("Queries the latest onchain fill status for all deposits", async function () {
const deposit = await depositV3(
spokePool_1,
destinationChainId,
depositor,
inputToken,
inputAmount,
outputToken,
outputAmount
);
await updateAllClients();
let unfilledDeposits = await getUnfilledDeposits(spokePoolClients, hubPoolClient);
expect(Object.values(unfilledDeposits).flat().length).to.equal(1);

await relayerInstance.checkForUnfilledDepositsAndFill();
expect(lastSpyLogIncludes(spy, "Filling v3 deposit")).to.be.true;
expect(multiCallerClient.transactionCount()).to.equal(1); // One transaction, filling the one deposit.

await fillV3Relay(spokePool_2, deposit, relayer);
unfilledDeposits = await getUnfilledDeposits(spokePoolClients, hubPoolClient);
expect(Object.values(unfilledDeposits).flat().length).to.equal(0);

// Verify that the relayer now sees that the deposit has been filled.
await relayerInstance.checkForUnfilledDepositsAndFill();
expect(lastSpyLogIncludes(spy, "0 unfilled deposits")).to.be.true;
expect(multiCallerClient.transactionCount()).to.equal(0);
});

it("Respects configured relayer routes", async function () {
relayerInstance = new Relayer(
relayer.address,
Expand Down
Loading