Skip to content

Decentralized allocation protocol for DeFi lending pools, enabling miner-driven asset distribution for optimal yield generation.

License

Notifications You must be signed in to change notification settings

Sturdy-Subnet/sturdy-subnet

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Sturdy Subnet

License: MIT


Decentralized Yield Farming Fund


Introduction

The Sturdy Subnet is a Bittensor subnetwork that enables the creation of decentralized, autonomous yield optimizers. A yield optimizer is a smart contract that seeks to provide users with the best possible yields by depositing assets to a variety of strategies. On the Sturdy Subnet, every yield optimizer has a fixed set of strategies (or 'pools') that it can deposit to. In turn, each pool has its own interest rate curve, described in more detail below. The goal for each miner is to create an algorithm that computes the allocation of assets among pools that results in the highest yield possible. Validators then evaluate miners based on how much yield their allocation produces.

The outputs of the subnet will be used by third-party applications to move real assets on the Ethereum network. The first application using the Sturdy Subnet is the Sturdy protocol, with more to come.

Codebase

There are three core files.

  1. sturdy/protocol.py: Contains the definition of the protocol used by subnet miners and subnet validators. At the moment it only has one kind of synapse - AllocateAssets - which contains the inputs (assets_and_pools) validators need to send to miners to generate return allocations for. See generate_challenge_data() in pools.py to see how assets and pools are defined.
  2. neurons/miner.py: Script that defines the subnet miner's behavior, i.e., how the subnet miner responds to requests from subnet validators.
  3. neurons/validator.py: This script defines the subnet validator's behavior, i.e., how the subnet validator requests information from the subnet miners and determines the scores.

Subnet Overview

  • Validators are responsible for distributing lists of pools (of which contain relevant parameters such as base interest rate, base interest rate slope, minimum borrow amount, etc), as well as a maximum token balance miners can allocate to pools. Below are the function present in the codebase used for generating challenge data in pools.py used for synthetic requests. The selection of different assets and pools which can be used in such requests are defined in the pool registry, and are all based on pools which are real and do indeed exist on-chain (i.e. on the Ethereum Mainnet):

    def generate_challenge_data(
        web3_provider: Web3,
        rng_gen: np.random.RandomState = np.random.RandomState(),  # noqa: B008
    ) -> dict[str, dict[str, ChainBasedPoolModel] | int]:  # generate pools
        selected_entry = POOL_REGISTRY[rng_gen.choice(list(POOL_REGISTRY.keys()))]
        bt.logging.debug(f"Selected pool registry entry: {selected_entry}")
    
        return assets_pools_for_challenge_data(selected_entry, web3_provider)
    
    
    def assets_pools_for_challenge_data(
        selected_entry, web3_provider: Web3
    ) -> dict[str, dict[str, ChainBasedPoolModel] | int]:  # generate pools
        challenge_data = {}
    
        selected_assets_and_pools = selected_entry["assets_and_pools"]
        selected_pools = selected_assets_and_pools["pools"]
        global_user_address = selected_entry.get("user_address", None)
    
        pool_list = []
    
        for pool_dict in selected_pools.values():
            user_address = pool_dict.get("user_address", None)
            pool = PoolFactory.create_pool(
                pool_type=POOL_TYPES._member_map_[pool_dict["pool_type"]],
                user_address=global_user_address if user_address is None else user_address,
                contract_address=pool_dict["contract_address"],
            )
            pool_list.append(pool)
    
        pools = {str(pool.contract_address): pool for pool in pool_list}
    
        # we assume that the user address is the same across pools (valid)
        # and also that the asset contracts are the same across said pools
        total_assets = selected_entry.get("total_assets", None)
    
        if total_assets is None:
            total_assets = 0
            first_pool = pool_list[0]
            first_pool.sync(web3_provider)
            match first_pool.pool_type:
                case T if T in (
                    POOL_TYPES.STURDY_SILO,
                    POOL_TYPES.AAVE_DEFAULT,
                    POOL_TYPES.AAVE_TARGET,
                    POOL_TYPES.MORPHO,
                    POOL_TYPES.YEARN_V3,
                ):
                    total_assets = first_pool._user_asset_balance
                case _:
                    pass
    
            for pool in pools.values():
                pool.sync(web3_provider)
                total_asset = 0
                match pool.pool_type:
                    case T if T in (
                        POOL_TYPES.STURDY_SILO,
                        POOL_TYPES.AAVE_DEFAULT,
                        POOL_TYPES.AAVE_TARGET,
                        POOL_TYPES.MORPHO,
                        POOL_TYPES.YEARN_V3,
                    ):
                        total_asset += pool._user_deposits
                    case _:
                        pass
    
                total_assets += total_asset
    
        challenge_data["assets_and_pools"] = {}
        challenge_data["assets_and_pools"]["pools"] = pools
        challenge_data["assets_and_pools"]["total_assets"] = total_assets
        if global_user_address is not None:
            challenge_data["user_address"] = global_user_address
    
        return challenge_data

    Validators can optionally run an API server and sell their bandwidth to outside users to send their own pools (organic requests) to the subnet. For more information on this process - please read docs/validator.md

  • NOTE: Validators use large numbers (by following the ERC20 decimal convention) for handling some pool parameters and miner allocations.

  • The miners, after receiving these pools from validators, must then attempt to allocate the TOTAL_ASSETS into the given pools, with the ultimate goal of trying to maximize their yield. This repository comes with a default asset allocation algorithm in the form of naive_algorithm (a naive allocation algorithm) in algo.py. The naive allocation essentially works by divvying assets across pools, and allocating more to pools which have a higher current supply rate.

  • After generating allocations, miners then send their outputs to validators to be scored. These requests are generated and sent to miners roughly every 15 minutes. Organic requests, on the other hand, are sent by to validators, upon which they are then routed to miners. After the "scoring period" for requests have passed, miners are then scored based on how much yield pools have generated within the scoring period - with the miner with the most yield obtaining the highest score. Scoring these miners involves gather on chain info about pools, with most if not all such information being obtained from smart contracts on the the Ethereum Network. Miners which have similar allocations to other miners will be penalized if they are not perceived as being original. If miners fail to respond in ~45 seconds after receiving the request they are scored poorly. The best allocating miner will receive the most emissions. For more information on how miners are rewarded - please see forward.py, reward.py, and validator.py. A diagram is provided below highlighting the interactions that takes place within the subnet when processing synthetic and organic requests:


