Skip to content

KSoft-Si/vrm-client

Repository files navigation

Victron Energy VRM API Client

An async Python client for the Victron Energy VRM API. This client is designed to be modular, easy to use, and compatible with Home Assistant.

Features

  • Fully async API client using httpx
  • Modular design organized by API categories
  • Comprehensive error handling with custom exceptions
  • Automatic token management and refresh
  • Retry mechanism for failed requests
  • Type hints and Pydantic models for data validation
  • Compatible with Home Assistant

Installation

pip install victron-vrm

Usage

Demo Script

The package includes a demo script that uses the Victron Energy VRM API demo account to demonstrate the basic functionality of the library. You can run it without needing to provide any credentials:

python examples/demo.py

This script will:

  1. Get a demo token using the /auth/loginAsDemo endpoint
  2. Get all sites available in the demo account using the users module
  3. Get user information
  4. Get alarms for the site using the installations module
  5. Get tags for the site
  6. Get statistics for the site
  7. Get timezone for the site

Basic Usage

import asyncio
from victron_vrm import VictronVRMClient

async def main():
    # Create the client with username and password
    async with VictronVRMClient(
        username="your_username",
        password="your_password",
        client_id="your_client_id",
    ) as client:
        # Get all sites using the users module
        sites = await client.users.list_sites()
        print(f"Found {len(sites)} sites")

        # Get a specific site
        if sites:
            site = sites[0]
            # Get alarms for the site using the installations module
            alarms = await client.installations.get_alarms(site.id)
            print(f"Found {len(alarms.alarms)} alarms for site {site.name}")

    # Alternatively, create the client with a token
    async with VictronVRMClient(
        token="your_token",
        token_type="Bearer",  # or "Token" for access tokens
    ) as client:
        # Get all sites
        sites = await client.users.list_sites()
        print(f"Found {len(sites)} sites")

if __name__ == "__main__":
    asyncio.run(main())

Using with Home Assistant

The client is designed to work with Home Assistant's async architecture. You can provide an existing httpx.AsyncClient session and use either username/password or token authentication:

import httpx
from victron_vrm import VictronVRMClient

async def setup_victron_vrm(hass, config):
    # Use Home Assistant's existing httpx session
    session = httpx.AsyncClient()

    # Option 1: Using username and password
    if "username" in config and "password" in config and "client_id" in config:
        client = VictronVRMClient(
            username=config["username"],
            password=config["password"],
            client_id=config["client_id"],
            client_session=session,
        )
    # Option 2: Using a token
    elif "token" in config:
        client = VictronVRMClient(
            token=config["token"],
            token_type=config.get("token_type", "Bearer"),
            client_session=session,
        )
    else:
        raise ValueError("Either username/password/client_id or token must be provided")

    # Get data from the API using the modular approach
    sites = await client.users.list_sites()

    # Get alarms for the first site
    if sites:
        site = sites[0]
        alarms = await client.installations.get_alarms(site.id)

    # Do something with the data
    # ...

    return True

API Reference

Client Initialization

VictronVRMClient(
    username: Optional[str] = None,
    password: Optional[str] = None,
    client_id: Optional[str] = None,
    token: Optional[str] = None,
    token_type: str = "Bearer",
    client_session: Optional[httpx.AsyncClient] = None,
    request_timeout: int = 10,
    max_retries: int = 3,
    retry_delay: int = 1,
)

Either username, password, and client_id OR token must be provided. The token_type can be either "Bearer" (for JWT tokens) or "Token" (for access tokens).

Available Modules

The client is designed with a modular approach, organizing API endpoints into logical modules:

Users Module

Access via client.users:

  • get_me() -> User: Get the current user
  • add_site(installation_identifier: str) -> dict[str, str]: Add a new site
  • list_sites(extended: bool = False, site_id: int | None = None) -> list[Site]: List all sites
  • get_site(site_id: int, extended: bool = False) -> Site | None: Get a specific site
  • create_access_token(name: str, expiry: Optional[int | datetime] = None) -> str: Create an access token
  • list_access_tokens() -> list[AccessToken]: List all access tokens
  • revoke_access_token(token_id: int | AccessToken) -> bool: Revoke an access token
  • get_site_id_from_identifier(installation_identifier: str) -> Optional[int]: Get site ID from identifier
  • search(query: str) -> list[dict[str, Any]]: Search for users

