Skip to content

Commit

Permalink
refactor: move aave-v2-like aave-v3-like logics (without flashloan) t…
Browse files Browse the repository at this point in the history
…o modules
  • Loading branch information
Bob Lu committed Jul 21, 2024
1 parent cf0787e commit 16640e0
Show file tree
Hide file tree
Showing 75 changed files with 22,313 additions and 870 deletions.
2 changes: 2 additions & 0 deletions src/logics/aave-v2/configs.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import * as common from '@protocolink/common';

export const protocolId = 'aave-v2';

type ContractNames = 'ProtocolDataProvider' | 'AaveV2FlashLoanCallback';

export interface Config {
Expand Down
54 changes: 10 additions & 44 deletions src/logics/aave-v2/logic.borrow.ts
Original file line number Diff line number Diff line change
@@ -1,53 +1,19 @@
import { InterestRateMode } from './types';
import { LendingPool__factory } from './contracts';
import { Service } from './service';
import * as common from '@protocolink/common';
import * as core from '@protocolink/core';
import { supportedChainIds } from './configs';
import * as aavev2 from 'src/modules/aavev2';
import { protocolId, supportedChainIds } from './configs';
import { providers } from 'ethers';

export type BorrowLogicTokenList = common.Token[];
export type BorrowLogicFields = aavev2.BorrowLogicFields;

export type BorrowLogicFields = core.TokenOutFields<{ interestRateMode: InterestRateMode; referralCode?: number }>;
export type BorrowLogicOptions = aavev2.BorrowLogicOptions;

export type BorrowLogicOptions = Pick<core.GlobalOptions, 'account'>;
export type BorrowLogicTokenList = aavev2.BorrowLogicTokenList;

export class BorrowLogic extends core.Logic implements core.LogicTokenListInterface, core.LogicBuilderInterface {
static id = 'borrow';
static protocolId = 'aave-v2';
export class BorrowLogic extends aavev2.BorrowLogic {
static protocolId = protocolId;
static readonly supportedChainIds = supportedChainIds;

async getTokenList() {
const service = new Service(this.chainId, this.provider);
const reserveTokens = await service.getBorrowTokens();

const tokenList: BorrowLogicTokenList = [];
for (const { asset } of reserveTokens) {
if (asset.isWrapped) {
tokenList.push(asset.unwrapped);
}
tokenList.push(asset);
}

return tokenList;
}

async build(fields: BorrowLogicFields, options: BorrowLogicOptions) {
const { output, interestRateMode, referralCode = 0 } = fields;
const { account } = options;

const tokenOut = output.token.wrapped;

const service = new Service(this.chainId, this.provider);
const to = await service.getLendingPoolAddress();
const data = LendingPool__factory.createInterface().encodeFunctionData('borrow', [
tokenOut.address,
output.amountWei,
interestRateMode,
referralCode,
account,
]);
const wrapMode = output.token.isNative ? core.WrapMode.unwrapAfter : core.WrapMode.none;

return core.newLogic({ to, data, wrapMode });
constructor(chainId: number, provider?: providers.Provider) {
super({ chainId, provider, service: new Service(chainId, provider) });
}
}
68 changes: 14 additions & 54 deletions src/logics/aave-v2/logic.deposit.ts
Original file line number Diff line number Diff line change
@@ -1,68 +1,28 @@
import { LendingPool__factory } from './contracts';
import { Service } from './service';
import * as common from '@protocolink/common';
import * as core from '@protocolink/core';
import { supportedChainIds } from './configs';
import * as aavev2 from 'src/modules/aavev2';
import { protocolId, supportedChainIds } from './configs';
import { providers } from 'ethers';

export type DepositLogicTokenList = [common.Token, common.Token][];
export type DepositLogicTokenList = aavev2.DepositLogicTokenList;

export type DepositLogicParams = core.TokenToTokenExactInParams;
export type DepositLogicParams = aavev2.DepositLogicParams;

export type DepositLogicFields = core.TokenToTokenExactInFields<{ referralCode?: number }>;
export type DepositLogicFields = aavev2.DepositLogicFields;

export type DepositLogicOptions = Pick<core.GlobalOptions, 'account'>;
export type DepositLogicOptions = aavev2.DepositLogicOptions;

export class DepositLogic
extends core.Logic
implements core.LogicTokenListInterface, core.LogicOracleInterface, core.LogicBuilderInterface
{
export class DepositLogic extends aavev2.DepositLogics {
static id = 'deposit';
static protocolId = 'aave-v2';
static protocolId = protocolId;
static readonly supportedChainIds = supportedChainIds;

async getTokenList() {
const service = new Service(this.chainId, this.provider);
const reserveTokens = await service.getSupplyTokens();

const tokenList: DepositLogicTokenList = [];
for (const reserveToken of reserveTokens) {
if (reserveToken.asset.isWrapped) {
tokenList.push([reserveToken.asset.unwrapped, reserveToken.aToken]);
}
tokenList.push([reserveToken.asset, reserveToken.aToken]);
}

return tokenList;
constructor(chainId: number, provider?: providers.Provider) {
super({ chainId, provider, service: new Service(chainId, provider) });
}

async quote(params: DepositLogicParams) {
const { input, tokenOut } = params;
const output = new common.TokenAmount(tokenOut, input.amount);

return { input, output };
}

async build(fields: DepositLogicFields, options: DepositLogicOptions) {
const { input, balanceBps, referralCode = 0 } = fields;
const { account } = options;

const tokenIn = input.token.wrapped;
const agent = await this.calcAgent(account);

const service = new Service(this.chainId, this.provider);
const to = await service.getLendingPoolAddress();
const data = LendingPool__factory.createInterface().encodeFunctionData('deposit', [
tokenIn.address,
input.amountWei,
agent,
referralCode,
]);
const amountOffset = balanceBps ? common.getParamOffset(1) : undefined;
const inputs = [
core.newLogicInput({ input: new common.TokenAmount(tokenIn, input.amount), balanceBps, amountOffset }),
];
const wrapMode = input.token.isNative ? core.WrapMode.wrapBefore : core.WrapMode.none;
async getTokenList() {
const reserveTokens = await this.service.getSupplyTokens();

return core.newLogic({ to, data, inputs, wrapMode });
return aavev2.createDepositTokenList(reserveTokens, 'rToken');
}
}
74 changes: 10 additions & 64 deletions src/logics/aave-v2/logic.repay.ts
Original file line number Diff line number Diff line change
@@ -1,73 +1,19 @@
import { InterestRateMode } from './types';
import { LendingPool__factory } from './contracts';
import { Service } from './service';
import * as common from '@protocolink/common';
import * as core from '@protocolink/core';
import { supportedChainIds } from './configs';
import * as aavev2 from 'src/modules/aavev2';
import { protocolId, supportedChainIds } from './configs';
import { providers } from 'ethers';

export type RepayLogicTokenList = common.Token[];
export type RepayLogicTokenList = aavev2.RepayLogicTokenList;

export type RepayLogicParams = core.RepayParams<{ interestRateMode: InterestRateMode }>;
export type RepayLogicParams = aavev2.RepayLogicParams;

export type RepayLogicFields = core.RepayFields<{ interestRateMode: InterestRateMode }>;
export type RepayLogicFields = aavev2.RepayLogicFields;

export class RepayLogic
extends core.Logic
implements core.LogicTokenListInterface, core.LogicOracleInterface, core.LogicBuilderInterface
{
static id = 'repay';
static protocolId = 'aave-v2';
export class RepayLogic extends aavev2.RepayLogic {
static protocolId = protocolId;
static readonly supportedChainIds = supportedChainIds;

async getTokenList() {
const service = new Service(this.chainId, this.provider);
const tokens = await service.getAssets();

const tokenList: RepayLogicTokenList = [];
for (const token of tokens) {
if (token.isWrapped) {
tokenList.push(token.unwrapped);
}
tokenList.push(token);
}

return tokenList;
}

async quote(params: RepayLogicParams) {
const { borrower, tokenIn, interestRateMode } = params;

const service = new Service(this.chainId, this.provider);
const { currentStableDebt, currentVariableDebt } = await service.protocolDataProvider.getUserReserveData(
tokenIn.wrapped.address,
borrower
);
const currentDebt = interestRateMode === InterestRateMode.variable ? currentVariableDebt : currentStableDebt;
const amountWei = common.calcSlippage(currentDebt, -1); // slightly higher than the current borrowed amount
const input = new common.TokenAmount(tokenIn).setWei(amountWei);

return { borrower, interestRateMode, input };
}

async build(fields: RepayLogicFields) {
const { input, interestRateMode, borrower, balanceBps } = fields;

const tokenIn = input.token.wrapped;

const service = new Service(this.chainId, this.provider);
const to = await service.getLendingPoolAddress();
const data = LendingPool__factory.createInterface().encodeFunctionData('repay', [
tokenIn.address,
input.amountWei,
interestRateMode,
borrower,
]);
const amountOffset = balanceBps ? common.getParamOffset(1) : undefined;
const inputs = [
core.newLogicInput({ input: new common.TokenAmount(tokenIn, input.amount), balanceBps, amountOffset }),
];
const wrapMode = input.token.isNative ? core.WrapMode.wrapBefore : core.WrapMode.none;

return core.newLogic({ to, data, inputs, wrapMode });
constructor(chainId: number, provider?: providers.Provider) {
super({ chainId, provider, service: new Service(chainId, provider) });
}
}
66 changes: 14 additions & 52 deletions src/logics/aave-v2/logic.withdraw.ts
Original file line number Diff line number Diff line change
@@ -1,65 +1,27 @@
import { LendingPool__factory } from './contracts';
import { Service } from './service';
import * as common from '@protocolink/common';
import * as core from '@protocolink/core';
import { supportedChainIds } from './configs';
import * as aavev2 from 'src/modules/aavev2';
import { protocolId, supportedChainIds } from './configs';
import { providers } from 'ethers';

export type WithdrawLogicTokenList = [common.Token, common.Token][];
export type WithdrawLogicTokenList = aavev2.WithdrawLogicTokenList;

export type WithdrawLogicParams = core.TokenToTokenExactInParams;
export type WithdrawLogicParams = aavev2.WithdrawLogicParams;

export type WithdrawLogicFields = core.TokenToTokenExactInFields;
export type WithdrawLogicFields = aavev2.WithdrawLogicFields;

export type WithdrawLogicOptions = Pick<core.GlobalOptions, 'account'>;
export type WithdrawLogicOptions = aavev2.WithdrawLogicOptions;

export class WithdrawLogic
extends core.Logic
implements core.LogicTokenListInterface, core.LogicOracleInterface, core.LogicBuilderInterface
{
static id = 'withdraw';
static protocolId = 'aave-v2';
export class WithdrawLogic extends aavev2.WithdrawLogic {
static protocolId = protocolId;
static readonly supportedChainIds = supportedChainIds;

async getTokenList() {
const service = new Service(this.chainId, this.provider);
const { reserveTokens } = await service.getReserveTokens();

const tokenList: WithdrawLogicTokenList = [];
for (const reserveToken of reserveTokens) {
if (reserveToken.asset.isWrapped) {
tokenList.push([reserveToken.aToken, reserveToken.asset.unwrapped]);
}
tokenList.push([reserveToken.aToken, reserveToken.asset]);
}

return tokenList;
constructor(chainId: number, provider?: providers.Provider) {
super({ chainId, provider, service: new Service(chainId, provider) });
}

async quote(params: WithdrawLogicParams) {
const { input, tokenOut } = params;
const output = new common.TokenAmount(tokenOut, input.amount);

return { input, output };
}

async build(fields: WithdrawLogicFields, options: WithdrawLogicOptions) {
const { input, output, balanceBps } = fields;
const { account } = options;

const tokenOut = output.token.wrapped;
const agent = await this.calcAgent(account);

const service = new Service(this.chainId, this.provider);
const to = await service.getLendingPoolAddress();
const data = LendingPool__factory.createInterface().encodeFunctionData('withdraw', [
tokenOut.address,
input.amountWei,
agent,
]);
const amountOffset = balanceBps ? common.getParamOffset(1) : undefined;
const inputs = [core.newLogicInput({ input, balanceBps, amountOffset })];
const wrapMode = output.token.isNative ? core.WrapMode.unwrapAfter : core.WrapMode.none;
async getTokenList() {
const reserveTokens = await this.service.getSupplyTokens();

return core.newLogic({ to, data, inputs, wrapMode });
return aavev2.createWithdrawTokenList(reserveTokens, 'aToken');
}
}
2 changes: 2 additions & 0 deletions src/logics/aave-v3/configs.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import * as common from '@protocolink/common';

export const protocolId = 'aave-v3';

type ContractNames = 'PoolDataProvider' | 'AaveV3FlashLoanCallback';

export interface Config {
Expand Down
54 changes: 10 additions & 44 deletions src/logics/aave-v3/logic.borrow.ts
Original file line number Diff line number Diff line change
@@ -1,53 +1,19 @@
import { InterestRateMode } from './types';
import { Pool__factory } from './contracts';
import { Service } from './service';
import * as common from '@protocolink/common';
import * as core from '@protocolink/core';
import { supportedChainIds } from './configs';
import * as aavev3 from 'src/modules/aavev3';
import { protocolId, supportedChainIds } from './configs';
import { providers } from 'ethers';

export type BorrowLogicTokenList = common.Token[];
export type BorrowLogicTokenList = aavev3.BorrowLogicTokenList;

export type BorrowLogicFields = core.TokenOutFields<{ interestRateMode: InterestRateMode; referralCode?: number }>;
export type BorrowLogicFields = aavev3.BorrowLogicFields;

export type BorrowLogicOptions = Pick<core.GlobalOptions, 'account'>;
export type BorrowLogicOptions = aavev3.BorrowLogicOptions;

export class BorrowLogic extends core.Logic implements core.LogicTokenListInterface, core.LogicBuilderInterface {
static id = 'borrow';
static protocolId = 'aave-v3';
export class BorrowLogic extends aavev3.BorrowLogic {
static protocolId = protocolId;
static readonly supportedChainIds = supportedChainIds;

async getTokenList() {
const service = new Service(this.chainId, this.provider);
const reserveTokens = await service.getBorrowTokens();

const tokenList: BorrowLogicTokenList = [];
for (const { asset } of reserveTokens) {
if (asset.isWrapped) {
tokenList.push(asset.unwrapped);
}
tokenList.push(asset);
}

return tokenList;
}

async build(fields: BorrowLogicFields, options: BorrowLogicOptions) {
const { output, interestRateMode, referralCode = 0 } = fields;
const { account } = options;

const tokenOut = output.token.wrapped;

const service = new Service(this.chainId, this.provider);
const to = await service.getPoolAddress();
const data = Pool__factory.createInterface().encodeFunctionData('borrow', [
tokenOut.address,
output.amountWei,
interestRateMode,
referralCode,
account,
]);
const wrapMode = output.token.isNative ? core.WrapMode.unwrapAfter : core.WrapMode.none;

return core.newLogic({ to, data, wrapMode });
constructor(chainId: number, provider?: providers.Provider) {
super({ chainId, provider, service: new Service(chainId, provider) });
}
}
Loading

0 comments on commit 16640e0

Please sign in to comment.