Skip to main content

Modern Python CLI for Odoo with support for helpdesk, projects, tasks, CRM, and knowledge articles

Project description

Vodoo

Vodoo

License: MIT Python 3.12+ PyPI Documentation Code style: ruff Type checked: mypy

A Python library and CLI for Odoo. Use it as a library in your own scripts, services, and automations โ€” or as a CLI for quick ad-hoc operations and AI-assisted workflows.

Supports helpdesk tickets, project tasks, projects, CRM leads/opportunities, accounting moves, knowledge articles, and timesheets across Odoo 17โ€“19.

๐Ÿ“– Full Documentation โ€” Getting started, CLI reference, Python library guide, and API docs.

Quick Start โ€” Library

from vodoo import OdooClient, OdooConfig, RecordNotFoundError

config = OdooConfig(
    url="https://my-instance.odoo.com",
    database="mydb",
    username="bot@example.com",
    password="api-key-or-password",
)
client = OdooClient(config)

# Namespace helpers on the client
tasks = client.tasks.list(limit=10)

# Generic client for any model
partners = client.search_read("res.partner", fields=["name", "email"], limit=5)

# Structured exceptions โ€” catch what you need
try:
    record = client.generic.search("res.partner", [("id", "=", 999999999)])
except RecordNotFoundError as e:
    print(f"{e.model} #{e.record_id} not found")

Quick Start โ€” CLI

# Run without installing
uvx vodoo crm list --search "acme"

# Or install globally
pip install vodoo
vodoo helpdesk list --stage "In Progress"
vodoo project-task show 42
vodoo timer start 42

Works well with AI assistants like Claude Code โ€” natural language in, structured Odoo operations out.

Odoo Version Support

Version Protocol Status
Odoo 17 Legacy JSON-RPC โœ… Fully tested
Odoo 18 Legacy JSON-RPC โœ… Fully tested
Odoo 19 JSON-2 (bearer auth) โœ… Fully tested

Auto-detects the Odoo version and selects the appropriate transport. Odoo 19's JSON-2 API is ~3โ€“4x faster than legacy JSON-RPC.

Features

Library

  • ๐Ÿ Clean Python API โ€” OdooClient with namespace helpers (client.helpdesk, client.crm, etc.)
  • โšก Full async support via vodoo.aio โ€” AsyncOdooClient with async context manager
  • ๐ŸŽฏ Structured exception hierarchy mirroring Odoo server errors
  • ๐Ÿ“ฆ No CLI dependencies loaded when imported as a library
  • ๐Ÿ”’ Strict mypy typing throughout

CLI

  • ๐Ÿ“‹ Helpdesk tickets, project tasks, projects, CRM leads, knowledge articles
  • โฑ๏ธ Timer / timesheet management (start, stop, status)
  • ๐Ÿ’ฌ Comments, internal notes, tags, attachments
  • ๐Ÿ” Search across text fields (name, email, phone, description)
  • ๐Ÿงฐ Generic CRUD for any Odoo model
  • ๐ŸŽจ Rich terminal output with tables

Shared

  • ๐Ÿ”€ Auto-detecting transport layer (JSON-2 for Odoo 19+, legacy JSON-RPC for 17โ€“18)
  • โš™๏ธ Configuration via environment variables, .env, or OdooConfig
  • ๐Ÿ” HTTPS enforcement warnings for production safety

Installation

# From PyPI
pip install vodoo

# Or run the CLI without installing
uvx vodoo helpdesk list

# From source (development)
git clone https://github.com/julian-r/vodoo.git
cd vodoo
uv sync --all-extras

Configuration

Create a .vodoo.env, ~/.config/vodoo/config.env, or .env file:

ODOO_URL=https://your-odoo-instance.com
ODOO_DATABASE=your_database
ODOO_USERNAME=your_username
ODOO_PASSWORD=your_password_or_api_key
# Optional alternative: ODOO_PASSWORD_REF=op://Vault/Item/password
ODOO_DEFAULT_USER_ID=123  # Optional: default user for sudo operations

Multi-instance profiles are also supported:

  • .vodoo/instances/<instance>.env
  • ~/.config/vodoo/instances/<instance>.env
  • select via vodoo --instance <instance> ... or VODOO_INSTANCE=<instance>

Or set environment variables directly, or pass values to OdooConfig() in code.

Useful config helpers:

vodoo config list-instances
vodoo config show
vodoo config use staging
vodoo config test --instance staging

Exception Hierarchy

All exceptions inherit from VodooError so you can catch broadly or narrowly:

