Modern Python CLI for Odoo with support for helpdesk, projects, tasks, CRM, and knowledge articles
Project description
Vodoo
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, 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 โ
OdooClientwith namespace helpers (client.helpdesk,client.crm, etc.) - โก Full async support via
vodoo.aioโAsyncOdooClientwith 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, orOdooConfig - ๐ 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
ODOO_DEFAULT_USER_ID=123 # Optional: default user for sudo operations
Or set environment variables directly, or pass values to OdooConfig() in code.
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
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 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:
- Getting Started โ Installation, configuration, quick start
- CLI Reference โ All commands with examples
- Library Guide โ Using Vodoo as a Python library
- API Reference โ Auto-generated from docstrings
- Security Guide โ Service account setup
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
โโโ 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.
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file vodoo-3.0.0.tar.gz.
File metadata
- Download URL: vodoo-3.0.0.tar.gz
- Upload date:
- Size: 1.7 MB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e358d4093ff2987c8652d85c7c340337a21688d34ffbb466e1710c5995e0405b
|
|
| MD5 |
8135dd678dd656894daa0b35a197ca88
|
|
| BLAKE2b-256 |
0ca060a44f59fe22483eac71f509b99da33bdd81106c2bf958e1901b52214408
|
Provenance
The following attestation bundles were made for vodoo-3.0.0.tar.gz:
Publisher:
publish.yml on julian-r/vodoo
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
vodoo-3.0.0.tar.gz -
Subject digest:
e358d4093ff2987c8652d85c7c340337a21688d34ffbb466e1710c5995e0405b - Sigstore transparency entry: 953488248
- Sigstore integration time:
-
Permalink:
julian-r/vodoo@3b8f676458f337aae1cbb8ca3be286531e6aac9e -
Branch / Tag:
refs/tags/v3.0.0 - Owner: https://github.com/julian-r
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@3b8f676458f337aae1cbb8ca3be286531e6aac9e -
Trigger Event:
release
-
Statement type:
File details
Details for the file vodoo-3.0.0-py3-none-any.whl.
File metadata
- Download URL: vodoo-3.0.0-py3-none-any.whl
- Upload date:
- Size: 80.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7bc4dc19937e340d547a39c0443d426db7d7db13180013b6f6936f4b7dd391f8
|
|
| MD5 |
1782ec78beb17e361bd68e8fbad0827c
|
|
| BLAKE2b-256 |
f01bcb157c6796ed0646b0389ebc205fa4d5e941e4ccfd2b183c0a82d6c48011
|
Provenance
The following attestation bundles were made for vodoo-3.0.0-py3-none-any.whl:
Publisher:
publish.yml on julian-r/vodoo
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
vodoo-3.0.0-py3-none-any.whl -
Subject digest:
7bc4dc19937e340d547a39c0443d426db7d7db13180013b6f6936f4b7dd391f8 - Sigstore transparency entry: 953488251
- Sigstore integration time:
-
Permalink:
julian-r/vodoo@3b8f676458f337aae1cbb8ca3be286531e6aac9e -
Branch / Tag:
refs/tags/v3.0.0 - Owner: https://github.com/julian-r
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@3b8f676458f337aae1cbb8ca3be286531e6aac9e -
Trigger Event:
release
-
Statement type: