diff --git a/apps/dashboard_app/tests/protocol_stats_test.py b/apps/dashboard_app/tests/protocol_stats_test.py index 3c20127f..bda9a358 100644 --- a/apps/dashboard_app/tests/protocol_stats_test.py +++ b/apps/dashboard_app/tests/protocol_stats_test.py @@ -1,10 +1,13 @@ -import pandas as pd -import pytest +""" +Tests for the protocol_stats module. +""" + from decimal import Decimal from unittest.mock import MagicMock, patch +import pandas as pd +import pytest -from shared.state import State -from shared.types import Prices +# from shared.types import Prices from dashboard_app.helpers.protocol_stats import ( get_general_stats, get_supply_stats, @@ -12,10 +15,14 @@ get_debt_stats, get_utilization_stats, ) +from shared.state import State @pytest.fixture def token_addresses(): + """ + Returns a dictionary of token addresses. + """ return { "ETH": "0x049d36570d4e46f48e99674bd3fcc84644ddd6b96f7c741b1562b82f9e004dc7", "WBTC": "0x03fe2b97c1fd336e750087d68b9b867997fd64a2661ff3ca5a7c771641e8e7ac", @@ -31,6 +38,9 @@ def token_addresses(): @pytest.fixture def mock_loan_stats(): + """ + Returns a dictionary of loan stats. + """ return { "zkLend": pd.DataFrame({ "Debt (USD)": [1000], @@ -42,12 +52,15 @@ def mock_loan_stats(): @pytest.fixture def mock_state(token_addresses): + """ + Returns a mock state object. + """ state = MagicMock(spec=State) state.get_protocol_name.return_value = "zkLend" state.PROTOCOL_NAME = "zkLend" state.compute_number_of_active_loan_entities.return_value = 10 state.compute_number_of_active_loan_entities_with_debt.return_value = 5 - + # Setup loan entity with default values loan_entity = MagicMock() loan_entity.collateral.values = {addr: "0" for addr in token_addresses.values()} @@ -55,36 +68,45 @@ def mock_state(token_addresses): loan_entity.collateral.values[token_addresses["ETH"]] = "1000000000000000000" loan_entity.debt[token_addresses["ETH"]] = "2000000000000000000" state.loan_entities = {"user1": loan_entity} - + # Setup token parameters class TokenParam: + """ + A class to represent token parameters. + """ def __init__(self, address, symbol): self.address = address self.underlying_symbol = symbol self.underlying_address = address self.decimal_factor = 1e18 - + token_params = { symbol: TokenParam(addr, symbol) for symbol, addr in token_addresses.items() } - + state.token_parameters = MagicMock() state.token_parameters.collateral = token_params state.token_parameters.debt = token_params.copy() - + # Setup interest rate models interest_rates = {addr: "1.0" for addr in token_addresses.values()} state.interest_rate_models = MagicMock() state.interest_rate_models.collateral = interest_rates state.interest_rate_models.debt = interest_rates.copy() - - with patch("dashboard_app.helpers.protocol_stats.get_protocol", return_value="zkLend"): + + with patch( + "dashboard_app.helpers.protocol_stats.get_protocol", + return_value="zkLend" + ): yield state @pytest.fixture def mock_prices(token_addresses): + """ + Returns a dictionary of prices. + """ return { token_addresses["ETH"]: "2000", token_addresses["WBTC"]: "30000", @@ -99,6 +121,9 @@ def mock_prices(token_addresses): @pytest.fixture def mock_token_settings(token_addresses): + """ + Returns a dictionary of token settings. + """ return { symbol: MagicMock( decimal_factor=Decimal("1e18"), @@ -109,13 +134,22 @@ def mock_token_settings(token_addresses): @pytest.fixture(autouse=True) def patch_token_settings(mock_token_settings): - with patch("dashboard_app.helpers.protocol_stats.TOKEN_SETTINGS", mock_token_settings): + """ + Patches the TOKEN_SETTINGS dictionary. + """ + with patch( + "dashboard_app.helpers.protocol_stats.TOKEN_SETTINGS", + mock_token_settings + ): yield def test_get_general_stats(mock_state, mock_loan_stats): + """ + Tests the get_general_stats function. + """ result = get_general_stats([mock_state], mock_loan_stats) - + assert isinstance(result, pd.DataFrame) assert "Protocol" in result.columns assert result["Number of active users"].iloc[0] == 10 @@ -124,12 +158,18 @@ def test_get_general_stats(mock_state, mock_loan_stats): def test_get_general_stats_empty_state(): + """ + Tests the get_general_stats function with an empty state. + """ result = get_general_stats([], {}) assert isinstance(result, pd.DataFrame) assert len(result) == 0 def test_get_general_stats_invalid_loan_stats(mock_state): + """ + Tests the get_general_stats function with an invalid loan stats. + """ with pytest.raises(KeyError): get_general_stats([mock_state], {"InvalidProtocol": pd.DataFrame()}) @@ -146,15 +186,18 @@ def test_get_supply_stats( token_addresses, mock_token_settings ): + """ + Tests the get_supply_stats function. + """ mock_get_protocol.return_value = "zkLend" mock_get_params.return_value = ([token_addresses["ETH"]], "felt_total_supply") mock_run.return_value = [Decimal("1000000000000000000")] - + # Convert prices to Decimal prices = {k: Decimal(v) for k, v in mock_prices.items()} - + result = get_supply_stats([mock_state], prices) - + assert isinstance(result, pd.DataFrame) assert "Protocol" in result.columns assert "ETH supply" in result.columns @@ -164,34 +207,45 @@ def test_get_supply_stats( @patch("dashboard_app.helpers.protocol_stats.get_supply_function_call_parameters") @patch("dashboard_app.helpers.protocol_stats.asyncio.run") def test_get_supply_stats_blockchain_error( - mock_run, - mock_get_params, + mock_run, + mock_get_params, mock_get_protocol, mock_state, mock_prices, token_addresses ): + """ + Tests the get_supply_stats function with a blockchain error. + """ mock_get_protocol.return_value = "zkLend" mock_get_params.return_value = ([token_addresses["ETH"]], "felt_total_supply") mock_run.side_effect = Exception("Blockchain call failed") - + with pytest.raises(Exception): get_supply_stats([mock_state], mock_prices) @patch("dashboard_app.helpers.protocol_stats.get_protocol") def test_get_collateral_stats(mock_get_protocol, mock_state, token_addresses): + """ + Tests the get_collateral_stats function. + """ mock_get_protocol.return_value = "zkLend" - + result = get_collateral_stats([mock_state]) - + assert isinstance(result, pd.DataFrame) assert "Protocol" in result.columns assert "ETH collateral" in result.columns def test_get_collateral_stats_invalid_protocol(mock_state): - with patch("dashboard_app.helpers.protocol_stats.get_protocol") as mock_get_protocol: + """ + Tests the get_collateral_stats function with an invalid protocol. + """ + with patch( + "dashboard_app.helpers.protocol_stats.get_protocol" + ) as mock_get_protocol: mock_get_protocol.return_value = "InvalidProtocol" with pytest.raises(ValueError): get_collateral_stats([mock_state]) @@ -199,46 +253,65 @@ def test_get_collateral_stats_invalid_protocol(mock_state): @patch("dashboard_app.helpers.protocol_stats.get_protocol") def test_get_debt_stats(mock_get_protocol, mock_state, token_addresses): + """ + Tests the get_debt_stats function. + """ mock_get_protocol.return_value = "zkLend" - + result = get_debt_stats([mock_state]) - + assert isinstance(result, pd.DataFrame) assert "Protocol" in result.columns assert "ETH debt" in result.columns def test_get_debt_stats_invalid_protocol(mock_state): - with patch("dashboard_app.helpers.protocol_stats.get_protocol") as mock_get_protocol: + """ + Tests the get_debt_stats function with an invalid protocol. + """ + with patch( + "dashboard_app.helpers.protocol_stats.get_protocol" + ) as mock_get_protocol: mock_get_protocol.return_value = "InvalidProtocol" with pytest.raises(ValueError): get_debt_stats([mock_state]) def test_get_utilization_stats(): + """ + Tests the get_utilization_stats function. + """ general_stats = pd.DataFrame({ "Protocol": ["zkLend"], "Total debt (USD)": [1000], }) - + supply_stats = pd.DataFrame({ "Protocol": ["zkLend"], "Total supply (USD)": [4000], - **{f"{token} supply": [1000 if token in ["USDC", "DAI", "USDT", "LORDS", "STRK"] else 1] - for token in ["ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK"]} + **{f"{token} supply": [ + 1000 if token in ["USDC", "DAI", "USDT", "LORDS", "STRK"] else 1 + ] for token in [ + "ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK" + ]} }) - + debt_stats = pd.DataFrame({ "Protocol": ["zkLend"], - **{f"{token} debt": [500 if token in ["USDC", "DAI", "USDT", "LORDS", "STRK"] else 0.5] - for token in ["ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK"]} + **{f"{token} debt": [ + 500 if token in ["USDC", "DAI", "USDT", "LORDS", "STRK"] else 0.5 + ] for token in [ + "ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK" + ]} }) - + result = get_utilization_stats(general_stats, supply_stats, debt_stats) - + utilization_columns = [col for col in result.columns if col != "Protocol"] - result[utilization_columns] = result[utilization_columns].applymap(lambda x: round(x, 4)) - + result[utilization_columns] = result[utilization_columns].applymap( + lambda x: round(x, 4) + ) + assert isinstance(result, pd.DataFrame) assert "Protocol" in result.columns assert "Total utilization" in result.columns @@ -246,25 +319,31 @@ def test_get_utilization_stats(): def test_get_utilization_stats_division_by_zero(): + """ + Tests the get_utilization_stats function with a division by zero. + """ general_stats = pd.DataFrame({ "Protocol": ["zkLend"], "Total debt (USD)": [1000], }) - + supply_stats = pd.DataFrame({ "Protocol": ["zkLend"], "Total supply (USD)": [0], - **{f"{token} supply": [0] for token in ["ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK"]} + **{f"{token} supply": [0] for token in [ + "ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK" + ]} }) - + debt_stats = pd.DataFrame({ "Protocol": ["zkLend"], - **{f"{token} debt": [0] for token in ["ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK"]} + **{f"{token} debt": [0] for token in [ + "ETH", "WBTC", "USDC", "DAI", "USDT", "wstETH", "LORDS", "STRK" + ]} }) - + result = get_utilization_stats(general_stats, supply_stats, debt_stats) - + # Check if division by zero results in NaN or infinity - assert result["Total utilization"].iloc[0] == 1 # When supply is 0, utilization should be 100% - assert pd.isna(result["ETH utilization"].iloc[0]) # When both supply and debt are 0, result should be NaN - \ No newline at end of file + assert result["Total utilization"].iloc[0] == 1 + assert pd.isna(result["ETH utilization"].iloc[0])