Installation

Before you proceed

Before you proceed with the installation, note the following:

  • Python version 3.10.x is required to run code in this repo. We highly recommend that you use some thing like conda to create virtual environments with its own python 3.10.x interpreter. For more information on how to do this, please refer to conda's documentation regarding installation and environment creation.
  • IMPORTANT: Make sure you are aware of the minimum compute requirements for your subnet. See the Minimum compute YAML configuration.
  • Note that installation instructions differ based on your situation: For example, installing for local development and testing will require a few additional steps compared to installing for testnet or mainnet. For running a local subtensor - please visit: https://github.com/opentensor/subtensor.
  • We also urge miners to set up a firewall as shown here to mitigate naive DDOS attacks.

Install

git clone https://github.com/Sturdy-subnet/sturdy-subnet/
cd sturdy-subnet
python -m pip install -e .

Setup WandB (HIGHLY RECOMMENDED - VALIDATORS PLEASE READ)

Before running your miner and validator, you may also choose to set up Weights & Biases (WANDB). It is a popular tool for tracking and visualizing machine learning experiments, and we use it for logging and tracking key metrics across miners and validators, all of which is available publicly here. We highly recommend validators use wandb, as it allows subnet developers and miners to diagnose issues more quickly and effectively, say, in the event a validator were to be set abnormal weights. Wandb logs are collected by default, and done so in an anonymous fashion, but we recommend setting up an account to make it easier to differentiate between validators when searching for runs on our dashboard. If you would not like to run WandB, you can do so by adding the flag --wandb.off when running your miner/validator.

Before getting started, as mentioned previously, you'll first need to register for a WANDB account, and then set your API key on your system. Here's a step-by-step guide on how to do this on Ubuntu:

Step 1: Installation of WANDB

Before logging in, make sure you have the WANDB Python package installed. If you haven't installed it yet, you can do so using pip:

# Should already be installed with the sturdy repo
pip install wandb

Step 2: Obtain Your API Key

  1. Log in to your Weights & Biases account through your web browser.
  2. Go to your account settings, usually accessible from the top right corner under your profile.
  3. Find the section labeled "API keys".
  4. Copy your API key. It's a long string of characters unique to your account.

Step 3: Setting Up the API Key in Ubuntu

To configure your WANDB API key on your Ubuntu machine, follow these steps:

  1. Log into WANDB: Run the following command in the terminal:

    wandb login
  2. Enter Your API Key: When prompted, paste the API key you copied from your WANDB account settings.

    • After pasting your API key, press Enter.
    • WANDB should display a message confirming that you are logged in.
  3. Verifying the Login: To verify that the API key was set correctly, you can start a small test script in Python that uses WANDB. If everything is set up correctly, the script should run without any authentication errors.

  4. Setting API Key Environment Variable (Optional): If you prefer not to log in every time, you can set your API key as an environment variable in your ~/.bashrc or ~/.bash_profile file:

    echo 'export WANDB_API_KEY=your_api_key' >> ~/.bashrc
    source ~/.bashrc

    Replace your_api_key with the actual API key. This method automatically authenticates you with wandb every time you open a new terminal session.


Running

Acknowledgement for Vision Subnet!

We extend our heartfelt appreciation to namoray et al. for their exceptional work on the Vision subnet. Our API, which enables third-party applications to integrate the subnet, draws significant inspiration from their work.

License

This repository is licensed under the MIT License.

# The MIT License (MIT)
# Copyright © 2024 Syeam Bin Abdullah

# Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
# documentation files (the “Software”), to deal in the Software without restriction, including without limitation
# the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software,
# and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

# The above copyright notice and this permission notice shall be included in all copies or substantial portions of
# the Software.

# THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO
# THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
# DEALINGS IN THE SOFTWARE.

About

Decentralized allocation protocol for DeFi lending pools, enabling miner-driven asset distribution for optimal yield generation.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •