Problem

If Hyperdrive simply adjusts bond liquidity proportional to the liquidity being added we run into issues with price discovery and sandwich attacks. If someone adds liquidity at an irrationally high interest rate, it inflates the bond reserves so much that the rate can't be arbed back. Additionally, this creates an opportunity for a long-range sandwich attack. To mitigate this, we implement the following initialize routine.

Zeta and Bond Reserves for Initialization

We want to solve for y and ζ using the following systems of equations:(1)czcζ+py=cz(2)p=(μzμζy)tswhere cz is the amount of base the pool is initialized with. Combining (1) and (2) we have:(3)ζ=z1μp1tsyCombining (1) and (3) gives us:czc(z1μp1tsy)+py=czthis simplifies to: (4)y=czcμp1ts+p

Visualization of The Fix

Test Code

    function test_example() external {
        IHyperdrive.PoolConfig memory config = testConfig(
            0.05e18,
            POSITION_DURATION
        );
        config.fees.curve = 0.001e18;
        config.fees.flat = 0.0005e18;
        deploy(alice, config);
        uint256 apr = 0.05e18;
        uint256 contribution = 100_000_000e18;
        uint256 lpShares = initialize(alice, apr, contribution);

        // Celine opens a large short.
        uint256 shortAmount = hyperdrive.calculateMaxShort();
        (, uint256 shortPaid) = openShort(celine, shortAmount);
        console.log(
            "spot rate = %s",
            hyperdrive.calculateSpotAPR().toString(18)
        );

        // Celine adds liquidity.
        uint256 celineLpShares = addLiquidity(celine, 100_000_000e18);

        // Celine opens a large long.
        uint256 basePaid = hyperdrive.calculateMaxLong();
        (uint256 maturityTime, uint256 bondAmount) = openLong(
            celine,
            hyperdrive.calculateMaxLong()
        );

        // The term advances. Alice removes her liquidity.
        advanceTime(POSITION_DURATION, 0.05e18);

        // Remove liquidity.
        (uint256 baseProceeds, uint256 withdrawalShares) = removeLiquidity(
            alice,
            lpShares
        );
        console.log("alice contribution     = %s", contribution.toString(18));
        console.log("alice baseProceeds     = %s", baseProceeds.toString(18));
        console.log(
            "alice withdrawalShares = %s",
            withdrawalShares.toString(18)
        );

        // Celine closes her long and removes liquidity.
        uint256 shortProceeds = closeShort(celine, maturityTime, shortAmount);
        uint256 longProceeds = closeLong(celine, maturityTime, bondAmount);
        (baseProceeds, withdrawalShares) = removeLiquidity(
            celine,
            celineLpShares
        );
        console.log(
            "celine paid           = %s",
            (100_000_000e18 + basePaid + shortPaid).toString(18)
        );
        console.log(
            "celine proceeds       = %s",
            (baseProceeds + longProceeds + shortProceeds).toString(18)
        );
    }