Skip to main content

MCP Server for Odoo Integration

Project description

Odoo MCP

A precise MCP bridge for Odoo.
Give AI agents a safe, typed, and testable way to read, inspect, diagnose, and operate Odoo.

PyPI Python License

Odoo MCP turns an Odoo database into a Model Context Protocol server. It is built for local agents, IDEs, and automation tools that need real Odoo context without hand-rolled scripts or unsafe direct write access.

It speaks XML-RPC for Odoo 16-18 and External JSON-2 for Odoo 19. It exposes a compact MCP surface with read tools, diagnostics, schema discovery, migration helpers, local addon scanning, and a gated write workflow.

Highlights

Capability What it gives you
22 MCP tools Read records, inspect schema, build domains, scan addons, diagnose calls, access rules, and validate writes.
5 agent prompts Reusable workflows for failed calls, fit/gap workshops, JSON-2 migration, safe writes, and module audits.
Odoo 16-19 coverage XML-RPC by default, JSON-2 opt-in for Odoo 19.
Streamable HTTP Local HTTP/SSE support for clients that do not use stdio.
Safe writes Direct create, write, and unlink are blocked; approved writes require live metadata, a same-session token, explicit confirmation, and an env gate.
Real smoke tests Docker Compose validation boots disposable Odoo 16.0, 17.0, 18.0, and 19.0 stacks, including restricted users, custom record rules, and packaged addon XML install/update.

Install

pip install odoo-mcp

For local development:

git clone https://github.com/tuanle96/mcp-odoo.git
cd mcp-odoo
uv sync --extra dev

Configure

Set connection values in the environment:

export ODOO_URL="https://your-odoo-instance.com"
export ODOO_DB="your-database"
export ODOO_USERNAME="your-user"
export ODOO_PASSWORD="your-password-or-api-key"
export ODOO_TRANSPORT="xmlrpc"

For Odoo 19 JSON-2:

export ODOO_TRANSPORT="json2"
export ODOO_API_KEY="your-odoo-api-key"
export ODOO_JSON2_DATABASE_HEADER="1"

ODOO_JSON2_DATABASE_HEADER=1 sends X-Odoo-Database on JSON-2 calls. Set it to 0 only when host or dbfilter routing already selects the intended database.

You can also use odoo_config.json:

{
  "url": "https://your-odoo-instance.com",
  "db": "your-database",
  "username": "your-user",
  "password": "your-password-or-api-key"
}

Run

Start the MCP server over stdio:

odoo-mcp

or:

python -m odoo_mcp

Start Streamable HTTP for local clients:

odoo-mcp --transport streamable-http --host 127.0.0.1 --port 8000 --path /mcp

Non-local HTTP binds are rejected unless you pass --allow-remote-http or set MCP_ALLOW_REMOTE_HTTP=1. This server does not include built-in HTTP authentication. Put remote HTTP deployments behind your own authentication, TLS, and network policy.

Check runtime posture without starting the server loop:

odoo-mcp --health

MCP Tools

Tool Purpose
execute_method Execute a reviewed model method. Direct create, write, and unlink are blocked. Side-effect methods require an exact allowlist or ODOO_MCP_ALLOW_UNKNOWN_METHODS=1.
list_models List Odoo model technical names and labels.
get_model_fields Read field metadata for one model.
search_records Run bounded read-only search_read.
read_record Read one record by model and ID.
search_employee Search employees by name.
search_holidays Search leave records by date range.
diagnose_odoo_call Diagnose a model call without executing it.
diagnose_access Diagnose ACL and record-rule visibility for the current Odoo credential.
inspect_model_relationships Group relationship fields, required fields, and create/write hints.
generate_json2_payload Convert XML-RPC-shaped input into JSON-2 endpoint, headers, and named body.
upgrade_risk_report Surface transport, method, and migration risks across Odoo versions.
fit_gap_report Classify requirements into standard, configuration, Studio, custom module, avoid, or unknown.
get_odoo_profile Read server version, user context, transport, database, and installed module summary.
schema_catalog Build a bounded model catalog with optional field metadata.
preview_write Produce a non-executing approval payload for create, write, or unlink.
validate_write Validate a write payload against trusted live fields_get metadata.
execute_approved_write Execute only a same-session, live-validated, confirmed write when ODOO_MCP_ENABLE_WRITES=1.
scan_addons_source Scan local addon source without importing addon code.
build_domain Build and validate an Odoo domain from structured conditions.
business_pack_report Report expected modules, models, and discovery calls for sales, CRM, inventory, accounting, or HR.
health_check Report non-secret MCP runtime posture.

Resources

URI Description
odoo://models List available models.
odoo://model/{model_name} Read model metadata and fields.
odoo://record/{model_name}/{record_id} Read one record.
odoo://search/{model_name}/{domain} Search records with a bounded domain.

Prompts

Prompt Use it for
diagnose_failed_odoo_call Root-cause a failing Odoo call before retrying.
fit_gap_workshop Turn raw requirements into Odoo fit/gap buckets.
json2_migration_plan Plan XML-RPC or JSON-RPC migration to External JSON-2.
safe_write_review Review a proposed create, write, or unlink.
custom_module_audit Audit local addon source with scan, risk, and business evidence.

Safe Write Model