VodooError
โ”œโ”€โ”€ ConfigurationError
โ”œโ”€โ”€ AuthenticationError
โ”œโ”€โ”€ RecordNotFoundError          โ† .model, .record_id attributes
โ”œโ”€โ”€ RecordOperationError
โ”œโ”€โ”€ TransportError               โ† .code, .data attributes
โ”‚   โ””โ”€โ”€ OdooUserError            โ† odoo.exceptions.UserError
โ”‚       โ”œโ”€โ”€ OdooAccessDeniedError
โ”‚       โ”œโ”€โ”€ OdooAccessError
โ”‚       โ”œโ”€โ”€ OdooMissingError
โ”‚       โ””โ”€โ”€ OdooValidationError
โ””โ”€โ”€ FieldParsingError

Server-side Odoo errors are automatically mapped to the matching exception class, so you can handle OdooAccessError separately from OdooValidationError without parsing error strings.

Library Usage

Namespace Helpers

Each Odoo domain is a namespace on the client with high-level methods: from vodoo import OdooClient, OdooConfig client = OdooClient(OdooConfig( url="https://odoo.example.com", database="prod", username="bot@example.com", password="api-key", ))

Project tasks

tasks = client.tasks.list(domain=[("stage_id.name", "=", "In Progress")], limit=20) task = client.tasks.get(42) client.tasks.comment(42, "Deployed to staging") leads = client.crm.list(domain=[("type", "=", "opportunity")], limit=20) client.crm.set(123, {"expected_revenue": 50000, "probability": 75}) tickets = client.helpdesk.list(domain=[("stage_id.name", "=", "New")], limit=10) client.timer.start_task(task_id=42) client.timer.stop() records = client.search_read("res.partner", [("is_company", "=", True)], fields=["name"]) new_id = client.create("res.partner", {"name": "Acme Corp", "is_company": True}) client.write("res.partner", [new_id], {"phone": "+1234567890"})


### Error Handling

