A Python client for fetching data from the Furthermore API.
This client provides methods to interact with the Furthermore API endpoints for retrieving vault data (referred to as 'articles'), BGT prices, and extracting data source names (protocols and incentivizers).
- Fetch a list of articles (vaults) with pagination and sorting options.
- Fetch current BGT (Berachain Governance Token) derivative prices.
- Extract unique protocol and incentivizer names from vault data.
- Configurable base URL and logger.
- Python >=3.13 (as per
pyproject.toml
) requests
library (will be installed viapyproject.toml
)- A Furthermore API key
-
Clone the repository (if you haven't already):
git clone https://github.com/itsdevbear/furthermore-py cd furthermore-py
-
Create and activate a virtual environment using
uv
:uv venv source .venv/bin/activate # On macOS/Linux # .venv\Scripts\activate # On Windows
-
Install dependencies and the project for development: To install dependencies and make your project importable (recommended for development, especially for running examples):
uv pip install -e .
This installs the project in "editable" mode. If you only want to install dependencies as defined (e.g., for a non-development setup or CI), you can use:
uv sync # or uv pip install .
-
Set the API Key: The client requires the
FURTHERMORE_API_KEY
environment variable to be set.export FURTHERMORE_API_KEY="your_actual_api_key_here"
Alternatively, you can pass the
api_key
directly when initializing theFurthermoreClient
.
Here's a basic example of how to use the FurthermoreClient
:
import os
import logging
from furthermore_py import FurthermoreClient # Adjust import based on your project structure
if __name__ == '__main__':
# Ensure API key is set
api_key = os.getenv(FurthermoreClient.FURTHERMORE_API_KEY_ENV_VAR)
if not api_key:
print(f"CRITICAL: The environment variable '{FurthermoreClient.FURTHERMORE_API_KEY_ENV_VAR}' is not set.")
print(f"Please set it before running this example (e.g., export {FurthermoreClient.FURTHERMORE_API_KEY_ENV_VAR}=your_key).")
exit()
# Optional: Configure logging for the example
example_logger = logging.getLogger("FurthermoreClientExample")
example_logger.setLevel(logging.INFO)
if not example_logger.hasHandlers():
ch = logging.StreamHandler()
formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
ch.setFormatter(formatter)
example_logger.addHandler(ch)
example_logger.propagate = False # Avoid duplicate logs if root logger is configured
client = FurthermoreClient(logger=example_logger)
try:
print("\nFetching articles (first 3 vaults):")
articles = client.get_articles(limit=3)
print(f"Total articles (vaults) reported by API: {articles.get('count', 'N/A')}")
if articles.get("vaults"):
for i, article in enumerate(articles["vaults"]):
print(f" Article {i+1} ID: {article.get('id')}, Protocol: {article.get('metadata', {}).get('protocolName', 'N/A')}")
else:
print("No articles found or 'vaults' key missing in response.")
print("\nFetching BGT prices:")
bgt_prices = client.get_bgt_prices()
if bgt_prices.get("data"):
print(f"Average BGT price: {bgt_prices.get('average', 'N/A')}")
for price_info in bgt_prices["data"][:3]:
print(f" Token: {price_info.get('token', 'N/A')}, Price: {price_info.get('price', 'N/A')}")
else:
print("No BGT prices found or 'data' key missing in response.")
print("\nFetching sources:")
sources = client.get_sources(vault_limit_for_scan=20)
print(f"Found {len(sources['protocols'])} unique protocols: {sources['protocols'] or 'None'}")
print(f"Found {len(sources['incentivizers'])} unique incentivizers: {sources['incentivizers'] or 'None'}")
except ValueError as ve:
print(f"Initialization error: {ve}")
except requests.exceptions.RequestException as re:
print(f"API request error: {re}")
except Exception as e:
print(f"An unexpected error occurred: {e}")
### How to Run the Example Script
To run the example script (e.g., `src/examples/basic.py`):
1. Ensure your virtual environment is activated (`source .venv/bin/activate`).
2. Make sure the package is installed in editable mode from the project root directory: `uv pip install -e .`
3. Run the script from the project root directory:
```bash
python src/examples/basic.py
```
## API Documentation
For more details on the API endpoints and responses, refer to the unofficial Gist:
[https://gist.github.com/asianviking/b87cad7b9e0b0519f1ae8bdb8121398b](https://gist.github.com/asianviking/b87cad7b9e0b0519f1ae8bdb8121398b)
## Development
This section outlines tools and practices for developing `furthermore-py`.
### Linting and Formatting
This project uses [Ruff](https://github.com/astral-sh/ruff) for linting and code formatting to ensure code quality and consistency.
**Installation:**
Ruff is included as a development dependency in `pyproject.toml`. If you followed the main installation steps using `uv pip install .` or `uv sync`, and it included development dependencies (e.g., via an extras group like `dev` if configured, or by default if not specified otherwise), Ruff should already be installed.
To ensure you have all development dependencies, you can explicitly install them. If your `pyproject.toml` defines a `dev` extras group like this:
```toml
[project.optional-dependencies]
dev = [
"ruff>=0.11.8",
# ... other dev tools
]
Then install with:
uv pip install .[dev]
If development dependencies are listed directly under [project.dependencies]
or there isn't a specific dev
group for them, the standard uv pip install .
or uv sync
should suffice.
Configuration:
Ruff is configured in the pyproject.toml
file under the [tool.ruff]
section. This includes settings for line length, selected rules, and formatting preferences.
Usage:
Make sure your virtual environment is activated (source .venv/bin/activate
).
-
To check for linting issues:
uv run ruff check .
-
To automatically fix linting issues (where possible) and format the code: First, apply formatting:
uv run ruff format .
Then, apply lint fixes:
uv run ruff check . --fix
Ruff's formatter will handle most style issues, and
check --fix
will address other auto-fixable linting errors.
(Details on how to contribute to this project, if applicable.)
(Specify the license for this project, e.g., MIT, Apache 2.0.)