Writes are intentionally boring.

  1. preview_write creates a canonical, non-executing payload.
  2. validate_write checks model metadata, required fields, readonly fields, relation hints, record IDs, and payload shape.
  3. execute_approved_write runs only when all gates pass:
    • the approval came from validate_write in the same server process,
    • validation used trusted, non-empty live Odoo fields_get metadata,
    • the token has not expired or been consumed,
    • confirm=true is passed,
    • ODOO_MCP_ENABLE_WRITES=1 is set.

Odoo access rules, record rules, and server-side constraints still decide the final result.

Reviewed side-effect methods such as sale.order.action_confirm can be enabled one by one:

export ODOO_MCP_ALLOWED_SIDE_EFFECT_METHODS="sale.order.action_confirm,res.partner.message_post"

ODOO_MCP_ALLOW_UNKNOWN_METHODS=1 is still supported for trusted deployments, but health_check reports it as broad mode. Prefer exact allowlist entries when you only need a small number of reviewed methods.

Client Setup

Claude Desktop on macOS reads MCP configuration from:

~/Library/Application Support/Claude/claude_desktop_config.json

Use an absolute Python path because GUI apps may not inherit your shell PATH:

{
  "mcpServers": {
    "odoo": {
      "command": "/opt/homebrew/bin/python3",
      "args": ["-m", "odoo_mcp"],
      "env": {
        "ODOO_URL": "https://your-odoo-instance.com",
        "ODOO_DB": "your-database",
        "ODOO_USERNAME": "your-user",
        "ODOO_PASSWORD": "your-password-or-api-key",
        "ODOO_TRANSPORT": "xmlrpc"
      }
    }
  }
}

More examples are in docs/client-configs.md.

Docker

Build the image:

docker build -t mcp/odoo:latest -f Dockerfile .

Run over stdio from an MCP client:

{
  "mcpServers": {
    "odoo": {
      "command": "docker",
      "args": [
        "run",
        "-i",
        "--rm",
        "-e", "ODOO_URL",
        "-e", "ODOO_DB",
        "-e", "ODOO_USERNAME",
        "-e", "ODOO_PASSWORD",
        "-e", "ODOO_TRANSPORT",
        "-e", "ODOO_API_KEY",
        "mcp/odoo:latest"
      ]
    }
  }
}

Run Streamable HTTP locally:

docker run --rm \
  -p 127.0.0.1:8000:8000 \
  -e ODOO_URL \
  -e ODOO_DB \
  -e ODOO_USERNAME \
  -e ODOO_PASSWORD \
  -e ODOO_TRANSPORT \
  -e ODOO_API_KEY \
  mcp/odoo:latest \
  --transport streamable-http \
  --host 0.0.0.0 \
  --port 8000 \
  --allow-remote-http

Test

Run the normal quality gates:

uv run python -m ruff check .
uv run python -m mypy src
uv run python -m pytest

Run real Odoo smoke tests:

uv run --python 3.12 --with-editable . scripts/odoo_compose_smoke.py \
  --versions 16.0 17.0 18.0 19.0 \
  --timeout 360 \
  --inspector-smoke

The smoke harness boots disposable Docker Compose stacks, validates direct Odoo access, validates MCP stdio, and for Odoo 19 also validates JSON-2 and Streamable HTTP.

Compatibility

XML-RPC remains the default transport for broad compatibility. Odoo 19 supports External JSON-2 through ODOO_TRANSPORT=json2. Odoo has documented XML-RPC and JSON-RPC deprecation for Odoo 20, so new integrations should plan for JSON-2.

Contributing

Issues, pull requests, and compatibility reports are welcome. Start with CONTRIBUTING.md, include your Odoo version, transport, client type, and the verification you ran.

Security

Do not publish logs that contain Odoo credentials, API keys, database names from private environments, or full Odoo debug traces. Report vulnerabilities through SECURITY.md.

License

MIT. See LICENSE.

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

odoo_mcp-0.2.0.tar.gz (75.6 kB view details)

Uploaded Source

Built Distribution

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

odoo_mcp-0.2.0-py3-none-any.whl (44.4 kB view details)

Uploaded Python 3

File details

Details for the file odoo_mcp-0.2.0.tar.gz.

File metadata

  • Download URL: odoo_mcp-0.2.0.tar.gz
  • Upload date:
  • Size: 75.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.10.20

File hashes

Hashes for odoo_mcp-0.2.0.tar.gz
Algorithm Hash digest
SHA256 53c8735304d2f80bea1c58d5710ec76228282b62a5a1f7920243a4c182d2abda
MD5 cb44ff60775ff71870eeafc34f3e14ed
BLAKE2b-256 1a06fe1536f9974ed147bfb6a315625d7ef67987bdd1ffa242f5d02c4cab6bac

See more details on using hashes here.

File details

Details for the file odoo_mcp-0.2.0-py3-none-any.whl.

File metadata

  • Download URL: odoo_mcp-0.2.0-py3-none-any.whl
  • Upload date:
  • Size: 44.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.10.20

File hashes

Hashes for odoo_mcp-0.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 7ab8492e204a2f18de5c7c788132aa98a994902032c0b14e97f439cee7df8233
MD5 d311c438d4492c2172d8a27b41a8b28d
BLAKE2b-256 497703e2e288a9b92017a85704940314b124e64a4c3a5c31edebd93d865ef1eb

See more details on using hashes here.

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