import { ethers, waffle } from 'hardhat' import { BigNumber, BigNumberish, constants, Wallet } from 'ethers' import { TestERC20 } from '../typechain/TestERC20' import { UniswapV3Factory } from '../typechain/UniswapV3Factory' import { MockTimeUniswapV3Pool } from '../typechain/MockTimeUniswapV3Pool' import { TestUniswapV3SwapPay } from '../typechain/TestUniswapV3SwapPay' import checkObservationEquals from './shared/checkObservationEquals' import { expect } from './shared/expect' import { poolFixture, TEST_POOL_START_TIME } from './shared/fixtures' import { expandTo18Decimals, FeeAmount, getPositionKey, getMaxTick, getMinTick, encodePriceSqrt, TICK_SPACINGS, createPoolFunctions, SwapFunction, MintFunction, getMaxLiquidityPerTick, FlashFunction, MaxUint128, MAX_SQRT_RATIO, MIN_SQRT_RATIO, SwapToPriceFunction, } from './shared/utilities' import { TestUniswapV3Callee } from '../typechain/TestUniswapV3Callee' import { TestUniswapV3ReentrantCallee } from '../typechain/TestUniswapV3ReentrantCallee' import { TickMathTest } from '../typechain/TickMathTest' import { SwapMathTest } from '../typechain/SwapMathTest' const createFixtureLoader = waffle.createFixtureLoader type ThenArg = T extends PromiseLike ? U : T describe('UniswapV3Pool', () => { let wallet: Wallet, other: Wallet let token0: TestERC20 let token1: TestERC20 let token2: TestERC20 let factory: UniswapV3Factory let pool: MockTimeUniswapV3Pool let swapTarget: TestUniswapV3Callee let swapToLowerPrice: SwapToPriceFunction let swapToHigherPrice: SwapToPriceFunction let swapExact0For1: SwapFunction let swap0ForExact1: SwapFunction let swapExact1For0: SwapFunction let swap1ForExact0: SwapFunction let feeAmount: number let tickSpacing: number let minTick: number let maxTick: number let mint: MintFunction let flash: FlashFunction let loadFixture: ReturnType let createPool: ThenArg>['createPool'] before('create fixture loader', async () => { ;[wallet, other] = await (ethers as any).getSigners() loadFixture = createFixtureLoader([wallet, other]) }) beforeEach('deploy fixture', async () => { ;({ token0, token1, token2, factory, createPool, swapTargetCallee: swapTarget } = await loadFixture(poolFixture)) const oldCreatePool = createPool createPool = async (_feeAmount, _tickSpacing) => { const pool = await oldCreatePool(_feeAmount, _tickSpacing) ;({ swapToLowerPrice, swapToHigherPrice, swapExact0For1, swap0ForExact1, swapExact1For0, swap1ForExact0, mint, flash, } = createPoolFunctions({ token0, token1, swapTarget, pool, })) minTick = getMinTick(_tickSpacing) maxTick = getMaxTick(_tickSpacing) feeAmount = _feeAmount tickSpacing = _tickSpacing return pool } // default to the 30 bips pool pool = await createPool(FeeAmount.MEDIUM, TICK_SPACINGS[FeeAmount.MEDIUM]) }) it('constructor initializes immutables', async () => { expect(await pool.factory()).to.eq(factory.address) expect(await pool.token0()).to.eq(token0.address) expect(await pool.token1()).to.eq(token1.address) expect(await pool.maxLiquidityPerTick()).to.eq(getMaxLiquidityPerTick(tickSpacing)) }) describe('#initialize', () => { it('fails if already initialized', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await expect(pool.initialize(encodePriceSqrt(1, 1))).to.be.reverted }) it('fails if starting price is too low', async () => { await expect(pool.initialize(1)).to.be.revertedWith('R') await expect(pool.initialize(MIN_SQRT_RATIO.sub(1))).to.be.revertedWith('R') }) it('fails if starting price is too high', async () => { await expect(pool.initialize(MAX_SQRT_RATIO)).to.be.revertedWith('R') await expect(pool.initialize(BigNumber.from(2).pow(160).sub(1))).to.be.revertedWith('R') }) it('can be initialized at MIN_SQRT_RATIO', async () => { await pool.initialize(MIN_SQRT_RATIO) expect((await pool.slot0()).tick).to.eq(getMinTick(1)) }) it('can be initialized at MAX_SQRT_RATIO - 1', async () => { await pool.initialize(MAX_SQRT_RATIO.sub(1)) expect((await pool.slot0()).tick).to.eq(getMaxTick(1) - 1) }) it('sets initial variables', async () => { const price = encodePriceSqrt(1, 2) await pool.initialize(price) const { sqrtPriceX96, observationIndex } = await pool.slot0() expect(sqrtPriceX96).to.eq(price) expect(observationIndex).to.eq(0) expect((await pool.slot0()).tick).to.eq(-6932) }) it('initializes the first observations slot', async () => { await pool.initialize(encodePriceSqrt(1, 1)) checkObservationEquals(await pool.observations(0), { secondsPerLiquidityCumulativeX128: 0, initialized: true, blockTimestamp: TEST_POOL_START_TIME, tickCumulative: 0, }) }) it('emits a Initialized event with the input tick', async () => { const sqrtPriceX96 = encodePriceSqrt(1, 2) await expect(pool.initialize(sqrtPriceX96)).to.emit(pool, 'Initialize').withArgs(sqrtPriceX96, -6932) }) }) describe('#increaseObservationCardinalityNext', () => { it('can only be called after initialize', async () => { await expect(pool.increaseObservationCardinalityNext(2)).to.be.revertedWith('LOK') }) it('emits an event including both old and new', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await expect(pool.increaseObservationCardinalityNext(2)) .to.emit(pool, 'IncreaseObservationCardinalityNext') .withArgs(1, 2) }) it('does not emit an event for no op call', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await pool.increaseObservationCardinalityNext(3) await expect(pool.increaseObservationCardinalityNext(2)).to.not.emit(pool, 'IncreaseObservationCardinalityNext') }) it('does not change cardinality next if less than current', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await pool.increaseObservationCardinalityNext(3) await pool.increaseObservationCardinalityNext(2) expect((await pool.slot0()).observationCardinalityNext).to.eq(3) }) it('increases cardinality and cardinality next first time', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await pool.increaseObservationCardinalityNext(2) const { observationCardinality, observationCardinalityNext } = await pool.slot0() expect(observationCardinality).to.eq(1) expect(observationCardinalityNext).to.eq(2) }) }) describe('#mint', () => { it('fails if not initialized', async () => { await expect(mint(wallet.address, -tickSpacing, tickSpacing, 1)).to.be.revertedWith('LOK') }) describe('after initialization', () => { beforeEach('initialize the pool at price of 10:1', async () => { await pool.initialize(encodePriceSqrt(1, 10)) await mint(wallet.address, minTick, maxTick, 3161) }) describe('failure cases', () => { it('fails if tickLower greater than tickUpper', async () => { // should be TLU but...hardhat await expect(mint(wallet.address, 1, 0, 1)).to.be.reverted }) it('fails if tickLower less than min tick', async () => { // should be TLM but...hardhat await expect(mint(wallet.address, -887273, 0, 1)).to.be.reverted }) it('fails if tickUpper greater than max tick', async () => { // should be TUM but...hardhat await expect(mint(wallet.address, 0, 887273, 1)).to.be.reverted }) it('fails if amount exceeds the max', async () => { // these should fail with 'LO' but hardhat is bugged const maxLiquidityGross = await pool.maxLiquidityPerTick() await expect(mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, maxLiquidityGross.add(1))).to .be.reverted await expect(mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, maxLiquidityGross)).to.not.be .reverted }) it('fails if total amount at tick exceeds the max', async () => { // these should fail with 'LO' but hardhat is bugged await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, 1000) const maxLiquidityGross = await pool.maxLiquidityPerTick() await expect( mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, maxLiquidityGross.sub(1000).add(1)) ).to.be.reverted await expect( mint(wallet.address, minTick + tickSpacing * 2, maxTick - tickSpacing, maxLiquidityGross.sub(1000).add(1)) ).to.be.reverted await expect( mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing * 2, maxLiquidityGross.sub(1000).add(1)) ).to.be.reverted await expect(mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, maxLiquidityGross.sub(1000))) .to.not.be.reverted }) it('fails if amount is 0', async () => { await expect(mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, 0)).to.be.reverted }) }) describe('success cases', () => { it('initial balances', async () => { expect(await token0.balanceOf(pool.address)).to.eq(9996) expect(await token1.balanceOf(pool.address)).to.eq(1000) }) it('initial tick', async () => { expect((await pool.slot0()).tick).to.eq(-23028) }) describe('above current price', () => { it('transfers token0 only', async () => { await expect(mint(wallet.address, -22980, 0, 10000)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 21549) .to.not.emit(token1, 'Transfer') expect(await token0.balanceOf(pool.address)).to.eq(9996 + 21549) expect(await token1.balanceOf(pool.address)).to.eq(1000) }) it('max tick with max leverage', async () => { await mint(wallet.address, maxTick - tickSpacing, maxTick, BigNumber.from(2).pow(102)) expect(await token0.balanceOf(pool.address)).to.eq(9996 + 828011525) expect(await token1.balanceOf(pool.address)).to.eq(1000) }) it('works for max tick', async () => { await expect(mint(wallet.address, -22980, maxTick, 10000)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 31549) expect(await token0.balanceOf(pool.address)).to.eq(9996 + 31549) expect(await token1.balanceOf(pool.address)).to.eq(1000) }) it('removing works', async () => { await mint(wallet.address, -240, 0, 10000) await pool.burn(-240, 0, 10000) const { amount0, amount1 } = await pool.callStatic.collect(wallet.address, -240, 0, MaxUint128, MaxUint128) expect(amount0, 'amount0').to.eq(120) expect(amount1, 'amount1').to.eq(0) }) it('adds liquidity to liquidityGross', async () => { await mint(wallet.address, -240, 0, 100) expect((await pool.ticks(-240)).liquidityGross).to.eq(100) expect((await pool.ticks(0)).liquidityGross).to.eq(100) expect((await pool.ticks(tickSpacing)).liquidityGross).to.eq(0) expect((await pool.ticks(tickSpacing * 2)).liquidityGross).to.eq(0) await mint(wallet.address, -240, tickSpacing, 150) expect((await pool.ticks(-240)).liquidityGross).to.eq(250) expect((await pool.ticks(0)).liquidityGross).to.eq(100) expect((await pool.ticks(tickSpacing)).liquidityGross).to.eq(150) expect((await pool.ticks(tickSpacing * 2)).liquidityGross).to.eq(0) await mint(wallet.address, 0, tickSpacing * 2, 60) expect((await pool.ticks(-240)).liquidityGross).to.eq(250) expect((await pool.ticks(0)).liquidityGross).to.eq(160) expect((await pool.ticks(tickSpacing)).liquidityGross).to.eq(150) expect((await pool.ticks(tickSpacing * 2)).liquidityGross).to.eq(60) }) it('removes liquidity from liquidityGross', async () => { await mint(wallet.address, -240, 0, 100) await mint(wallet.address, -240, 0, 40) await pool.burn(-240, 0, 90) expect((await pool.ticks(-240)).liquidityGross).to.eq(50) expect((await pool.ticks(0)).liquidityGross).to.eq(50) }) it('clears tick lower if last position is removed', async () => { await mint(wallet.address, -240, 0, 100) await pool.burn(-240, 0, 100) const { liquidityGross, feeGrowthOutside0X128, feeGrowthOutside1X128 } = await pool.ticks(-240) expect(liquidityGross).to.eq(0) expect(feeGrowthOutside0X128).to.eq(0) expect(feeGrowthOutside1X128).to.eq(0) }) it('clears tick upper if last position is removed', async () => { await mint(wallet.address, -240, 0, 100) await pool.burn(-240, 0, 100) const { liquidityGross, feeGrowthOutside0X128, feeGrowthOutside1X128 } = await pool.ticks(0) expect(liquidityGross).to.eq(0) expect(feeGrowthOutside0X128).to.eq(0) expect(feeGrowthOutside1X128).to.eq(0) }) it('only clears the tick that is not used at all', async () => { await mint(wallet.address, -240, 0, 100) await mint(wallet.address, -tickSpacing, 0, 250) await pool.burn(-240, 0, 100) let { liquidityGross, feeGrowthOutside0X128, feeGrowthOutside1X128 } = await pool.ticks(-240) expect(liquidityGross).to.eq(0) expect(feeGrowthOutside0X128).to.eq(0) expect(feeGrowthOutside1X128).to.eq(0) ;({ liquidityGross, feeGrowthOutside0X128, feeGrowthOutside1X128 } = await pool.ticks(-tickSpacing)) expect(liquidityGross).to.eq(250) expect(feeGrowthOutside0X128).to.eq(0) expect(feeGrowthOutside1X128).to.eq(0) }) it('does not write an observation', async () => { checkObservationEquals(await pool.observations(0), { tickCumulative: 0, blockTimestamp: TEST_POOL_START_TIME, initialized: true, secondsPerLiquidityCumulativeX128: 0, }) await pool.advanceTime(1) await mint(wallet.address, -240, 0, 100) checkObservationEquals(await pool.observations(0), { tickCumulative: 0, blockTimestamp: TEST_POOL_START_TIME, initialized: true, secondsPerLiquidityCumulativeX128: 0, }) }) }) describe('including current price', () => { it('price within range: transfers current price of both tokens', async () => { await expect(mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, 100)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 317) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 32) expect(await token0.balanceOf(pool.address)).to.eq(9996 + 317) expect(await token1.balanceOf(pool.address)).to.eq(1000 + 32) }) it('initializes lower tick', async () => { await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, 100) const { liquidityGross } = await pool.ticks(minTick + tickSpacing) expect(liquidityGross).to.eq(100) }) it('initializes upper tick', async () => { await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, 100) const { liquidityGross } = await pool.ticks(maxTick - tickSpacing) expect(liquidityGross).to.eq(100) }) it('works for min/max tick', async () => { await expect(mint(wallet.address, minTick, maxTick, 10000)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 31623) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 3163) expect(await token0.balanceOf(pool.address)).to.eq(9996 + 31623) expect(await token1.balanceOf(pool.address)).to.eq(1000 + 3163) }) it('removing works', async () => { await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, 100) await pool.burn(minTick + tickSpacing, maxTick - tickSpacing, 100) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, minTick + tickSpacing, maxTick - tickSpacing, MaxUint128, MaxUint128 ) expect(amount0, 'amount0').to.eq(316) expect(amount1, 'amount1').to.eq(31) }) it('writes an observation', async () => { checkObservationEquals(await pool.observations(0), { tickCumulative: 0, blockTimestamp: TEST_POOL_START_TIME, initialized: true, secondsPerLiquidityCumulativeX128: 0, }) await pool.advanceTime(1) await mint(wallet.address, minTick, maxTick, 100) checkObservationEquals(await pool.observations(0), { tickCumulative: -23028, blockTimestamp: TEST_POOL_START_TIME + 1, initialized: true, secondsPerLiquidityCumulativeX128: '107650226801941937191829992860413859', }) }) }) describe('below current price', () => { it('transfers token1 only', async () => { await expect(mint(wallet.address, -46080, -23040, 10000)) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 2162) .to.not.emit(token0, 'Transfer') expect(await token0.balanceOf(pool.address)).to.eq(9996) expect(await token1.balanceOf(pool.address)).to.eq(1000 + 2162) }) it('min tick with max leverage', async () => { await mint(wallet.address, minTick, minTick + tickSpacing, BigNumber.from(2).pow(102)) expect(await token0.balanceOf(pool.address)).to.eq(9996) expect(await token1.balanceOf(pool.address)).to.eq(1000 + 828011520) }) it('works for min tick', async () => { await expect(mint(wallet.address, minTick, -23040, 10000)) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 3161) expect(await token0.balanceOf(pool.address)).to.eq(9996) expect(await token1.balanceOf(pool.address)).to.eq(1000 + 3161) }) it('removing works', async () => { await mint(wallet.address, -46080, -46020, 10000) await pool.burn(-46080, -46020, 10000) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, -46080, -46020, MaxUint128, MaxUint128 ) expect(amount0, 'amount0').to.eq(0) expect(amount1, 'amount1').to.eq(3) }) it('does not write an observation', async () => { checkObservationEquals(await pool.observations(0), { tickCumulative: 0, blockTimestamp: TEST_POOL_START_TIME, initialized: true, secondsPerLiquidityCumulativeX128: 0, }) await pool.advanceTime(1) await mint(wallet.address, -46080, -23040, 100) checkObservationEquals(await pool.observations(0), { tickCumulative: 0, blockTimestamp: TEST_POOL_START_TIME, initialized: true, secondsPerLiquidityCumulativeX128: 0, }) }) }) }) it('protocol fees accumulate as expected during swap', async () => { await pool.setFeeProtocol(6, 6) await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, expandTo18Decimals(1)) await swapExact0For1(expandTo18Decimals(1).div(10), wallet.address) await swapExact1For0(expandTo18Decimals(1).div(100), wallet.address) let { token0: token0ProtocolFees, token1: token1ProtocolFees } = await pool.protocolFees() expect(token0ProtocolFees).to.eq('50000000000000') expect(token1ProtocolFees).to.eq('5000000000000') }) it('positions are protected before protocol fee is turned on', async () => { await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, expandTo18Decimals(1)) await swapExact0For1(expandTo18Decimals(1).div(10), wallet.address) await swapExact1For0(expandTo18Decimals(1).div(100), wallet.address) let { token0: token0ProtocolFees, token1: token1ProtocolFees } = await pool.protocolFees() expect(token0ProtocolFees).to.eq(0) expect(token1ProtocolFees).to.eq(0) await pool.setFeeProtocol(6, 6) ;({ token0: token0ProtocolFees, token1: token1ProtocolFees } = await pool.protocolFees()) expect(token0ProtocolFees).to.eq(0) expect(token1ProtocolFees).to.eq(0) }) it('poke is not allowed on uninitialized position', async () => { await mint(other.address, minTick + tickSpacing, maxTick - tickSpacing, expandTo18Decimals(1)) await swapExact0For1(expandTo18Decimals(1).div(10), wallet.address) await swapExact1For0(expandTo18Decimals(1).div(100), wallet.address) // missing revert reason due to hardhat await expect(pool.burn(minTick + tickSpacing, maxTick - tickSpacing, 0)).to.be.reverted await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, 1) let { liquidity, feeGrowthInside0LastX128, feeGrowthInside1LastX128, tokensOwed1, tokensOwed0, } = await pool.positions(getPositionKey(wallet.address, minTick + tickSpacing, maxTick - tickSpacing)) expect(liquidity).to.eq(1) expect(feeGrowthInside0LastX128).to.eq('102084710076281216349243831104605583') expect(feeGrowthInside1LastX128).to.eq('10208471007628121634924383110460558') expect(tokensOwed0, 'tokens owed 0 before').to.eq(0) expect(tokensOwed1, 'tokens owed 1 before').to.eq(0) await pool.burn(minTick + tickSpacing, maxTick - tickSpacing, 1) ;({ liquidity, feeGrowthInside0LastX128, feeGrowthInside1LastX128, tokensOwed1, tokensOwed0, } = await pool.positions(getPositionKey(wallet.address, minTick + tickSpacing, maxTick - tickSpacing))) expect(liquidity).to.eq(0) expect(feeGrowthInside0LastX128).to.eq('102084710076281216349243831104605583') expect(feeGrowthInside1LastX128).to.eq('10208471007628121634924383110460558') expect(tokensOwed0, 'tokens owed 0 after').to.eq(3) expect(tokensOwed1, 'tokens owed 1 after').to.eq(0) }) }) }) describe('#burn', () => { beforeEach('initialize at zero tick', () => initializeAtZeroTick(pool)) async function checkTickIsClear(tick: number) { const { liquidityGross, feeGrowthOutside0X128, feeGrowthOutside1X128, liquidityNet } = await pool.ticks(tick) expect(liquidityGross).to.eq(0) expect(feeGrowthOutside0X128).to.eq(0) expect(feeGrowthOutside1X128).to.eq(0) expect(liquidityNet).to.eq(0) } async function checkTickIsNotClear(tick: number) { const { liquidityGross } = await pool.ticks(tick) expect(liquidityGross).to.not.eq(0) } it('does not clear the position fee growth snapshot if no more liquidity', async () => { // some activity that would make the ticks non-zero await pool.advanceTime(10) await mint(other.address, minTick, maxTick, expandTo18Decimals(1)) await swapExact0For1(expandTo18Decimals(1), wallet.address) await swapExact1For0(expandTo18Decimals(1), wallet.address) await pool.connect(other).burn(minTick, maxTick, expandTo18Decimals(1)) const { liquidity, tokensOwed0, tokensOwed1, feeGrowthInside0LastX128, feeGrowthInside1LastX128, } = await pool.positions(getPositionKey(other.address, minTick, maxTick)) expect(liquidity).to.eq(0) expect(tokensOwed0).to.not.eq(0) expect(tokensOwed1).to.not.eq(0) expect(feeGrowthInside0LastX128).to.eq('340282366920938463463374607431768211') expect(feeGrowthInside1LastX128).to.eq('340282366920938576890830247744589365') }) it('clears the tick if its the last position using it', async () => { const tickLower = minTick + tickSpacing const tickUpper = maxTick - tickSpacing // some activity that would make the ticks non-zero await pool.advanceTime(10) await mint(wallet.address, tickLower, tickUpper, 1) await swapExact0For1(expandTo18Decimals(1), wallet.address) await pool.burn(tickLower, tickUpper, 1) await checkTickIsClear(tickLower) await checkTickIsClear(tickUpper) }) it('clears only the lower tick if upper is still used', async () => { const tickLower = minTick + tickSpacing const tickUpper = maxTick - tickSpacing // some activity that would make the ticks non-zero await pool.advanceTime(10) await mint(wallet.address, tickLower, tickUpper, 1) await mint(wallet.address, tickLower + tickSpacing, tickUpper, 1) await swapExact0For1(expandTo18Decimals(1), wallet.address) await pool.burn(tickLower, tickUpper, 1) await checkTickIsClear(tickLower) await checkTickIsNotClear(tickUpper) }) it('clears only the upper tick if lower is still used', async () => { const tickLower = minTick + tickSpacing const tickUpper = maxTick - tickSpacing // some activity that would make the ticks non-zero await pool.advanceTime(10) await mint(wallet.address, tickLower, tickUpper, 1) await mint(wallet.address, tickLower, tickUpper - tickSpacing, 1) await swapExact0For1(expandTo18Decimals(1), wallet.address) await pool.burn(tickLower, tickUpper, 1) await checkTickIsNotClear(tickLower) await checkTickIsClear(tickUpper) }) }) // the combined amount of liquidity that the pool is initialized with (including the 1 minimum liquidity that is burned) const initializeLiquidityAmount = expandTo18Decimals(2) async function initializeAtZeroTick(pool: MockTimeUniswapV3Pool): Promise { await pool.initialize(encodePriceSqrt(1, 1)) const tickSpacing = await pool.tickSpacing() const [min, max] = [getMinTick(tickSpacing), getMaxTick(tickSpacing)] await mint(wallet.address, min, max, initializeLiquidityAmount) } describe('#observe', () => { beforeEach(() => initializeAtZeroTick(pool)) // zero tick it('current tick accumulator increases by tick over time', async () => { let { tickCumulatives: [tickCumulative], } = await pool.observe([0]) expect(tickCumulative).to.eq(0) await pool.advanceTime(10) ;({ tickCumulatives: [tickCumulative], } = await pool.observe([0])) expect(tickCumulative).to.eq(0) }) it('current tick accumulator after single swap', async () => { // moves to tick -1 await swapExact0For1(1000, wallet.address) await pool.advanceTime(4) let { tickCumulatives: [tickCumulative], } = await pool.observe([0]) expect(tickCumulative).to.eq(-4) }) it('current tick accumulator after two swaps', async () => { await swapExact0For1(expandTo18Decimals(1).div(2), wallet.address) expect((await pool.slot0()).tick).to.eq(-4452) await pool.advanceTime(4) await swapExact1For0(expandTo18Decimals(1).div(4), wallet.address) expect((await pool.slot0()).tick).to.eq(-1558) await pool.advanceTime(6) let { tickCumulatives: [tickCumulative], } = await pool.observe([0]) // -4452*4 + -1558*6 expect(tickCumulative).to.eq(-27156) }) }) describe('miscellaneous mint tests', () => { beforeEach('initialize at zero tick', async () => { pool = await createPool(FeeAmount.LOW, TICK_SPACINGS[FeeAmount.LOW]) await initializeAtZeroTick(pool) }) it('mint to the right of the current price', async () => { const liquidityDelta = 1000 const lowerTick = tickSpacing const upperTick = tickSpacing * 2 const liquidityBefore = await pool.liquidity() const b0 = await token0.balanceOf(pool.address) const b1 = await token1.balanceOf(pool.address) await mint(wallet.address, lowerTick, upperTick, liquidityDelta) const liquidityAfter = await pool.liquidity() expect(liquidityAfter).to.be.gte(liquidityBefore) expect((await token0.balanceOf(pool.address)).sub(b0)).to.eq(1) expect((await token1.balanceOf(pool.address)).sub(b1)).to.eq(0) }) it('mint to the left of the current price', async () => { const liquidityDelta = 1000 const lowerTick = -tickSpacing * 2 const upperTick = -tickSpacing const liquidityBefore = await pool.liquidity() const b0 = await token0.balanceOf(pool.address) const b1 = await token1.balanceOf(pool.address) await mint(wallet.address, lowerTick, upperTick, liquidityDelta) const liquidityAfter = await pool.liquidity() expect(liquidityAfter).to.be.gte(liquidityBefore) expect((await token0.balanceOf(pool.address)).sub(b0)).to.eq(0) expect((await token1.balanceOf(pool.address)).sub(b1)).to.eq(1) }) it('mint within the current price', async () => { const liquidityDelta = 1000 const lowerTick = -tickSpacing const upperTick = tickSpacing const liquidityBefore = await pool.liquidity() const b0 = await token0.balanceOf(pool.address) const b1 = await token1.balanceOf(pool.address) await mint(wallet.address, lowerTick, upperTick, liquidityDelta) const liquidityAfter = await pool.liquidity() expect(liquidityAfter).to.be.gte(liquidityBefore) expect((await token0.balanceOf(pool.address)).sub(b0)).to.eq(1) expect((await token1.balanceOf(pool.address)).sub(b1)).to.eq(1) }) it('cannot remove more than the entire position', async () => { const lowerTick = -tickSpacing const upperTick = tickSpacing await mint(wallet.address, lowerTick, upperTick, expandTo18Decimals(1000)) // should be 'LS', hardhat is bugged await expect(pool.burn(lowerTick, upperTick, expandTo18Decimals(1001))).to.be.reverted }) it('collect fees within the current price after swap', async () => { const liquidityDelta = expandTo18Decimals(100) const lowerTick = -tickSpacing * 100 const upperTick = tickSpacing * 100 await mint(wallet.address, lowerTick, upperTick, liquidityDelta) const liquidityBefore = await pool.liquidity() const amount0In = expandTo18Decimals(1) await swapExact0For1(amount0In, wallet.address) const liquidityAfter = await pool.liquidity() expect(liquidityAfter, 'k increases').to.be.gte(liquidityBefore) const token0BalanceBeforePool = await token0.balanceOf(pool.address) const token1BalanceBeforePool = await token1.balanceOf(pool.address) const token0BalanceBeforeWallet = await token0.balanceOf(wallet.address) const token1BalanceBeforeWallet = await token1.balanceOf(wallet.address) await pool.burn(lowerTick, upperTick, 0) await pool.collect(wallet.address, lowerTick, upperTick, MaxUint128, MaxUint128) await pool.burn(lowerTick, upperTick, 0) const { amount0: fees0, amount1: fees1 } = await pool.callStatic.collect( wallet.address, lowerTick, upperTick, MaxUint128, MaxUint128 ) expect(fees0).to.be.eq(0) expect(fees1).to.be.eq(0) const token0BalanceAfterWallet = await token0.balanceOf(wallet.address) const token1BalanceAfterWallet = await token1.balanceOf(wallet.address) const token0BalanceAfterPool = await token0.balanceOf(pool.address) const token1BalanceAfterPool = await token1.balanceOf(pool.address) expect(token0BalanceAfterWallet).to.be.gt(token0BalanceBeforeWallet) expect(token1BalanceAfterWallet).to.be.eq(token1BalanceBeforeWallet) expect(token0BalanceAfterPool).to.be.lt(token0BalanceBeforePool) expect(token1BalanceAfterPool).to.be.eq(token1BalanceBeforePool) }) }) describe('post-initialize at medium fee', () => { describe('k (implicit)', () => { it('returns 0 before initialization', async () => { expect(await pool.liquidity()).to.eq(0) }) describe('post initialized', () => { beforeEach(() => initializeAtZeroTick(pool)) it('returns initial liquidity', async () => { expect(await pool.liquidity()).to.eq(expandTo18Decimals(2)) }) it('returns in supply in range', async () => { await mint(wallet.address, -tickSpacing, tickSpacing, expandTo18Decimals(3)) expect(await pool.liquidity()).to.eq(expandTo18Decimals(5)) }) it('excludes supply at tick above current tick', async () => { await mint(wallet.address, tickSpacing, tickSpacing * 2, expandTo18Decimals(3)) expect(await pool.liquidity()).to.eq(expandTo18Decimals(2)) }) it('excludes supply at tick below current tick', async () => { await mint(wallet.address, -tickSpacing * 2, -tickSpacing, expandTo18Decimals(3)) expect(await pool.liquidity()).to.eq(expandTo18Decimals(2)) }) it('updates correctly when exiting range', async () => { const kBefore = await pool.liquidity() expect(kBefore).to.be.eq(expandTo18Decimals(2)) // add liquidity at and above current tick const liquidityDelta = expandTo18Decimals(1) const lowerTick = 0 const upperTick = tickSpacing await mint(wallet.address, lowerTick, upperTick, liquidityDelta) // ensure virtual supply has increased appropriately const kAfter = await pool.liquidity() expect(kAfter).to.be.eq(expandTo18Decimals(3)) // swap toward the left (just enough for the tick transition function to trigger) await swapExact0For1(1, wallet.address) const { tick } = await pool.slot0() expect(tick).to.be.eq(-1) const kAfterSwap = await pool.liquidity() expect(kAfterSwap).to.be.eq(expandTo18Decimals(2)) }) it('updates correctly when entering range', async () => { const kBefore = await pool.liquidity() expect(kBefore).to.be.eq(expandTo18Decimals(2)) // add liquidity below the current tick const liquidityDelta = expandTo18Decimals(1) const lowerTick = -tickSpacing const upperTick = 0 await mint(wallet.address, lowerTick, upperTick, liquidityDelta) // ensure virtual supply hasn't changed const kAfter = await pool.liquidity() expect(kAfter).to.be.eq(kBefore) // swap toward the left (just enough for the tick transition function to trigger) await swapExact0For1(1, wallet.address) const { tick } = await pool.slot0() expect(tick).to.be.eq(-1) const kAfterSwap = await pool.liquidity() expect(kAfterSwap).to.be.eq(expandTo18Decimals(3)) }) }) }) }) describe('limit orders', () => { beforeEach('initialize at tick 0', () => initializeAtZeroTick(pool)) it('limit selling 0 for 1 at tick 0 thru 1', async () => { await expect(mint(wallet.address, 0, 120, expandTo18Decimals(1))) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, '5981737760509663') // somebody takes the limit order await swapExact1For0(expandTo18Decimals(2), other.address) await expect(pool.burn(0, 120, expandTo18Decimals(1))) .to.emit(pool, 'Burn') .withArgs(wallet.address, 0, 120, expandTo18Decimals(1), 0, '6017734268818165') .to.not.emit(token0, 'Transfer') .to.not.emit(token1, 'Transfer') await expect(pool.collect(wallet.address, 0, 120, MaxUint128, MaxUint128)) .to.emit(token1, 'Transfer') .withArgs(pool.address, wallet.address, BigNumber.from('6017734268818165').add('18107525382602')) // roughly 0.3% despite other liquidity .to.not.emit(token0, 'Transfer') expect((await pool.slot0()).tick).to.be.gte(120) }) it('limit selling 1 for 0 at tick 0 thru -1', async () => { await expect(mint(wallet.address, -120, 0, expandTo18Decimals(1))) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, '5981737760509663') // somebody takes the limit order await swapExact0For1(expandTo18Decimals(2), other.address) await expect(pool.burn(-120, 0, expandTo18Decimals(1))) .to.emit(pool, 'Burn') .withArgs(wallet.address, -120, 0, expandTo18Decimals(1), '6017734268818165', 0) .to.not.emit(token0, 'Transfer') .to.not.emit(token1, 'Transfer') await expect(pool.collect(wallet.address, -120, 0, MaxUint128, MaxUint128)) .to.emit(token0, 'Transfer') .withArgs(pool.address, wallet.address, BigNumber.from('6017734268818165').add('18107525382602')) // roughly 0.3% despite other liquidity expect((await pool.slot0()).tick).to.be.lt(-120) }) describe('fee is on', () => { beforeEach(() => pool.setFeeProtocol(6, 6)) it('limit selling 0 for 1 at tick 0 thru 1', async () => { await expect(mint(wallet.address, 0, 120, expandTo18Decimals(1))) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, '5981737760509663') // somebody takes the limit order await swapExact1For0(expandTo18Decimals(2), other.address) await expect(pool.burn(0, 120, expandTo18Decimals(1))) .to.emit(pool, 'Burn') .withArgs(wallet.address, 0, 120, expandTo18Decimals(1), 0, '6017734268818165') .to.not.emit(token0, 'Transfer') .to.not.emit(token1, 'Transfer') await expect(pool.collect(wallet.address, 0, 120, MaxUint128, MaxUint128)) .to.emit(token1, 'Transfer') .withArgs(pool.address, wallet.address, BigNumber.from('6017734268818165').add('15089604485501')) // roughly 0.25% despite other liquidity .to.not.emit(token0, 'Transfer') expect((await pool.slot0()).tick).to.be.gte(120) }) it('limit selling 1 for 0 at tick 0 thru -1', async () => { await expect(mint(wallet.address, -120, 0, expandTo18Decimals(1))) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, '5981737760509663') // somebody takes the limit order await swapExact0For1(expandTo18Decimals(2), other.address) await expect(pool.burn(-120, 0, expandTo18Decimals(1))) .to.emit(pool, 'Burn') .withArgs(wallet.address, -120, 0, expandTo18Decimals(1), '6017734268818165', 0) .to.not.emit(token0, 'Transfer') .to.not.emit(token1, 'Transfer') await expect(pool.collect(wallet.address, -120, 0, MaxUint128, MaxUint128)) .to.emit(token0, 'Transfer') .withArgs(pool.address, wallet.address, BigNumber.from('6017734268818165').add('15089604485501')) // roughly 0.25% despite other liquidity expect((await pool.slot0()).tick).to.be.lt(-120) }) }) }) describe('#collect', () => { beforeEach(async () => { pool = await createPool(FeeAmount.LOW, TICK_SPACINGS[FeeAmount.LOW]) await pool.initialize(encodePriceSqrt(1, 1)) }) it('works with multiple LPs', async () => { await mint(wallet.address, minTick, maxTick, expandTo18Decimals(1)) await mint(wallet.address, minTick + tickSpacing, maxTick - tickSpacing, expandTo18Decimals(2)) await swapExact0For1(expandTo18Decimals(1), wallet.address) // poke positions await pool.burn(minTick, maxTick, 0) await pool.burn(minTick + tickSpacing, maxTick - tickSpacing, 0) const { tokensOwed0: tokensOwed0Position0 } = await pool.positions( getPositionKey(wallet.address, minTick, maxTick) ) const { tokensOwed0: tokensOwed0Position1 } = await pool.positions( getPositionKey(wallet.address, minTick + tickSpacing, maxTick - tickSpacing) ) expect(tokensOwed0Position0).to.be.eq('166666666666667') expect(tokensOwed0Position1).to.be.eq('333333333333334') }) describe('works across large increases', () => { beforeEach(async () => { await mint(wallet.address, minTick, maxTick, expandTo18Decimals(1)) }) // type(uint128).max * 2**128 / 1e18 // https://www.wolframalpha.com/input/?i=%282**128+-+1%29+*+2**128+%2F+1e18 const magicNumber = BigNumber.from('115792089237316195423570985008687907852929702298719625575994') it('works just before the cap binds', async () => { await pool.setFeeGrowthGlobal0X128(magicNumber) await pool.burn(minTick, maxTick, 0) const { tokensOwed0, tokensOwed1 } = await pool.positions(getPositionKey(wallet.address, minTick, maxTick)) expect(tokensOwed0).to.be.eq(MaxUint128.sub(1)) expect(tokensOwed1).to.be.eq(0) }) it('works just after the cap binds', async () => { await pool.setFeeGrowthGlobal0X128(magicNumber.add(1)) await pool.burn(minTick, maxTick, 0) const { tokensOwed0, tokensOwed1 } = await pool.positions(getPositionKey(wallet.address, minTick, maxTick)) expect(tokensOwed0).to.be.eq(MaxUint128) expect(tokensOwed1).to.be.eq(0) }) it('works well after the cap binds', async () => { await pool.setFeeGrowthGlobal0X128(constants.MaxUint256) await pool.burn(minTick, maxTick, 0) const { tokensOwed0, tokensOwed1 } = await pool.positions(getPositionKey(wallet.address, minTick, maxTick)) expect(tokensOwed0).to.be.eq(MaxUint128) expect(tokensOwed1).to.be.eq(0) }) }) describe('works across overflow boundaries', () => { beforeEach(async () => { await pool.setFeeGrowthGlobal0X128(constants.MaxUint256) await pool.setFeeGrowthGlobal1X128(constants.MaxUint256) await mint(wallet.address, minTick, maxTick, expandTo18Decimals(10)) }) it('token0', async () => { await swapExact0For1(expandTo18Decimals(1), wallet.address) await pool.burn(minTick, maxTick, 0) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, minTick, maxTick, MaxUint128, MaxUint128 ) expect(amount0).to.be.eq('499999999999999') expect(amount1).to.be.eq(0) }) it('token1', async () => { await swapExact1For0(expandTo18Decimals(1), wallet.address) await pool.burn(minTick, maxTick, 0) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, minTick, maxTick, MaxUint128, MaxUint128 ) expect(amount0).to.be.eq(0) expect(amount1).to.be.eq('499999999999999') }) it('token0 and token1', async () => { await swapExact0For1(expandTo18Decimals(1), wallet.address) await swapExact1For0(expandTo18Decimals(1), wallet.address) await pool.burn(minTick, maxTick, 0) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, minTick, maxTick, MaxUint128, MaxUint128 ) expect(amount0).to.be.eq('499999999999999') expect(amount1).to.be.eq('500000000000000') }) }) }) describe('#feeProtocol', () => { const liquidityAmount = expandTo18Decimals(1000) beforeEach(async () => { pool = await createPool(FeeAmount.LOW, TICK_SPACINGS[FeeAmount.LOW]) await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, liquidityAmount) }) it('is initially set to 0', async () => { expect((await pool.slot0()).feeProtocol).to.eq(0) }) it('can be changed by the owner', async () => { await pool.setFeeProtocol(6, 6) expect((await pool.slot0()).feeProtocol).to.eq(102) }) it('cannot be changed out of bounds', async () => { await expect(pool.setFeeProtocol(3, 3)).to.be.reverted await expect(pool.setFeeProtocol(11, 11)).to.be.reverted }) it('cannot be changed by addresses that are not owner', async () => { await expect(pool.connect(other).setFeeProtocol(6, 6)).to.be.reverted }) async function swapAndGetFeesOwed({ amount, zeroForOne, poke, }: { amount: BigNumberish zeroForOne: boolean poke: boolean }) { await (zeroForOne ? swapExact0For1(amount, wallet.address) : swapExact1For0(amount, wallet.address)) if (poke) await pool.burn(minTick, maxTick, 0) const { amount0: fees0, amount1: fees1 } = await pool.callStatic.collect( wallet.address, minTick, maxTick, MaxUint128, MaxUint128 ) expect(fees0, 'fees owed in token0 are greater than 0').to.be.gte(0) expect(fees1, 'fees owed in token1 are greater than 0').to.be.gte(0) return { token0Fees: fees0, token1Fees: fees1 } } it('position owner gets full fees when protocol fee is off', async () => { const { token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, }) // 6 bips * 1e18 expect(token0Fees).to.eq('499999999999999') expect(token1Fees).to.eq(0) }) it('swap fees accumulate as expected (0 for 1)', async () => { let token0Fees let token1Fees ;({ token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, })) expect(token0Fees).to.eq('499999999999999') expect(token1Fees).to.eq(0) ;({ token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, })) expect(token0Fees).to.eq('999999999999998') expect(token1Fees).to.eq(0) ;({ token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, })) expect(token0Fees).to.eq('1499999999999997') expect(token1Fees).to.eq(0) }) it('swap fees accumulate as expected (1 for 0)', async () => { let token0Fees let token1Fees ;({ token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: false, poke: true, })) expect(token0Fees).to.eq(0) expect(token1Fees).to.eq('499999999999999') ;({ token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: false, poke: true, })) expect(token0Fees).to.eq(0) expect(token1Fees).to.eq('999999999999998') ;({ token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: false, poke: true, })) expect(token0Fees).to.eq(0) expect(token1Fees).to.eq('1499999999999997') }) it('position owner gets partial fees when protocol fee is on', async () => { await pool.setFeeProtocol(6, 6) const { token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, }) expect(token0Fees).to.be.eq('416666666666666') expect(token1Fees).to.be.eq(0) }) describe('#collectProtocol', () => { it('returns 0 if no fees', async () => { await pool.setFeeProtocol(6, 6) const { amount0, amount1 } = await pool.callStatic.collectProtocol(wallet.address, MaxUint128, MaxUint128) expect(amount0).to.be.eq(0) expect(amount1).to.be.eq(0) }) it('can collect fees', async () => { await pool.setFeeProtocol(6, 6) await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, }) await expect(pool.collectProtocol(other.address, MaxUint128, MaxUint128)) .to.emit(token0, 'Transfer') .withArgs(pool.address, other.address, '83333333333332') }) it('fees collected can differ between token0 and token1', async () => { await pool.setFeeProtocol(8, 5) await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: false, }) await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: false, poke: false, }) await expect(pool.collectProtocol(other.address, MaxUint128, MaxUint128)) .to.emit(token0, 'Transfer') // more token0 fees because it's 1/5th the swap fees .withArgs(pool.address, other.address, '62499999999999') .to.emit(token1, 'Transfer') // less token1 fees because it's 1/8th the swap fees .withArgs(pool.address, other.address, '99999999999998') }) }) it('fees collected by lp after two swaps should be double one swap', async () => { await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, }) const { token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, }) // 6 bips * 2e18 expect(token0Fees).to.eq('999999999999998') expect(token1Fees).to.eq(0) }) it('fees collected after two swaps with fee turned on in middle are fees from last swap (not confiscatory)', async () => { await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: false, }) await pool.setFeeProtocol(6, 6) const { token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, }) expect(token0Fees).to.eq('916666666666666') expect(token1Fees).to.eq(0) }) it('fees collected by lp after two swaps with intermediate withdrawal', async () => { await pool.setFeeProtocol(6, 6) const { token0Fees, token1Fees } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: true, }) expect(token0Fees).to.eq('416666666666666') expect(token1Fees).to.eq(0) // collect the fees await pool.collect(wallet.address, minTick, maxTick, MaxUint128, MaxUint128) const { token0Fees: token0FeesNext, token1Fees: token1FeesNext } = await swapAndGetFeesOwed({ amount: expandTo18Decimals(1), zeroForOne: true, poke: false, }) expect(token0FeesNext).to.eq(0) expect(token1FeesNext).to.eq(0) let { token0: token0ProtocolFees, token1: token1ProtocolFees } = await pool.protocolFees() expect(token0ProtocolFees).to.eq('166666666666666') expect(token1ProtocolFees).to.eq(0) await pool.burn(minTick, maxTick, 0) // poke to update fees await expect(pool.collect(wallet.address, minTick, maxTick, MaxUint128, MaxUint128)) .to.emit(token0, 'Transfer') .withArgs(pool.address, wallet.address, '416666666666666') ;({ token0: token0ProtocolFees, token1: token1ProtocolFees } = await pool.protocolFees()) expect(token0ProtocolFees).to.eq('166666666666666') expect(token1ProtocolFees).to.eq(0) }) }) describe('#tickSpacing', () => { describe('tickSpacing = 12', () => { beforeEach('deploy pool', async () => { pool = await createPool(FeeAmount.MEDIUM, 12) }) describe('post initialize', () => { beforeEach('initialize pool', async () => { await pool.initialize(encodePriceSqrt(1, 1)) }) it('mint can only be called for multiples of 12', async () => { await expect(mint(wallet.address, -6, 0, 1)).to.be.reverted await expect(mint(wallet.address, 0, 6, 1)).to.be.reverted }) it('mint can be called with multiples of 12', async () => { await mint(wallet.address, 12, 24, 1) await mint(wallet.address, -144, -120, 1) }) it('swapping across gaps works in 1 for 0 direction', async () => { const liquidityAmount = expandTo18Decimals(1).div(4) await mint(wallet.address, 120000, 121200, liquidityAmount) await swapExact1For0(expandTo18Decimals(1), wallet.address) await expect(pool.burn(120000, 121200, liquidityAmount)) .to.emit(pool, 'Burn') .withArgs(wallet.address, 120000, 121200, liquidityAmount, '30027458295511', '996999999999999999') .to.not.emit(token0, 'Transfer') .to.not.emit(token1, 'Transfer') expect((await pool.slot0()).tick).to.eq(120196) }) it('swapping across gaps works in 0 for 1 direction', async () => { const liquidityAmount = expandTo18Decimals(1).div(4) await mint(wallet.address, -121200, -120000, liquidityAmount) await swapExact0For1(expandTo18Decimals(1), wallet.address) await expect(pool.burn(-121200, -120000, liquidityAmount)) .to.emit(pool, 'Burn') .withArgs(wallet.address, -121200, -120000, liquidityAmount, '996999999999999999', '30027458295511') .to.not.emit(token0, 'Transfer') .to.not.emit(token1, 'Transfer') expect((await pool.slot0()).tick).to.eq(-120197) }) }) }) }) // https://github.com/Uniswap/uniswap-v3-core/issues/214 it('tick transition cannot run twice if zero for one swap ends at fractional price just below tick', async () => { pool = await createPool(FeeAmount.MEDIUM, 1) const sqrtTickMath = (await (await ethers.getContractFactory('TickMathTest')).deploy()) as TickMathTest const swapMath = (await (await ethers.getContractFactory('SwapMathTest')).deploy()) as SwapMathTest const p0 = (await sqrtTickMath.getSqrtRatioAtTick(-24081)).add(1) // initialize at a price of ~0.3 token1/token0 // meaning if you swap in 2 token0, you should end up getting 0 token1 await pool.initialize(p0) expect(await pool.liquidity(), 'current pool liquidity is 1').to.eq(0) expect((await pool.slot0()).tick, 'pool tick is -24081').to.eq(-24081) // add a bunch of liquidity around current price const liquidity = expandTo18Decimals(1000) await mint(wallet.address, -24082, -24080, liquidity) expect(await pool.liquidity(), 'current pool liquidity is now liquidity + 1').to.eq(liquidity) await mint(wallet.address, -24082, -24081, liquidity) expect(await pool.liquidity(), 'current pool liquidity is still liquidity + 1').to.eq(liquidity) // check the math works out to moving the price down 1, sending no amount out, and having some amount remaining { const { feeAmount, amountIn, amountOut, sqrtQ } = await swapMath.computeSwapStep( p0, p0.sub(1), liquidity, 3, FeeAmount.MEDIUM ) expect(sqrtQ, 'price moves').to.eq(p0.sub(1)) expect(feeAmount, 'fee amount is 1').to.eq(1) expect(amountIn, 'amount in is 1').to.eq(1) expect(amountOut, 'zero amount out').to.eq(0) } // swap 2 amount in, should get 0 amount out await expect(swapExact0For1(3, wallet.address)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 3) .to.not.emit(token1, 'Transfer') const { tick, sqrtPriceX96 } = await pool.slot0() expect(tick, 'pool is at the next tick').to.eq(-24082) expect(sqrtPriceX96, 'pool price is still on the p0 boundary').to.eq(p0.sub(1)) expect(await pool.liquidity(), 'pool has run tick transition and liquidity changed').to.eq(liquidity.mul(2)) }) describe('#flash', () => { it('fails if not initialized', async () => { await expect(flash(100, 200, other.address)).to.be.reverted await expect(flash(100, 0, other.address)).to.be.reverted await expect(flash(0, 200, other.address)).to.be.reverted }) it('fails if no liquidity', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await expect(flash(100, 200, other.address)).to.be.revertedWith('L') await expect(flash(100, 0, other.address)).to.be.revertedWith('L') await expect(flash(0, 200, other.address)).to.be.revertedWith('L') }) describe('after liquidity added', () => { let balance0: BigNumber let balance1: BigNumber beforeEach('add some tokens', async () => { await initializeAtZeroTick(pool) ;[balance0, balance1] = await Promise.all([token0.balanceOf(pool.address), token1.balanceOf(pool.address)]) }) describe('fee off', () => { it('emits an event', async () => { await expect(flash(1001, 2001, other.address)) .to.emit(pool, 'Flash') .withArgs(swapTarget.address, other.address, 1001, 2001, 4, 7) }) it('transfers the amount0 to the recipient', async () => { await expect(flash(100, 200, other.address)) .to.emit(token0, 'Transfer') .withArgs(pool.address, other.address, 100) }) it('transfers the amount1 to the recipient', async () => { await expect(flash(100, 200, other.address)) .to.emit(token1, 'Transfer') .withArgs(pool.address, other.address, 200) }) it('can flash only token0', async () => { await expect(flash(101, 0, other.address)) .to.emit(token0, 'Transfer') .withArgs(pool.address, other.address, 101) .to.not.emit(token1, 'Transfer') }) it('can flash only token1', async () => { await expect(flash(0, 102, other.address)) .to.emit(token1, 'Transfer') .withArgs(pool.address, other.address, 102) .to.not.emit(token0, 'Transfer') }) it('can flash entire token balance', async () => { await expect(flash(balance0, balance1, other.address)) .to.emit(token0, 'Transfer') .withArgs(pool.address, other.address, balance0) .to.emit(token1, 'Transfer') .withArgs(pool.address, other.address, balance1) }) it('no-op if both amounts are 0', async () => { await expect(flash(0, 0, other.address)).to.not.emit(token0, 'Transfer').to.not.emit(token1, 'Transfer') }) it('fails if flash amount is greater than token balance', async () => { await expect(flash(balance0.add(1), balance1, other.address)).to.be.reverted await expect(flash(balance0, balance1.add(1), other.address)).to.be.reverted }) it('calls the flash callback on the sender with correct fee amounts', async () => { await expect(flash(1001, 2002, other.address)).to.emit(swapTarget, 'FlashCallback').withArgs(4, 7) }) it('increases the fee growth by the expected amount', async () => { await flash(1001, 2002, other.address) expect(await pool.feeGrowthGlobal0X128()).to.eq( BigNumber.from(4).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) expect(await pool.feeGrowthGlobal1X128()).to.eq( BigNumber.from(7).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) it('fails if original balance not returned in either token', async () => { await expect(flash(1000, 0, other.address, 999, 0)).to.be.reverted await expect(flash(0, 1000, other.address, 0, 999)).to.be.reverted }) it('fails if underpays either token', async () => { await expect(flash(1000, 0, other.address, 1002, 0)).to.be.reverted await expect(flash(0, 1000, other.address, 0, 1002)).to.be.reverted }) it('allows donating token0', async () => { await expect(flash(0, 0, constants.AddressZero, 567, 0)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 567) .to.not.emit(token1, 'Transfer') expect(await pool.feeGrowthGlobal0X128()).to.eq( BigNumber.from(567).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) it('allows donating token1', async () => { await expect(flash(0, 0, constants.AddressZero, 0, 678)) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 678) .to.not.emit(token0, 'Transfer') expect(await pool.feeGrowthGlobal1X128()).to.eq( BigNumber.from(678).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) it('allows donating token0 and token1 together', async () => { await expect(flash(0, 0, constants.AddressZero, 789, 1234)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 789) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 1234) expect(await pool.feeGrowthGlobal0X128()).to.eq( BigNumber.from(789).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) expect(await pool.feeGrowthGlobal1X128()).to.eq( BigNumber.from(1234).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) }) describe('fee on', () => { beforeEach('turn protocol fee on', async () => { await pool.setFeeProtocol(6, 6) }) it('emits an event', async () => { await expect(flash(1001, 2001, other.address)) .to.emit(pool, 'Flash') .withArgs(swapTarget.address, other.address, 1001, 2001, 4, 7) }) it('increases the fee growth by the expected amount', async () => { await flash(2002, 4004, other.address) const { token0: token0ProtocolFees, token1: token1ProtocolFees } = await pool.protocolFees() expect(token0ProtocolFees).to.eq(1) expect(token1ProtocolFees).to.eq(2) expect(await pool.feeGrowthGlobal0X128()).to.eq( BigNumber.from(6).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) expect(await pool.feeGrowthGlobal1X128()).to.eq( BigNumber.from(11).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) it('allows donating token0', async () => { await expect(flash(0, 0, constants.AddressZero, 567, 0)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 567) .to.not.emit(token1, 'Transfer') const { token0: token0ProtocolFees } = await pool.protocolFees() expect(token0ProtocolFees).to.eq(94) expect(await pool.feeGrowthGlobal0X128()).to.eq( BigNumber.from(473).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) it('allows donating token1', async () => { await expect(flash(0, 0, constants.AddressZero, 0, 678)) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 678) .to.not.emit(token0, 'Transfer') const { token1: token1ProtocolFees } = await pool.protocolFees() expect(token1ProtocolFees).to.eq(113) expect(await pool.feeGrowthGlobal1X128()).to.eq( BigNumber.from(565).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) it('allows donating token0 and token1 together', async () => { await expect(flash(0, 0, constants.AddressZero, 789, 1234)) .to.emit(token0, 'Transfer') .withArgs(wallet.address, pool.address, 789) .to.emit(token1, 'Transfer') .withArgs(wallet.address, pool.address, 1234) const { token0: token0ProtocolFees, token1: token1ProtocolFees } = await pool.protocolFees() expect(token0ProtocolFees).to.eq(131) expect(token1ProtocolFees).to.eq(205) expect(await pool.feeGrowthGlobal0X128()).to.eq( BigNumber.from(658).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) expect(await pool.feeGrowthGlobal1X128()).to.eq( BigNumber.from(1029).mul(BigNumber.from(2).pow(128)).div(expandTo18Decimals(2)) ) }) }) }) }) describe('#increaseObservationCardinalityNext', () => { it('cannot be called before initialization', async () => { await expect(pool.increaseObservationCardinalityNext(2)).to.be.reverted }) describe('after initialization', () => { beforeEach('initialize the pool', () => pool.initialize(encodePriceSqrt(1, 1))) it('oracle starting state after initialization', async () => { const { observationCardinality, observationIndex, observationCardinalityNext } = await pool.slot0() expect(observationCardinality).to.eq(1) expect(observationIndex).to.eq(0) expect(observationCardinalityNext).to.eq(1) const { secondsPerLiquidityCumulativeX128, tickCumulative, initialized, blockTimestamp, } = await pool.observations(0) expect(secondsPerLiquidityCumulativeX128).to.eq(0) expect(tickCumulative).to.eq(0) expect(initialized).to.eq(true) expect(blockTimestamp).to.eq(TEST_POOL_START_TIME) }) it('increases observation cardinality next', async () => { await pool.increaseObservationCardinalityNext(2) const { observationCardinality, observationIndex, observationCardinalityNext } = await pool.slot0() expect(observationCardinality).to.eq(1) expect(observationIndex).to.eq(0) expect(observationCardinalityNext).to.eq(2) }) it('is no op if target is already exceeded', async () => { await pool.increaseObservationCardinalityNext(5) await pool.increaseObservationCardinalityNext(3) const { observationCardinality, observationIndex, observationCardinalityNext } = await pool.slot0() expect(observationCardinality).to.eq(1) expect(observationIndex).to.eq(0) expect(observationCardinalityNext).to.eq(5) }) }) }) describe('#setFeeProtocol', () => { beforeEach('initialize the pool', async () => { await pool.initialize(encodePriceSqrt(1, 1)) }) it('can only be called by factory owner', async () => { await expect(pool.connect(other).setFeeProtocol(5, 5)).to.be.reverted }) it('fails if fee is lt 4 or gt 10', async () => { await expect(pool.setFeeProtocol(3, 3)).to.be.reverted await expect(pool.setFeeProtocol(6, 3)).to.be.reverted await expect(pool.setFeeProtocol(3, 6)).to.be.reverted await expect(pool.setFeeProtocol(11, 11)).to.be.reverted await expect(pool.setFeeProtocol(6, 11)).to.be.reverted await expect(pool.setFeeProtocol(11, 6)).to.be.reverted }) it('succeeds for fee of 4', async () => { await pool.setFeeProtocol(4, 4) }) it('succeeds for fee of 10', async () => { await pool.setFeeProtocol(10, 10) }) it('sets protocol fee', async () => { await pool.setFeeProtocol(7, 7) expect((await pool.slot0()).feeProtocol).to.eq(119) }) it('can change protocol fee', async () => { await pool.setFeeProtocol(7, 7) await pool.setFeeProtocol(5, 8) expect((await pool.slot0()).feeProtocol).to.eq(133) }) it('can turn off protocol fee', async () => { await pool.setFeeProtocol(4, 4) await pool.setFeeProtocol(0, 0) expect((await pool.slot0()).feeProtocol).to.eq(0) }) it('emits an event when turned on', async () => { await expect(pool.setFeeProtocol(7, 7)).to.be.emit(pool, 'SetFeeProtocol').withArgs(0, 0, 7, 7) }) it('emits an event when turned off', async () => { await pool.setFeeProtocol(7, 5) await expect(pool.setFeeProtocol(0, 0)).to.be.emit(pool, 'SetFeeProtocol').withArgs(7, 5, 0, 0) }) it('emits an event when changed', async () => { await pool.setFeeProtocol(4, 10) await expect(pool.setFeeProtocol(6, 8)).to.be.emit(pool, 'SetFeeProtocol').withArgs(4, 10, 6, 8) }) it('emits an event when unchanged', async () => { await pool.setFeeProtocol(5, 9) await expect(pool.setFeeProtocol(5, 9)).to.be.emit(pool, 'SetFeeProtocol').withArgs(5, 9, 5, 9) }) }) describe('#lock', () => { beforeEach('initialize the pool', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, expandTo18Decimals(1)) }) it('cannot reenter from swap callback', async () => { const reentrant = (await ( await ethers.getContractFactory('TestUniswapV3ReentrantCallee') ).deploy()) as TestUniswapV3ReentrantCallee // the tests happen in solidity await expect(reentrant.swapToReenter(pool.address)).to.be.revertedWith('Unable to reenter') }) }) describe('#snapshotCumulativesInside', () => { const tickLower = -TICK_SPACINGS[FeeAmount.MEDIUM] const tickUpper = TICK_SPACINGS[FeeAmount.MEDIUM] const tickSpacing = TICK_SPACINGS[FeeAmount.MEDIUM] beforeEach(async () => { await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, tickLower, tickUpper, 10) }) it('throws if ticks are in reverse order', async () => { await expect(pool.snapshotCumulativesInside(tickUpper, tickLower)).to.be.reverted }) it('throws if ticks are the same', async () => { await expect(pool.snapshotCumulativesInside(tickUpper, tickUpper)).to.be.reverted }) it('throws if tick lower is too low', async () => { await expect(pool.snapshotCumulativesInside(getMinTick(tickSpacing) - 1, tickUpper)).be.reverted }) it('throws if tick upper is too high', async () => { await expect(pool.snapshotCumulativesInside(tickLower, getMaxTick(tickSpacing) + 1)).be.reverted }) it('throws if tick lower is not initialized', async () => { await expect(pool.snapshotCumulativesInside(tickLower - tickSpacing, tickUpper)).to.be.reverted }) it('throws if tick upper is not initialized', async () => { await expect(pool.snapshotCumulativesInside(tickLower, tickUpper + tickSpacing)).to.be.reverted }) it('is zero immediately after initialize', async () => { const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickLower, tickUpper) expect(secondsPerLiquidityInsideX128).to.eq(0) expect(tickCumulativeInside).to.eq(0) expect(secondsInside).to.eq(0) }) it('increases by expected amount when time elapses in the range', async () => { await pool.advanceTime(5) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickLower, tickUpper) expect(secondsPerLiquidityInsideX128).to.eq(BigNumber.from(5).shl(128).div(10)) expect(tickCumulativeInside, 'tickCumulativeInside').to.eq(0) expect(secondsInside).to.eq(5) }) it('does not account for time increase above range', async () => { await pool.advanceTime(5) await swapToHigherPrice(encodePriceSqrt(2, 1), wallet.address) await pool.advanceTime(7) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickLower, tickUpper) expect(secondsPerLiquidityInsideX128).to.eq(BigNumber.from(5).shl(128).div(10)) expect(tickCumulativeInside, 'tickCumulativeInside').to.eq(0) expect(secondsInside).to.eq(5) }) it('does not account for time increase below range', async () => { await pool.advanceTime(5) await swapToLowerPrice(encodePriceSqrt(1, 2), wallet.address) await pool.advanceTime(7) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickLower, tickUpper) expect(secondsPerLiquidityInsideX128).to.eq(BigNumber.from(5).shl(128).div(10)) // tick is 0 for 5 seconds, then not in range expect(tickCumulativeInside, 'tickCumulativeInside').to.eq(0) expect(secondsInside).to.eq(5) }) it('time increase below range is not counted', async () => { await swapToLowerPrice(encodePriceSqrt(1, 2), wallet.address) await pool.advanceTime(5) await swapToHigherPrice(encodePriceSqrt(1, 1), wallet.address) await pool.advanceTime(7) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickLower, tickUpper) expect(secondsPerLiquidityInsideX128).to.eq(BigNumber.from(7).shl(128).div(10)) // tick is not in range then tick is 0 for 7 seconds expect(tickCumulativeInside, 'tickCumulativeInside').to.eq(0) expect(secondsInside).to.eq(7) }) it('time increase above range is not counted', async () => { await swapToHigherPrice(encodePriceSqrt(2, 1), wallet.address) await pool.advanceTime(5) await swapToLowerPrice(encodePriceSqrt(1, 1), wallet.address) await pool.advanceTime(7) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickLower, tickUpper) expect(secondsPerLiquidityInsideX128).to.eq(BigNumber.from(7).shl(128).div(10)) expect((await pool.slot0()).tick).to.eq(-1) // justify the -7 tick cumulative inside value expect(tickCumulativeInside, 'tickCumulativeInside').to.eq(-7) expect(secondsInside).to.eq(7) }) it('positions minted after time spent', async () => { await pool.advanceTime(5) await mint(wallet.address, tickUpper, getMaxTick(tickSpacing), 15) await swapToHigherPrice(encodePriceSqrt(2, 1), wallet.address) await pool.advanceTime(8) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickUpper, getMaxTick(tickSpacing)) expect(secondsPerLiquidityInsideX128).to.eq(BigNumber.from(8).shl(128).div(15)) // the tick of 2/1 is 6931 // 8 seconds * 6931 = 55448 expect(tickCumulativeInside, 'tickCumulativeInside').to.eq(55448) expect(secondsInside).to.eq(8) }) it('overlapping liquidity is aggregated', async () => { await mint(wallet.address, tickLower, getMaxTick(tickSpacing), 15) await pool.advanceTime(5) await swapToHigherPrice(encodePriceSqrt(2, 1), wallet.address) await pool.advanceTime(8) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(tickLower, tickUpper) expect(secondsPerLiquidityInsideX128).to.eq(BigNumber.from(5).shl(128).div(25)) expect(tickCumulativeInside, 'tickCumulativeInside').to.eq(0) expect(secondsInside).to.eq(5) }) it('relative behavior of snapshots', async () => { await pool.advanceTime(5) await mint(wallet.address, getMinTick(tickSpacing), tickLower, 15) const { secondsPerLiquidityInsideX128: secondsPerLiquidityInsideX128Start, tickCumulativeInside: tickCumulativeInsideStart, secondsInside: secondsInsideStart, } = await pool.snapshotCumulativesInside(getMinTick(tickSpacing), tickLower) await pool.advanceTime(8) // 13 seconds in starting range, then 3 seconds in newly minted range await swapToLowerPrice(encodePriceSqrt(1, 2), wallet.address) await pool.advanceTime(3) const { secondsPerLiquidityInsideX128, tickCumulativeInside, secondsInside, } = await pool.snapshotCumulativesInside(getMinTick(tickSpacing), tickLower) const expectedDiffSecondsPerLiquidity = BigNumber.from(3).shl(128).div(15) expect(secondsPerLiquidityInsideX128.sub(secondsPerLiquidityInsideX128Start)).to.eq( expectedDiffSecondsPerLiquidity ) expect(secondsPerLiquidityInsideX128).to.not.eq(expectedDiffSecondsPerLiquidity) // the tick is the one corresponding to the price of 1/2, or log base 1.0001 of 0.5 // this is -6932, and 3 seconds have passed, so the cumulative computed from the diff equals 6932 * 3 expect(tickCumulativeInside.sub(tickCumulativeInsideStart), 'tickCumulativeInside').to.eq(-20796) expect(secondsInside - secondsInsideStart).to.eq(3) expect(secondsInside).to.not.eq(3) }) }) describe('fees overflow scenarios', async () => { it('up to max uint 128', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, 1) await flash(0, 0, wallet.address, MaxUint128, MaxUint128) const [feeGrowthGlobal0X128, feeGrowthGlobal1X128] = await Promise.all([ pool.feeGrowthGlobal0X128(), pool.feeGrowthGlobal1X128(), ]) // all 1s in first 128 bits expect(feeGrowthGlobal0X128).to.eq(MaxUint128.shl(128)) expect(feeGrowthGlobal1X128).to.eq(MaxUint128.shl(128)) await pool.burn(minTick, maxTick, 0) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, minTick, maxTick, MaxUint128, MaxUint128 ) expect(amount0).to.eq(MaxUint128) expect(amount1).to.eq(MaxUint128) }) it('overflow max uint 128', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, 1) await flash(0, 0, wallet.address, MaxUint128, MaxUint128) await flash(0, 0, wallet.address, 1, 1) const [feeGrowthGlobal0X128, feeGrowthGlobal1X128] = await Promise.all([ pool.feeGrowthGlobal0X128(), pool.feeGrowthGlobal1X128(), ]) // all 1s in first 128 bits expect(feeGrowthGlobal0X128).to.eq(0) expect(feeGrowthGlobal1X128).to.eq(0) await pool.burn(minTick, maxTick, 0) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, minTick, maxTick, MaxUint128, MaxUint128 ) // fees burned expect(amount0).to.eq(0) expect(amount1).to.eq(0) }) it('overflow max uint 128 after poke burns fees owed to 0', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, 1) await flash(0, 0, wallet.address, MaxUint128, MaxUint128) await pool.burn(minTick, maxTick, 0) await flash(0, 0, wallet.address, 1, 1) await pool.burn(minTick, maxTick, 0) const { amount0, amount1 } = await pool.callStatic.collect( wallet.address, minTick, maxTick, MaxUint128, MaxUint128 ) // fees burned expect(amount0).to.eq(0) expect(amount1).to.eq(0) }) it('two positions at the same snapshot', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, 1) await mint(other.address, minTick, maxTick, 1) await flash(0, 0, wallet.address, MaxUint128, 0) await flash(0, 0, wallet.address, MaxUint128, 0) const feeGrowthGlobal0X128 = await pool.feeGrowthGlobal0X128() expect(feeGrowthGlobal0X128).to.eq(MaxUint128.shl(128)) await flash(0, 0, wallet.address, 2, 0) await pool.burn(minTick, maxTick, 0) await pool.connect(other).burn(minTick, maxTick, 0) let { amount0 } = await pool.callStatic.collect(wallet.address, minTick, maxTick, MaxUint128, MaxUint128) expect(amount0, 'amount0 of wallet').to.eq(0) ;({ amount0 } = await pool .connect(other) .callStatic.collect(other.address, minTick, maxTick, MaxUint128, MaxUint128)) expect(amount0, 'amount0 of other').to.eq(0) }) it('two positions 1 wei of fees apart overflows exactly once', async () => { await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, 1) await flash(0, 0, wallet.address, 1, 0) await mint(other.address, minTick, maxTick, 1) await flash(0, 0, wallet.address, MaxUint128, 0) await flash(0, 0, wallet.address, MaxUint128, 0) const feeGrowthGlobal0X128 = await pool.feeGrowthGlobal0X128() expect(feeGrowthGlobal0X128).to.eq(0) await flash(0, 0, wallet.address, 2, 0) await pool.burn(minTick, maxTick, 0) await pool.connect(other).burn(minTick, maxTick, 0) let { amount0 } = await pool.callStatic.collect(wallet.address, minTick, maxTick, MaxUint128, MaxUint128) expect(amount0, 'amount0 of wallet').to.eq(1) ;({ amount0 } = await pool .connect(other) .callStatic.collect(other.address, minTick, maxTick, MaxUint128, MaxUint128)) expect(amount0, 'amount0 of other').to.eq(0) }) }) describe('swap underpayment tests', () => { let underpay: TestUniswapV3SwapPay beforeEach('deploy swap test', async () => { const underpayFactory = await ethers.getContractFactory('TestUniswapV3SwapPay') underpay = (await underpayFactory.deploy()) as TestUniswapV3SwapPay await token0.approve(underpay.address, constants.MaxUint256) await token1.approve(underpay.address, constants.MaxUint256) await pool.initialize(encodePriceSqrt(1, 1)) await mint(wallet.address, minTick, maxTick, expandTo18Decimals(1)) }) it('underpay zero for one and exact in', async () => { await expect( underpay.swap(pool.address, wallet.address, true, MIN_SQRT_RATIO.add(1), 1000, 1, 0) ).to.be.revertedWith('IIA') }) it('pay in the wrong token zero for one and exact in', async () => { await expect( underpay.swap(pool.address, wallet.address, true, MIN_SQRT_RATIO.add(1), 1000, 0, 2000) ).to.be.revertedWith('IIA') }) it('overpay zero for one and exact in', async () => { await expect( underpay.swap(pool.address, wallet.address, true, MIN_SQRT_RATIO.add(1), 1000, 2000, 0) ).to.not.be.revertedWith('IIA') }) it('underpay zero for one and exact out', async () => { await expect( underpay.swap(pool.address, wallet.address, true, MIN_SQRT_RATIO.add(1), -1000, 1, 0) ).to.be.revertedWith('IIA') }) it('pay in the wrong token zero for one and exact out', async () => { await expect( underpay.swap(pool.address, wallet.address, true, MIN_SQRT_RATIO.add(1), -1000, 0, 2000) ).to.be.revertedWith('IIA') }) it('overpay zero for one and exact out', async () => { await expect( underpay.swap(pool.address, wallet.address, true, MIN_SQRT_RATIO.add(1), -1000, 2000, 0) ).to.not.be.revertedWith('IIA') }) it('underpay one for zero and exact in', async () => { await expect( underpay.swap(pool.address, wallet.address, false, MAX_SQRT_RATIO.sub(1), 1000, 0, 1) ).to.be.revertedWith('IIA') }) it('pay in the wrong token one for zero and exact in', async () => { await expect( underpay.swap(pool.address, wallet.address, false, MAX_SQRT_RATIO.sub(1), 1000, 2000, 0) ).to.be.revertedWith('IIA') }) it('overpay one for zero and exact in', async () => { await expect( underpay.swap(pool.address, wallet.address, false, MAX_SQRT_RATIO.sub(1), 1000, 0, 2000) ).to.not.be.revertedWith('IIA') }) it('underpay one for zero and exact out', async () => { await expect( underpay.swap(pool.address, wallet.address, false, MAX_SQRT_RATIO.sub(1), -1000, 0, 1) ).to.be.revertedWith('IIA') }) it('pay in the wrong token one for zero and exact out', async () => { await expect( underpay.swap(pool.address, wallet.address, false, MAX_SQRT_RATIO.sub(1), -1000, 2000, 0) ).to.be.revertedWith('IIA') }) it('overpay one for zero and exact out', async () => { await expect( underpay.swap(pool.address, wallet.address, false, MAX_SQRT_RATIO.sub(1), -1000, 0, 2000) ).to.not.be.revertedWith('IIA') }) }) })