```python
from vodoo import (
    OdooClient, OdooConfig, VodooError,
    AuthenticationError, RecordNotFoundError,
    OdooAccessError, OdooValidationError,
)

try:
    client = OdooClient(OdooConfig(...))
    client.write("res.partner", [999], {"name": "Updated"})
except AuthenticationError:
    print("Bad credentials")
except RecordNotFoundError as e:
    print(f"{e.model} #{e.record_id} does not exist")
except OdooAccessError:
    print("Insufficient permissions")
except OdooValidationError:
    print("Data constraint violated")
except VodooError as e:
    print(f"Something else went wrong: {e}")

Async Usage

All library functionality is also available as async via vodoo.aio:

from vodoo import OdooConfig
from vodoo.aio import AsyncOdooClient

config = OdooConfig(
    url="https://my-instance.odoo.com",
    database="mydb",
    username="bot@example.com",
    password="api-key",
)

async with AsyncOdooClient(config) as client:
    # Namespace helpers
    tasks = await client.tasks.list(limit=10)
    partners = await client.search_read("res.partner", fields=["name", "email"], limit=5)
    # Comments / notes
    await client.crm.comment(123, "Async update")

Every sync namespace has an async counterpart โ€” same methods, just awaited.

CLI Usage

CRM Leads/Opportunities

vodoo crm list --search "acme" --type opportunity --stage "Qualified"
vodoo crm show 123
vodoo crm set 123 expected_revenue=50000 probability=75
vodoo crm note 123 "Followed up via phone"
vodoo crm attach 123 proposal.pdf
vodoo crm url 123

Project Tasks

vodoo project-task list --stage "In Progress"
vodoo project-task show 42
vodoo project-task comment 42 "Deployed to staging"
vodoo project-task attach 42 screenshot.png

Accounting Moves

vodoo account-move list --company "Rath Technologie" --year 2025 --state posted
vodoo account-move attachments 3552
vodoo account-move download-all 3552 --output "~/Belege 2025" --extension pdf

Projects

vodoo project list
vodoo project show 1
vodoo project note 1 "Sprint planning notes"

Helpdesk Tickets (Enterprise)

vodoo helpdesk list --stage "New" --assigned-to "John"
vodoo helpdesk show 123
vodoo helpdesk note 123 "Internal update"
vodoo helpdesk comment 123 "We're looking into this"
vodoo helpdesk download 456 --output ./attachments/

Knowledge Articles (Enterprise)

vodoo knowledge list --category workspace
vodoo knowledge create "Team Handbook" --body "# Onboarding\n\nWelcome!" --category workspace
vodoo knowledge show 123
vodoo knowledge note 123 "Updated installation section"

Timers / Timesheets

vodoo timer start 42
vodoo timer status
vodoo timer stop

Generic Model Operations

vodoo model read res.partner --domain='[["is_company","=",true]]' --field name --field email
vodoo model create res.partner name="Acme" email=info@acme.com
vodoo model update res.partner 123 phone="+123456789"
vodoo model delete res.partner 123
vodoo model call res.partner name_search --args='["Acme"]'

Security / Service Accounts

vodoo security create-groups
vodoo security assign-bot --login service-vodoo@company.com

For production use, run Vodoo with a dedicated least-privilege service account. See the Security Guide.

Documentation

Full docs at julian-r.github.io/vodoo:

Project Structure

src/vodoo/
โ”œโ”€โ”€ __init__.py           # Public API: OdooClient, OdooConfig, exceptions
โ”œโ”€โ”€ exceptions.py         # Exception hierarchy (VodooError and subclasses)
โ”œโ”€โ”€ client.py             # OdooClient โ€” delegates to transport layer
โ”œโ”€โ”€ transport.py          # Transport abstraction (JSON-2 + legacy JSON-RPC)
โ”œโ”€โ”€ config.py             # Pydantic configuration from env/.env files
โ”œโ”€โ”€ auth.py               # Authentication and sudo utilities
โ”œโ”€โ”€ _domain.py            # DomainNamespace base โ€” shared CRUD, messaging, attachments
โ”œโ”€โ”€ main.py               # CLI entry point (Typer) โ€” not loaded by library imports
โ”œโ”€โ”€ helpdesk.py           # Helpdesk ticket operations (enterprise)
โ”œโ”€โ”€ project_tasks.py      # Project task operations
โ”œโ”€โ”€ projects.py           # Project operations
โ”œโ”€โ”€ crm.py                # CRM lead/opportunity operations
โ”œโ”€โ”€ account_moves.py      # Accounting move operations
โ”œโ”€โ”€ knowledge.py          # Knowledge article operations (enterprise)
โ”œโ”€โ”€ generic.py            # Generic model CRUD
โ”œโ”€โ”€ security.py           # Security groups, user management
โ”œโ”€โ”€ timer.py              # Timer/timesheet start, stop, status
โ””โ”€โ”€ aio/                  # Async versions of all modules above
    โ”œโ”€โ”€ client.py         # AsyncOdooClient
    โ”œโ”€โ”€ _domain.py        # AsyncDomainNamespace base
    โ”œโ”€โ”€ transport.py      # Async JSON-2 + legacy transports
    โ””โ”€โ”€ ...               # Async domain modules (same API, awaitable)

Integration Tests

75+ tests per Odoo version against real instances in Docker:

./tests/integration/run.sh           # All community editions (17, 18, 19)
./tests/integration/run.sh 19        # Just Odoo 19
ENTERPRISE=1 ./tests/integration/run.sh 19   # Include enterprise

Development

uv sync --all-extras
uv run ruff check .
uv run ruff format .
uv run mypy src/vodoo

Publishing

Version is derived from git tags via hatch-vcs:

git tag vX.Y.Z && git push origin vX.Y.Z

GitHub Actions builds and publishes to PyPI automatically.

License

MIT โ€” see LICENSE. Copyright (c) 2025 Julian Rath.

Built with Typer, Rich, Pydantic, uv, Ruff, and mypy.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

vodoo-3.1.1.tar.gz (1.8 MB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

vodoo-3.1.1-py3-none-any.whl (99.6 kB view details)

Uploaded Python 3

File details

Details for the file vodoo-3.1.1.tar.gz.

File metadata

  • Download URL: vodoo-3.1.1.tar.gz
  • Upload date:
  • Size: 1.8 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for vodoo-3.1.1.tar.gz
Algorithm Hash digest
SHA256 a0d3a6834d6117c7eb7a40d34558556c5a86966c047e31586a490fa421088e82
MD5 84a657282b0b6ec16a8a19af44d24de1
BLAKE2b-256 9258b517ea5dab532a1d4ba9829e6a9c8dda13431d9e010d8168fecf31d316bf

See more details on using hashes here.

Provenance

The following attestation bundles were made for vodoo-3.1.1.tar.gz:

Publisher: publish.yml on julian-r/vodoo

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file vodoo-3.1.1-py3-none-any.whl.

File metadata

  • Download URL: vodoo-3.1.1-py3-none-any.whl
  • Upload date:
  • Size: 99.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for vodoo-3.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 be756e88c7d8f315560156170cd9708db0e4e8ca8921c0f20d976659849801e9
MD5 d42cc502e0f6e4bec933ac0726201737
BLAKE2b-256 fe4d6bdf4226bc6c67e5d2b52a890a3d48121814596d4e5ffa625f28f69291b4

See more details on using hashes here.

Provenance

The following attestation bundles were made for vodoo-3.1.1-py3-none-any.whl:

Publisher: publish.yml on julian-r/vodoo

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page