Skip to content

bjhiltbrand/drone_mobile_python

Folders and files

NameName
Last commit message
Last commit date

Latest commit

ย 

History

82 Commits
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 

Repository files navigation

drone_mobile-python

Upload Python Package PyPI version Python Versions License: MIT

A modern, fully-typed Python wrapper for the DroneMobile API. Control your Firstech/Compustar remote start system from Python.

โš ๏ธ Disclaimer

The code here is based off of an unsupported API from DroneMobile and is subject to change without notice. The authors claim no responsibility for damages to your vehicle by use of the code within.

โœจ Features

  • ๐Ÿ” Secure Authentication - Automatic token management with secure storage
  • ๐Ÿ“ฒ MFA / 2-Factor Authentication - Full support for SMS and TOTP (authenticator app) challenges
  • ๐Ÿ’พ Device Remembering - Stays authenticated across refresh-token expiry without re-prompting for MFA
  • ๐Ÿš— Vehicle Control - Start/stop engine, lock/unlock doors, and more
  • ๐Ÿ“Š Status Monitoring - Get detailed vehicle status information
  • ๐ŸŽฏ Type Safe - Full type hints for better IDE support
  • ๐Ÿงช Well Tested - Comprehensive test suite
  • ๐Ÿ› ๏ธ CLI Tool - Command-line interface for quick operations
  • ๐Ÿ”„ Context Manager - Clean resource management
  • ๐Ÿ“ Detailed Logging - Debug and track API interactions

๐Ÿ“ฆ Installation

pip install drone_mobile

For development:

pip install drone_mobile[dev]

๐Ÿš€ Quick Start

Basic Usage

from drone_mobile import DroneMobileClient

# Create a client and authenticate
client = DroneMobileClient("your_email@example.com", "your_password")

# Get all vehicles
vehicles = client.get_vehicles()

# Work with the first vehicle
vehicle = vehicles[0]
print(f"Vehicle: {vehicle.name}")

# Get status
status = vehicle.get_status()
print(f"Running: {status.is_running}")
print(f"Locked: {status.is_locked}")
print(f"Battery: {status.battery_percent}%")

# Control the vehicle
vehicle.start()  # Start the engine
vehicle.unlock()  # Unlock doors
vehicle.lock()  # Lock doors
vehicle.stop()  # Stop the engine

Using Context Manager

from drone_mobile import DroneMobileClient

# Automatically handles cleanup
with DroneMobileClient("email@example.com", "password") as client:
    vehicles = client.get_vehicles()
    for vehicle in vehicles:
        status = vehicle.get_status()
        print(f"{vehicle.name}: {'Running' if status.is_running else 'Stopped'}")

MFA / 2-Factor Authentication

If your DroneMobile account has MFA enabled, pass an mfa_callback that returns the one-time code:

from drone_mobile import DroneMobileClient

def prompt_mfa(challenge_name: str) -> str:
    label = "SMS code" if challenge_name == "SMS_MFA" else "Authenticator app code"
    return input(f"Two-factor authentication required. Enter {label}: ").strip()

client = DroneMobileClient("email@example.com", "password", mfa_callback=prompt_mfa)
vehicles = client.get_vehicles()

Both SMS_MFA (text message) and SOFTWARE_TOKEN_MFA (Google Authenticator, Authy, etc.) are supported.

After the first successful MFA login, the library remembers the device using Cognito's device SRP scheme. Subsequent re-authentications โ€” including those triggered by refresh-token expiry โ€” happen silently in the background without triggering another MFA prompt.

If MFA is required but no callback is supplied, MFARequiredError is raised:

from drone_mobile.exceptions import MFARequiredError

try:
    client = DroneMobileClient("email@example.com", "password")
    vehicles = client.get_vehicles()
except MFARequiredError as e:
    print(f"MFA required: {e.challenge_name}")
    # Re-create the client with a callback and retry

Error Handling

from drone_mobile import DroneMobileClient
from drone_mobile.exceptions import (
    AuthenticationError,
    CommandFailedError,
    VehicleNotFoundError,
)

try:
    client = DroneMobileClient("email@example.com", "password")
    vehicle = client.get_vehicle("vehicle_id")
    vehicle.start()
except AuthenticationError:
    print("Invalid credentials")
except CommandFailedError as e:
    print(f"Command failed: {e}")
except VehicleNotFoundError:
    print("Vehicle not found")

๐Ÿ–ฅ๏ธ Command Line Interface

The package includes a CLI tool for quick operations:

# List all vehicles
drone-mobile-demo user@example.com password list

# Show vehicle status
drone-mobile-demo user@example.com password status

# Send commands
drone-mobile-demo user@example.com password cmd start
drone-mobile-demo user@example.com password cmd lock
drone-mobile-demo user@example.com password cmd unlock
drone-mobile-demo user@example.com password cmd stop

# Use verbose logging
drone-mobile-demo -v user@example.com password status

The CLI automatically handles MFA challenges interactively โ€” no extra flags needed.

