这一讲将会设计整体合约的结构,定义各个合约的接口,并做初步的技术分析。
wtfswap 设计每个池子都有一个价格范围,swap 只能在此价格范围内成交
- 任何人都可以创建池子,创建池子可以指定当前价格、价格范围: [a, b] 和 费率 f;相同交易对和费率可以有多个池子;不能删除和修改池子;
- 任何人都可以添加流动性,添加流动性只能在指定价格范围 [a, b];
- 流动性提供者可以减少添加的流动性,并提取减少流动性对应的两种代币;
- 流动性提供者可以在任何人 swap 过程收取手续费,手续费为 f,按流动性贡献加权平分给流动性提供者;
- 任何人都可以 swap,swap 需要指定某个池子,swap 可以指定输入(最大化输出)或者指定输出(最小化输入),如果指定的池子的流动性不足,则只会部分成交。
以上手续费的收取方式和 Uniswap 有所差异,做了简化,会在后续手续费实现的章节继续展开说明。
以简单为原则,我们不按照 Uniswap V3 将合约分为 periphery
和 core
两个独立仓库,而是自顶向下分为以下四个合约。
PoolManager.sol
: 顶层合约,对应 Pool 页面,负责 Pool 的创建和管理;PositionManager.sol
: 顶层合约,对应 Position 页面,负责 LP 头寸和流动性的管理;SwapRouter.sol
: 顶层合约,对应 Swap 页面,负责预估价格和交易;Factory.sol
: 底层合约,Pool 的工厂合约;Pool.sol
: 最底层合约,对应一个交易池,记录了当前价格、头寸、流动性等信息。
因为我们是自顶向下设计合约,因此我们首先分析前端页面(Pool 页面、Position 页面和 Swap 页面)需要哪些功能,并为此设计出顶层合约,再进一步分析细节,设计出底层合约。
PoolManager.sol
对应 Pool 页面,我们首先来看 Pool 页面有哪些功能
首先是展示所有的 pool ,对应前端页面如下:
注意,由于相同交易对和费率的池子可以重复创建,因此我们定义一个 PoolKey
的结构,参数包含 token0
、token1
和 index
,index
表示相同交易对池子的序号,从 0 开始递增,池子的地址可以由 PoolKey
唯一计算出来。定义出返回所有 pool 的方法 getPools
,接口定义如下:
struct PoolKey {
address token0;
address token1;
uint32 index;
}
function getPools() external view returns (PoolKey[] memory pools);
每个 pool 的信息包括:
- token 对的符号以及数量;
- 费率;
- 价格范围;
- 当前价格;
- 三个区间的总流动性。
我们可以根据以上信息定义出 PoolInfo
,以及获取 PoolInfo
的方法 getPoolInfo
,接口定义如下:
struct PoolInfo {
// the current protocol fee as a percentage of the swap fee taken on withdrawal
// represented as an integer denominator (1/x)%
uint8 feeProtocol;
// tick range
int24 tickLower;
int24 tickUpper;
// the current tick
int24 tick;
// the current price
uint160 sqrtPriceX96;
}
function getPoolInfo(
address token0,
address token1,
uint32 index
) external view returns (PoolInfo memory poolInfo);
此外还有一个添加池子的操作,当添加头寸时如果发现还没有对应的池子,需要先创建一个池子。
参数包括:
- token0 的地址和数量;
- token1 的地址和数量;
- 费率(百分比);
- 价格范围;
- 当前价格。
token0 的数量不为 0 或 token1 的数量不为 0 意味着添加初始流动性。
接口定义如下:
struct CreateAndInitializeParams {
address token0;
address token1;
uint24 fee;
int24 tickLower;
int24 tickUpper;
uint160 sqrtPriceX96;
}
function createAndInitializePoolIfNecessary(
CreateAndInitializeParams calldata params
) external payable returns (address pool, uint32 index);
完整的接口在 IPoolManager 中。
PositionManager.sol
对应 Position 页面,我们首先来看 Position 页面有哪些功能
首先是展示当前地址创建的头寸,对应前端页面如下:
可以通过用户地址返回所有其创建的头寸,定义 getPositions
方法,接口定义如下:
function getPositions(
address owner
) external view returns (uint256[] memory positionIds);
每个头寸的信息包括:
- token 对的符号以及数量(这里的数量是头寸拥有的两种代币数量);
- 费率;
- 价格范围;
- 添加的流动性;
- 收取的两种代币的手续费。
我们可以根据以上信息定义出 PositionInfo
,以及获取 PositionInfo
的方法 getPositionInfo
,接口定义如下:
struct PositionInfo {
// address owner;
address token0;
address token1;
uint24 fee;
int128 liquidity;
// tick range
int24 tickLower;
int24 tickUpper;
uint256 tokensOwed0;
uint256 tokensOwed1;
}
function getPositionInfo(
uint256 positionId
) external view returns (PositionInfo memory positionInfo);
右上角有一个添加头寸的操作,点击弹出以下页面:
跟添加池子非常类似,只是不能填价格范围和当前价格,参数包括:
- token0 的地址和数量;
- token1 的地址和数量;
- 费率(百分比)。
定义 mint
方法,接口定义如下:
struct MintParams {
address token0;
address token1;
uint32 index;
uint256 amount0Desired;
uint256 amount1Desired;
address recipient;
uint256 deadline;
}
function mint(
MintParams calldata params
)
external
payable
returns (
uint256 positionId,
uint128 liquidity,
uint256 amount0,
uint256 amount1
);
值得注意的是,选定 token0 和 token1 是两个下拉框,费率会自动显示。要想实现这个逻辑,我们需要进一步完善 PoolManager.sol
,新增两个方法:
getTokens
: 获取所有的 tokens,用于用户第一个下拉框选择;getTokenPools
:参数是一个 token 地址,返回所有的池子,用于用户第二个下拉框选择。
接口定义如下:
function getTokens() external view returns (address[] memory tokens);
function getTokenPools(
address token
) external view returns (PoolKey[] memory pools);
每行头寸的信息还有两个按钮,分别是 burn
和 collect
,分别代表销毁头寸的流动性,以及提取全部手续费。
接口定义如下:
function burn(
uint256 positionId,
uint128 amount
) external returns (uint256 amount0, uint256 amount1);
function collect(
uint256 positionId,
address recipient
) external returns (uint256 amount0, uint256 amount1);
完整的接口在 IPositionManager 中。
SwapRouter.sol
对应 Swap 页面,我们首先来看 Swap 页面有哪些功能
对应前端页面如下:
首先选定 token0 和 token1 也是两个下拉框,实现和 添加头寸 页面一致,只是不会展示费率,因此用户选择完交易对后会有多个池子。
DApp 需要分析出最优的 Swap 路径,这里用 indexPath
和 sqrtPriceLimitX96
两个参数。indexPath
的类型为 uint32[]
,表示选取的池子的序号;sqrtPriceLimitX96
的类型为 uint160
,表示每个池子交易的限价。逻辑如下:
- 先从
indexPath
中取出一个index
确认池子; - 在池子中进行 Swap,如果满足用户要求(即没有剩余的
amount
)则结束; - 如果触及
sqrtPriceLimitX96
限价,并且 Swap 还未满足用户要求(即剩余amount
),则扣除成交的amount
,回到第一步。如果已经是最后一个池子则以部分成交结束。
然后就是估算逻辑了,有以下两种方法:
quoteExactInput
:用户输入框输入 token0 的数量,输出框自动展示 token1 的数量;quoteExactOutput
:用户输出框输入 token1 的数量,输入框自动展示 token0 的数量;
接口定义如下:
struct QuoteExactInputParams {
address tokenIn;
address tokenOut;
uint32[] indexPath;
uint256 amountIn;
uint160 sqrtPriceLimitX96;
}
function quoteExactInput(
QuoteExactInputParams memory params
) external returns (uint256 amountOut);
struct QuoteExactOutputParams {
address tokenIn;
address tokenOut;
uint32[] indexPath;
uint256 amount;
uint160 sqrtPriceLimitX96;
}
function quoteExactOutput(
QuoteExactOutputParams memory params
) external returns (uint256 amountIn);
最后,当用户点击 Swap 按钮,有两种估算逻辑对应的方法 exactInput
和 exactOutput
。
接口定义如下:
struct ExactInputParams {
address tokenIn;
address tokenOut;
uint32[] indexPath;
address recipient;
uint256 deadline;
uint256 amountIn;
uint256 amountOutMinimum;
uint160 sqrtPriceLimitX96;
}
function exactInput(
ExactInputParams calldata params
) external payable returns (uint256 amountOut);
struct ExactOutputParams {
address tokenIn;
address tokenOut;
uint32[] indexPath;
address recipient;
uint256 deadline;
uint256 amountOut;
uint256 amountInMaximum;
uint160 sqrtPriceLimitX96;
}
function exactOutput(
ExactOutputParams calldata params
) external payable returns (uint256 amountIn);
完整的接口在 ISwapRouter 中。
Factory.sol
是 Pool 的工厂合约,比较简单,定义了 getPool
和 createPool
的方法,以及 PoolCreated
事件。
接口定义如下:
event PoolCreated(
address indexed token0,
address indexed token1,
uint32 indexed index,
address pool
);
function getPool(
address tokenA,
address tokenB,
uint24 fee,
uint8 bump
) external view returns (address pool);
function createPool(
address tokenA,
address tokenB,
uint32 index,
uint24 fee
) external returns (address pool);
特别的,参照 Uniswap,工厂合约也设计成临时存储交易池合约初始化参数 parameters ,从而完成参数的传递。新增如下方法定义:
function parameters()
external
view
returns (address factory, address token0, address token1, uint24 fee);
完整的接口在 IFactory 中。
Pool.sol
是最底层的合约,实现了 WTFSwap
的核心逻辑。
首先是一些不可变量的读方法,如下:
function factory() external view returns (address);
function token0() external view returns (address);
function token1() external view returns (address);
function fee() external view returns (uint24);
function tickLower() external view returns (int24);
function tickUpper() external view returns (int24);
然后是当前状态变量的读方法,即当前价格、tick、流动性,以及不同头寸位置的流动性和代币数量,如下:
function sqrtPriceX96() external view returns (uint160);
function tick() external view returns (int24);
function liquidity() external view returns (uint128);
我们还要定义初始化方法,相比于 Uniswap,我们初始化时指定了价格范围,如下:
function initialize(
uint160 sqrtPriceX96,
int24 tickLower,
int24 tickUpper
) external;
最后是上层合约的底层实现,分别是 mint
、collect
、 burn
、 swap
方法以及事件。
接口定义如下:
event Mint(
address sender,
address indexed owner,
uint128 amount,
uint256 amount0,
uint256 amount1
);
function mint(
address recipient,
uint128 amount,
bytes calldata data
) external returns (uint256 amount0, uint256 amount1);
event Collect(
address indexed owner,
address recipient,
uint128 amount0,
uint128 amount1
);
function collect(
address recipient
) external returns (uint128 amount0, uint128 amount1);
event Burn(
address indexed owner,
uint128 amount,
uint256 amount0,
uint256 amount1
);
function burn(
uint128 amount
) external returns (uint256 amount0, uint256 amount1);
event Swap(
address indexed sender,
address indexed recipient,
int256 amount0,
int256 amount1,
uint160 sqrtPriceX96,
uint128 liquidity,
int24 tick
);
function swap(
address recipient,
bool zeroForOne,
int256 amountSpecified,
uint160 sqrtPriceLimitX96,
bytes calldata data
) external returns (int256 amount0, int256 amount1);
特别的,还需要定义两个回调接口,分别用于 pool 合约 mint
和 swap
的回调。接口定义如下:
interface IMintCallback {
function mintCallback(
uint256 amount0Owed,
uint256 amount1Owed,
bytes calldata data
) external;
}
interface ISwapCallback {
function swapCallback(
int256 amount0Delta,
int256 amount1Delta,
bytes calldata data
) external;
}
完整的接口在 IPool.sol 中。