Skip to content

Latest commit

 

History

History
 
 

protocol

Beanstalk logo

Beanstalk Protocol

Discord

Code Version: 2.3.1
Whitepaper Version: 2.3.0

About

This repository contains the code base for the Beanstalk protocol, all of its facets and related contracts in the Beanstalk ecosystem.

EIP-2535 Diamond

The Beanstalk contract is a multi-facet proxy that implements EIP-2535. Thus, the Beanstalk contract implements functionality from multiple different Facet contracts that all share a common storage.

Getting Started

Prerequistes

  • git
    • You'll know you did it right if you can run git --version and you see a response like git version x.x.x
  • Nodejs
    • You'll know you've installed nodejs right if you can run node --versionand get an ouput like: vx.x.x
  • Yarn
    • You'll know you've installed yarn right if you can run yarn --version And get an output like: x.x.x
  • Rosetta (Apple Silicon)
    • You'll know you've installed Rosetta right if you can execute the solc 0.7.6 binary .../solc-macosx-amd64-v0.7.6+commit.7338295f --version

Node and NPM compatibility

The latest Node.js and NPM versions are known to be incompatible with some parts of the Foundry toolchain. We recommend using Node.js v18.12.0 and npm v8.19.2.

Installation

Note: The Beanstalk repo is a monorepo with induvidual projects inside it. See the DEVELOPER.md in the root directory for more information.

  1. Clone the repository and install dependencies
git clone https://github.com/BeanstalkFarms/Beanstalk
cd Beanstalk/protocol
yarn
  1. Compile the contracts
yarn hardhat compile

Quickstart

Testing

Hardhat

  1. Ensure you are in the protocol directory
  2. Have a FORKING_RPC as an environment variable pointing to ETH Mainnet.
    1. We fork ETH Mainnet to run our test suite locally. Note, this will make a large number of API calls to your FORKING_RPC
  3. Run all tests
yarn test
  1. Generate coverage report
yarn hardhat coverage

Foundry

  1. Ensure you are in the protocol directory
  2. Install foundry with foundryup - instructions here
  3. forge install foundry-rs/forge-std --no-commit
  4. forge test

Forking Mainnet Locally

We elect to use anvil instead of hardhat for local node forking as anvil is considerably faster than hardhat and properly caches the blockchain locally.

  1. Ensure you are in the /protocol repository
  2. Install Foundry
# Be sure to use the official docs linked above if this doesn't work
curl -L https://foundry.paradigm.xyz | bash
  1. Close and re-open your terminal to ensure foundry is installed correctly
    1. You may need to update your path.
  2. Run foundryup to ensure you have downloaded the latest version
  3. Start a locally forked node with the following command:
anvil --fork-url <FORKING_RPC> --fork-block-number <BLOCK_NUMBER> --chain-id 1337

For <FORKING_RPC>, use an Alchemy or Infura RPC URL. It should be very clear if the node starts up properly.

Note: anvil will cache the blockchain provided that BLOCK_NUMBER does NOT change. Given this, we recommend picking a block and sticking to it.

Developing

Overview

As Beanstalk implements EIP-2535, Beanstalk is upgraded through a diamondCut function call.

There are two different ways a diamondCut can execute code:

  1. Adding, replacing and/or removing functions
    • Functions in Beanstalk are implemented in contracts known as facets. Facets are no different than normal smart contract with callable functions. In order to share a state, Facets can only define 1 internal state variable: The AppStorage struct defined in AppStorage.sol. Read more here.
  2. Calling the init function of a contract
    • This is a one time action and will be called when the diamondCut is executed. There can be 1 init call per diamondCut.

Creating a New Facet

For this tutorial, we will create a new Facet called SampleFacet.

  1. Ensure you are in the protocol directory
  2. In protocol/farm/facets/, create a new folder called SampleFacet
  3. Within the SampleFacet folder create a file called SampleFacet.sol.
  4. Implement your Facet. You can use SampleFacet.sol in protocol/samples as a template. Note that Facets can only have AppStorage as an internal state variable.
  5. Modify the deploy function in scripts/deploy to include your new Facet, so that the Facet will be deployed with the Beanstalk Diamond.

Testing a BIP

Overview

There are a couple of steps that must be done before forking mainnet and testing a BIP.

  1. Include the following code in the networks section of the hardhat.config.js, where RPC_URL is your RPC url. We recommend using Alchemy for this. The BLOCK_NUMBER is optional, but we recommend choosing a block number close to the current block.

    forking: {
        url: <RPC_URL>,
        blockNumber: <BLOCK_NUMBER>
      },
    localhost: {
      chainId: 1337,
      url: "http://127.0.0.1:8545",
      forking: {
        url: <RPC_URL>,
        blockNumber: <BLOCK_NUMBER>
      },
    },
  2. Include as imports:

    const BEANSTALK = "0xC1E088fC1323b20BCBee9bd1B9fC9546db5624C5";
    const ownerFacet = await ethers.getContractAt("OwnershipFacet", BEANSTALK);
    const owner = await ownerFacet.owner();
    const { upgradeWithNewFacets } = require("./scripts/diamond.js");
  3. Lastly, include the tasks required for upgrading above module.exports:

    task("upgrade", "Commits a bip", async () => {
      await hre.network.provider.request({
        method: "hardhat_impersonateAccount",
        params: [owner]
      });
      const account = await ethers.getSigner(owner);
      await upgradeWithNewFacets({
        diamondAddress: BEANSTALK,
        facetNames: [],
        initFacetName: "InitEmpty",
        initArgs: [],
        bip: false,
        verbose: true,
        account: account
      });
    });
  4. Here is an example of what BIP-11 deployment looked like:

    await upgradeWithNewFacets({
      diamondAddress: BEANSTALK,
      initFacetName: "InitBip11",
      facetNames: ["MarketplaceFacet"],
      libraryNames: ["LibClaim"],
      facetLibraries: {
        MarketplaceFacet: ["LibClaim"]
      },
      bip: false,
      verbose: true,
      account: account
    });

Running the Upgrade Tasks

  1. Spin up your mainnet fork node with:

    yarn hardhat node
  2. In another console, execute your tasks against your mainnet fork by running:

    yarn hardhat upgrade --network localhost

    Where upgrade is where you put the name of your task (in the example above it was named upgrade).

  3. Now you can test your changes using your local mainnet fork that should now have the latest version of Beanstalk that you upgraded.

License

MIT