๐Ÿ“š API Reference

DroneMobileClient

Main client for interacting with the DroneMobile API.

Constructor

DroneMobileClient(
    username: str,
    password: str,
    token_dir: Path | None = None,
    mfa_callback: Callable[[str], str] | None = None,
)

Methods

  • get_vehicles() -> List[Vehicle] - Get all vehicles
  • get_vehicle(vehicle_id: str) -> Vehicle - Get specific vehicle
  • get_vehicle_status(vehicle_id: str) -> VehicleStatus - Get vehicle status
  • send_command(device_key: str, command: str) -> CommandResponse - Send command

Vehicle

Represents a vehicle with control methods.

Properties

  • vehicle_id: str - Unique vehicle identifier
  • device_key: str - Device key for commands
  • name: str - Vehicle name
  • info: VehicleInfo - Detailed vehicle information

Methods

  • get_status() -> VehicleStatus - Get current status
  • start() -> CommandResponse - Start engine
  • stop() -> CommandResponse - Stop engine
  • lock() -> CommandResponse - Lock doors
  • unlock() -> CommandResponse - Unlock doors
  • trunk() -> CommandResponse - Open trunk
  • panic_on() -> CommandResponse - Activate panic
  • panic_off() -> CommandResponse - Deactivate panic
  • aux1() -> CommandResponse - Trigger auxiliary 1
  • aux2() -> CommandResponse - Trigger auxiliary 2
  • get_location() -> CommandResponse - Get GPS location
  • set_features(**features: bool) -> dict - Update controller feature flags

Data Models

VehicleStatus

@dataclass
class VehicleStatus:
    vehicle_id: str
    device_key: str
    is_running: bool
    is_locked: bool
    battery_voltage: Optional[float]
    battery_percent: Optional[int]
    odometer: Optional[float]
    fuel_level: Optional[int]
    interior_temperature: Optional[float]
    exterior_temperature: Optional[float]
    location: Optional[Location]
    last_updated: Optional[datetime]
    raw_data: Dict[str, Any]

VehicleInfo

@dataclass
class VehicleInfo:
    vehicle_id: str
    device_key: str
    name: str
    make: Optional[str]
    model: Optional[str]
    year: Optional[int]
    color: Optional[str]
    vin: Optional[str]
    raw_data: Dict[str, Any]

๐Ÿ”’ Security

  • Tokens are stored securely in ~/.config/drone_mobile/ with restrictive permissions (0600)
  • The remembered-device secret (device.json) is stored alongside tokens with the same permissions
  • Sensitive data (OTP codes, tokens) is never written to logs at any level
  • Automatic token refresh prevents credential exposure
  • The Cognito Session challenge token is kept in memory only and never persisted

๐Ÿงช Development

Setup Development Environment

# Clone the repository
git clone https://github.com/bjhiltbrand/drone_mobile_python.git
cd drone_mobile_python

# Install with development dependencies
pip install -e ".[dev]"

# Install pre-commit hooks
pre-commit install

Running Tests

# Run all tests
pytest

# Run with coverage
pytest --cov=drone_mobile --cov-report=html

# Run specific test file
pytest tests/test_auth.py
pytest tests/test_device_srp.py
pytest tests/test_device_remembering.py

Code Quality

# Format code
black drone_mobile/ tests/

# Lint
ruff check drone_mobile/

# Type check
mypy drone_mobile/

# Run all checks
make format lint type-check test

๐Ÿ“ Migration from 0.3.x to 0.4.x

No breaking changes. This release is fully backward-compatible.

Accounts without MFA are unaffected โ€” authentication works exactly as before.

Accounts with MFA benefit from device remembering automatically: after the first successful MFA login, the library stores a device secret in ~/.config/drone_mobile/device.json and uses Cognito's device SRP flow on subsequent re-authentications, so you won't be prompted for a second factor again unless you explicitly clear the stored device.

๐Ÿ“ Migration from 0.2.x

If you're upgrading from version 0.2.x, see CHANGELOG.md for a detailed migration guide.

Quick changes:

  • Import DroneMobileClient instead of Vehicle
  • Use get_vehicles() instead of getAllVehicles()
  • Methods now return typed objects instead of raw dicts
  • Token storage moved to ~/.config/drone_mobile/

๐Ÿค Contributing

Contributions are welcome! Please:

  1. Fork the repository
  2. Create a feature branch
  3. Make your changes with tests
  4. Run the test suite
  5. Submit a pull request

๐Ÿ“„ License

This project is licensed under the MIT License - see the LICENSE.md file for details.

๐Ÿ™ Acknowledgments

  • Original API reverse engineering by the community
  • Thanks to all contributors

๐Ÿ“ฎ Support

โš–๏ธ Legal

This is an unofficial API wrapper and is not affiliated with, endorsed by, or connected to DroneMobile, Firstech, or Compustar in any way.

About

Python library to interface with the DroneMobile API

Resources

License

Contributing

Stars

Watchers

Forks

Packages

 
 
 

Contributors