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: migrate bundle data client #709

Merged
merged 12 commits into from
Aug 27, 2024
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"name": "@across-protocol/sdk",
"author": "UMA Team",
"version": "3.1.27",
"version": "3.1.28-beta.1",
"license": "AGPL-3.0",
"homepage": "https://docs.across.to/reference/sdk",
"files": [
Expand Down
1,311 changes: 1,311 additions & 0 deletions src/clients/BundleDataClient/BundleDataClient.ts

Large diffs are not rendered by default.

2 changes: 2 additions & 0 deletions src/clients/BundleDataClient/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export * from "./BundleDataClient";
export * from "./utils";
268 changes: 268 additions & 0 deletions src/clients/BundleDataClient/utils/DataworkerUtils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,268 @@
// Create a combined `refunds` object containing refunds for V2 + V3 fills

import {
BundleDepositsV3,
BundleExcessSlowFills,
BundleFillsV3,
BundleSlowFills,
CombinedRefunds,
ExpiredDepositsToRefundV3,
PoolRebalanceLeaf,
Refund,
RunningBalances,
} from "../../../interfaces";
import {
bnZero,
AnyObject,
groupObjectCountsByTwoProps,
fixedPointAdjustment,
count2DDictionaryValues,
count3DDictionaryValues,
} from "../../../utils";
import {
addLastRunningBalance,
constructPoolRebalanceLeaves,
PoolRebalanceRoot,
updateRunningBalance,
updateRunningBalanceForDeposit,
} from "./PoolRebalanceUtils";
import { V3FillWithBlock } from "./shims";
import { AcrossConfigStoreClient } from "../../AcrossConfigStoreClient";
import { HubPoolClient } from "../../HubPoolClient";
import { buildPoolRebalanceLeafTree } from "./MerkleTreeUtils";

// and expired deposits.
export function getRefundsFromBundle(
bundleFillsV3: BundleFillsV3,
expiredDepositsToRefundV3: ExpiredDepositsToRefundV3
): CombinedRefunds {
const combinedRefunds: {
[repaymentChainId: string]: {
[repaymentToken: string]: Refund;
};
} = {};
Object.entries(bundleFillsV3).forEach(([repaymentChainId, fillsForChain]) => {
combinedRefunds[repaymentChainId] ??= {};
Object.entries(fillsForChain).forEach(([l2TokenAddress, { refunds }]) => {
// refunds can be undefined if these fills were all slow fill executions.
if (refunds === undefined) {
return;
}
// @dev use shallow copy so that modifying combinedRefunds doesn't modify the original refunds object.
const refundsShallowCopy = { ...refunds };
if (combinedRefunds[repaymentChainId][l2TokenAddress] === undefined) {
combinedRefunds[repaymentChainId][l2TokenAddress] = refundsShallowCopy;
} else {
// Each refunds object should have a unique refund address so we can add new ones to the
// existing dictionary.
combinedRefunds[repaymentChainId][l2TokenAddress] = {
...combinedRefunds[repaymentChainId][l2TokenAddress],
...refundsShallowCopy,
};
}
});
});
Object.entries(expiredDepositsToRefundV3).forEach(([originChainId, depositsForChain]) => {
combinedRefunds[originChainId] ??= {};
Object.entries(depositsForChain).forEach(([l2TokenAddress, deposits]) => {
deposits.forEach((deposit) => {
if (combinedRefunds[originChainId][l2TokenAddress] === undefined) {
combinedRefunds[originChainId][l2TokenAddress] = { [deposit.depositor]: deposit.inputAmount };
} else {
const existingRefundAmount = combinedRefunds[originChainId][l2TokenAddress][deposit.depositor];
combinedRefunds[originChainId][l2TokenAddress][deposit.depositor] = deposit.inputAmount.add(
existingRefundAmount ?? bnZero
);
}
});
});
});
return combinedRefunds;
}

export function prettyPrintV3SpokePoolEvents(
bundleDepositsV3: BundleDepositsV3,
bundleFillsV3: BundleFillsV3,
bundleInvalidFillsV3: V3FillWithBlock[],
bundleSlowFillsV3: BundleSlowFills,
expiredDepositsToRefundV3: ExpiredDepositsToRefundV3,
unexecutableSlowFills: BundleExcessSlowFills
): AnyObject {
return {
bundleDepositsV3: count2DDictionaryValues(bundleDepositsV3),
bundleFillsV3: count3DDictionaryValues(bundleFillsV3, "fills"),
bundleSlowFillsV3: count2DDictionaryValues(bundleSlowFillsV3),
expiredDepositsToRefundV3: count2DDictionaryValues(expiredDepositsToRefundV3),
unexecutableSlowFills: count2DDictionaryValues(unexecutableSlowFills),
allInvalidFillsInRangeByDestinationChainAndRelayer: groupObjectCountsByTwoProps(
bundleInvalidFillsV3,
"destinationChainId",
(fill) => `${fill.relayer}`
),
};
}

export function getEndBlockBuffers(
chainIdListForBundleEvaluationBlockNumbers: number[],
blockRangeEndBlockBuffer: { [chainId: number]: number }
): number[] {
// These buffers can be configured by the bot runner. They have two use cases:
// 1) Validate the end blocks specified in the pending root bundle. If the end block is greater than the latest
// block for its chain, then we should dispute the bundle because we can't look up events in the future for that
// chain. However, there are some cases where the proposer's node for that chain is returning a higher HEAD block
// than the bot-runner is seeing, so we can use this buffer to allow the proposer some margin of error. If
// the bundle end block is less than HEAD but within this buffer, then we won't dispute and we'll just exit
// early from this function.
// 2) Subtract from the latest block in a new root bundle proposal. This can be used to reduce the chance that
// bot runs using different providers see different contract state close to the HEAD block for a chain.
// Reducing the latest block that we query also gives partially filled deposits slightly more buffer for relayers
// to fully fill the deposit and reduces the chance that the data worker includes a slow fill payment that gets
// filled during the challenge period.
return chainIdListForBundleEvaluationBlockNumbers.map((chainId: number) => blockRangeEndBlockBuffer[chainId] ?? 0);
}

export function _buildPoolRebalanceRoot(
latestMainnetBlock: number,
mainnetBundleEndBlock: number,
bundleV3Deposits: BundleDepositsV3,
bundleFillsV3: BundleFillsV3,
bundleSlowFillsV3: BundleSlowFills,
unexecutableSlowFills: BundleExcessSlowFills,
expiredDepositsToRefundV3: ExpiredDepositsToRefundV3,
clients: { hubPoolClient: HubPoolClient; configStoreClient: AcrossConfigStoreClient },
maxL1TokenCountOverride?: number
): PoolRebalanceRoot {
// Running balances are the amount of tokens that we need to send to each SpokePool to pay for all instant and
// slow relay refunds. They are decreased by the amount of funds already held by the SpokePool. Balances are keyed
// by the SpokePool's network and L1 token equivalent of the L2 token to refund.
// Realized LP fees are keyed the same as running balances and represent the amount of LP fees that should be paid
// to LP's for each running balance.

// For each FilledRelay group, identified by { repaymentChainId, L1TokenAddress }, initialize a "running balance"
// to the total refund amount for that group.
const runningBalances: RunningBalances = {};
const realizedLpFees: RunningBalances = {};

/**
* REFUNDS FOR FAST FILLS
*/

// Add running balances and lp fees for v3 relayer refunds using BundleDataClient.bundleFillsV3. Refunds
// should be equal to inputAmount - lpFees so that relayers get to keep the relayer fee. Add the refund amount
// to the running balance for the repayment chain.
Object.entries(bundleFillsV3).forEach(([_repaymentChainId, fillsForChain]) => {
const repaymentChainId = Number(_repaymentChainId);
Object.entries(fillsForChain).forEach(
([l2TokenAddress, { realizedLpFees: totalRealizedLpFee, totalRefundAmount }]) => {
const l1TokenCounterpart = clients.hubPoolClient.getL1TokenForL2TokenAtBlock(
l2TokenAddress,
repaymentChainId,
mainnetBundleEndBlock
);

updateRunningBalance(runningBalances, repaymentChainId, l1TokenCounterpart, totalRefundAmount);
updateRunningBalance(realizedLpFees, repaymentChainId, l1TokenCounterpart, totalRealizedLpFee);
}
);
});

/**
* PAYMENTS SLOW FILLS
*/

// Add running balances and lp fees for v3 slow fills using BundleDataClient.bundleSlowFillsV3.
// Slow fills should still increment bundleLpFees and updatedOutputAmount should be equal to inputAmount - lpFees.
// Increment the updatedOutputAmount to the destination chain.
Object.entries(bundleSlowFillsV3).forEach(([_destinationChainId, depositsForChain]) => {
const destinationChainId = Number(_destinationChainId);
Object.entries(depositsForChain).forEach(([outputToken, deposits]) => {
deposits.forEach((deposit) => {
const l1TokenCounterpart = clients.hubPoolClient.getL1TokenForL2TokenAtBlock(
outputToken,
destinationChainId,
mainnetBundleEndBlock
);
const lpFee = deposit.lpFeePct.mul(deposit.inputAmount).div(fixedPointAdjustment);
updateRunningBalance(runningBalances, destinationChainId, l1TokenCounterpart, deposit.inputAmount.sub(lpFee));
// Slow fill LP fees are accounted for when the slow fill executes and a V3FilledRelay is emitted. i.e. when
// the slow fill execution is included in bundleFillsV3.
});
});
});

/**
* EXCESSES FROM UNEXECUTABLE SLOW FILLS
*/

// Subtract destination chain running balances for BundleDataClient.unexecutableSlowFills.
// These are all slow fills that are impossible to execute and therefore the amount to return would be
// the updatedOutputAmount = inputAmount - lpFees.
Object.entries(unexecutableSlowFills).forEach(([_destinationChainId, slowFilledDepositsForChain]) => {
const destinationChainId = Number(_destinationChainId);
Object.entries(slowFilledDepositsForChain).forEach(([outputToken, slowFilledDeposits]) => {
slowFilledDeposits.forEach((deposit) => {
const l1TokenCounterpart = clients.hubPoolClient.getL1TokenForL2TokenAtBlock(
outputToken,
destinationChainId,
mainnetBundleEndBlock
);
const lpFee = deposit.lpFeePct.mul(deposit.inputAmount).div(fixedPointAdjustment);
updateRunningBalance(runningBalances, destinationChainId, l1TokenCounterpart, lpFee.sub(deposit.inputAmount));
// Slow fills don't add to lpFees, only when the slow fill is executed and a V3FilledRelay is emitted, so
// we don't need to subtract it here. Moreover, the HubPoole expects bundleLpFees to be > 0.
});
});
});

/**
* DEPOSITS
*/

// Handle v3Deposits. These decrement running balances from the origin chain equal to the inputAmount.
// There should not be early deposits in v3.
Object.entries(bundleV3Deposits).forEach(([, depositsForChain]) => {
Object.entries(depositsForChain).forEach(([, deposits]) => {
deposits.forEach((deposit) => {
updateRunningBalanceForDeposit(runningBalances, clients.hubPoolClient, deposit, deposit.inputAmount.mul(-1));
});
});
});

/**
* REFUNDS FOR EXPIRED DEPOSITS
*/

// Add origin chain running balance for expired v3 deposits. These should refund the inputAmount.
Object.entries(expiredDepositsToRefundV3).forEach(([_originChainId, depositsForChain]) => {
const originChainId = Number(_originChainId);
Object.entries(depositsForChain).forEach(([inputToken, deposits]) => {
deposits.forEach((deposit) => {
const l1TokenCounterpart = clients.hubPoolClient.getL1TokenForL2TokenAtBlock(
inputToken,
originChainId,
mainnetBundleEndBlock
);
updateRunningBalance(runningBalances, originChainId, l1TokenCounterpart, deposit.inputAmount);
});
});
});

// Add to the running balance value from the last valid root bundle proposal for {chainId, l1Token}
// combination if found.
addLastRunningBalance(latestMainnetBlock, runningBalances, clients.hubPoolClient);

const leaves: PoolRebalanceLeaf[] = constructPoolRebalanceLeaves(
mainnetBundleEndBlock,
runningBalances,
realizedLpFees,
clients.configStoreClient,
maxL1TokenCountOverride
);

return {
runningBalances,
realizedLpFees,
leaves,
tree: buildPoolRebalanceLeafTree(leaves),
};
}
46 changes: 46 additions & 0 deletions src/clients/BundleDataClient/utils/FillUtils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
import { Fill } from "../../../interfaces";
import { getBlockRangeForChain, isSlowFill } from "../../../utils";
import { HubPoolClient } from "../../HubPoolClient";

export function getRefundInformationFromFill(
fill: Fill,
hubPoolClient: HubPoolClient,
blockRangesForChains: number[][],
chainIdListForBundleEvaluationBlockNumbers: number[],
fromLiteChain: boolean
): {
chainToSendRefundTo: number;
repaymentToken: string;
} {
// Handle slow relay where repaymentChainId = 0. Slow relays always pay recipient on destination chain.
// So, save the slow fill under the destination chain, and save the fast fill under its repayment chain.
let chainToSendRefundTo = isSlowFill(fill) ? fill.destinationChainId : fill.repaymentChainId;
// If the fill is for a deposit originating from the lite chain, the repayment chain is the origin chain
// regardless of whether it is a slow or fast fill (we ignore slow fills but this is for posterity).
if (fromLiteChain) {
chainToSendRefundTo = fill.originChainId;
}

// Save fill data and associate with repayment chain and L2 token refund should be denominated in.
const endBlockForMainnet = getBlockRangeForChain(
blockRangesForChains,
hubPoolClient.chainId,
chainIdListForBundleEvaluationBlockNumbers
)[1];

const l1TokenCounterpart = hubPoolClient.getL1TokenForL2TokenAtBlock(
fill.inputToken,
fill.originChainId,
endBlockForMainnet
);

const repaymentToken = hubPoolClient.getL2TokenForL1TokenAtBlock(
l1TokenCounterpart,
chainToSendRefundTo,
endBlockForMainnet
);
return {
chainToSendRefundTo,
repaymentToken,
};
}
26 changes: 26 additions & 0 deletions src/clients/BundleDataClient/utils/MerkleTreeUtils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import { MerkleTree } from "@across-protocol/contracts";
import { PoolRebalanceLeaf } from "../../../interfaces";
import { getParamType } from "../../../utils/ContractUtils";
import { utils } from "ethers";

export function buildPoolRebalanceLeafTree(poolRebalanceLeaves: PoolRebalanceLeaf[]): MerkleTree<PoolRebalanceLeaf> {
for (let i = 0; i < poolRebalanceLeaves.length; i++) {
// The 4 provided parallel arrays must be of equal length. Running Balances can optionally be 2x the length
if (
poolRebalanceLeaves[i].l1Tokens.length !== poolRebalanceLeaves[i].bundleLpFees.length ||
poolRebalanceLeaves[i].netSendAmounts.length !== poolRebalanceLeaves[i].bundleLpFees.length
) {
throw new Error("Provided lef arrays are not of equal length");
}
if (
poolRebalanceLeaves[i].runningBalances.length !== poolRebalanceLeaves[i].bundleLpFees.length * 2 &&
poolRebalanceLeaves[i].runningBalances.length !== poolRebalanceLeaves[i].bundleLpFees.length
) {
throw new Error("Running balances length unexpected");
}
}

const paramType = getParamType("MerkleLibTest", "verifyPoolRebalance", "rebalance");
const hashFn = (input: PoolRebalanceLeaf) => utils.keccak256(utils.defaultAbiCoder.encode([paramType], [input]));
return new MerkleTree<PoolRebalanceLeaf>(poolRebalanceLeaves, hashFn);
}
Loading
Loading