Installations Module

Access via client.installations:

  • get_alarms(site_id: int | Site) -> Alarms: Get alarms for a site
  • add_alarm(site_id: int | Site, alarm: AlarmSettings) -> None: Add an alarm to a site
  • delete_alarm(site_id: int | Site, alarm: AlarmSettings) -> None: Delete an alarm from a site
  • update_alarm(site_id: int | Site, alarm: AlarmSettings) -> AlarmSettings: Update an alarm
  • clear_alarm(site_id: int | Site, alarm: AlarmSettings | int) -> None: Clear an alarm
  • get_tags(site_id: int | Site) -> list[str]: Get tags for a site
  • add_tag(site_id: int | Site, tag: str) -> None: Add a tag to a site
  • delete_tag(site_id: int | Site, tag: str) -> None: Delete a tag from a site
  • stats(site_id: int | Site, ...) -> dict: Get statistics for a site
  • get_python_timezone(site_id: int | Site) -> Any: Get the Python timezone for a site

Error Handling

The client provides custom exceptions for different error scenarios:

from victron_vrm.exceptions import (
    VictronVRMError,
    AuthenticationError,
    AuthorizationError,
    NotFoundError,
    RateLimitError,
    ServerError,
    ClientError,
    ConnectionError,
    TimeoutError,
    ParseError,
)

async def example_with_error_handling():
    try:
        async with VictronVRMClient(...) as client:
            sites = await client.get_sites()
    except AuthenticationError:
        print("Authentication failed")
    except ConnectionError:
        print("Connection error")
    except VictronVRMError as err:
        print(f"API error: {err}")

License

MIT

Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

Development

To set up the development environment:

# Clone the repository
git clone https://github.com/KSoft-Si/victron-vrm.git
cd victron-vrm

# Install the package in development mode with test dependencies
pip install -e ".[test,dev]"

# Alternatively, if you have uv installed
uv pip install -e ".[test,dev]"

Testing

The project includes a test suite that uses the Victron Energy VRM API demo account. To run the tests:

# Run all tests
pytest

# Run with coverage report
pytest --cov=victron_vrm

# Run a specific test
pytest tests/test_client.py::test_get_sites

See the tests/README.md file for more information about the test suite.

CI/CD Workflow

This project uses GitHub Actions for continuous integration and deployment:

  • Automated Testing: All pull requests and pushes to all branches are automatically tested against Python 3.11, 3.12, and 3.13.
  • Automatic Versioning: When code is pushed to the main branch, the version is automatically incremented (patch version), a new tag is created, and a GitHub release is generated.
  • Package Publishing: When code is pushed to the main branch, the package is automatically built and published to PyPI.

The workflow uses uv instead of pip for faster and more reliable dependency management.

Automatic Releases

The project is configured with automatic versioning and releases:

  1. When you push changes to the main branch, a GitHub Actions workflow automatically:
    • Increments the patch version in pyproject.toml (e.g., 0.1.0 → 0.1.1)
    • Commits the version change
    • Creates a new tag (e.g., v0.1.1)
    • Creates a GitHub release with automatically generated release notes
    • Builds and publishes the package to PyPI

This means you don't need to manually update versions or create tags for routine updates.

Manual Releases

For major or minor version updates, you can still create releases manually:

  1. Update the version in pyproject.toml (e.g., change 0.1.0 to 0.2.0 or 1.0.0)
  2. Commit the changes: git commit -am "Bump version to X.Y.Z"
  3. Create a new tag: git tag vX.Y.Z
  4. Push the changes and tag: git push && git push --tags

The GitHub Actions workflow will automatically build and publish the new version to PyPI.