Skip to main content

Modern Python client for CiviCRM API v4 with web framework integration

Project description

civicrm-py

CI PyPI version Python Version License: MIT

A Python client for CiviCRM's API v4. Async by default, with a query interface that'll feel familiar if you've used Django's ORM.

What it does

  • Async HTTP via httpx (sync wrappers available)
  • Type-safe models using msgspec
  • Chainable queries: client.Contact.filter(is_deleted=False).limit(10).all()
  • Integrations for Django, Litestar, FastAPI, Flask, and others

Install

pip install civicrm-py

With a framework:

pip install civicrm-py[django]
pip install civicrm-py[litestar]
pip install civicrm-py[fastapi]

Optional extras:

pip install civicrm-py[sqlspec]    # Local database caching
pip install civicrm-py[workflows]  # Workflow automation

See docs for sqlspec, workflows, and pytest-databases.

Setup

export CIVI_BASE_URL=https://your-site.org/civicrm/ajax/api4
export CIVI_API_KEY=your-api-key
export CIVI_SITE_KEY=your-site-key

Usage

import asyncio
from civicrm_py import CiviClient

async def main():
    async with CiviClient() as client:
        # Get all active contacts
        contacts = await client.Contact.filter(is_deleted=False).all()

        # Single contact by ID
        contact = await client.Contact.get(id=123)

        # Create
        new_contact = await client.Contact.create(
            first_name="John",
            last_name="Doe",
            contact_type="Individual",
        )

        # Update
        await client.Contact.update(id=123, first_name="Jane")

        # Delete
        await client.Contact.delete(id=123)

asyncio.run(main())

Queries

# Select specific fields, filter, sort, paginate
contacts = await (
    client.Contact
    .select("id", "display_name", "email_primary.email")
    .filter(is_deleted=False, contact_type="Individual")
    .order_by("-created_date")
    .limit(25)
    .all()
)

# Operators
contacts = await (
    client.Contact
    .filter_by("created_date", ">=", "2024-01-01")
    .filter_by("display_name", "LIKE", "%Smith%")
    .all()
)

# Count and exists
total = await client.Contact.filter(is_deleted=False).count()
exists = await client.Contact.filter(id=123).exists()

Litestar

from litestar import Litestar, get
from civicrm_py import CiviClient
from civicrm_py.contrib.litestar import CiviPlugin

@get("/contacts")
async def get_contacts(civi: CiviClient) -> list[dict]:
    return await civi.Contact.filter(is_deleted=False).all()

app = Litestar(
    route_handlers=[get_contacts],
    plugins=[CiviPlugin()],
)

Django

# settings.py
CIVI_SETTINGS = {
    "BASE_URL": "https://example.org/civicrm/ajax/api4",
    "API_KEY": env("CIVI_API_KEY"),
    "SITE_KEY": env("CIVI_SITE_KEY"),
}

# views.py
from civicrm_py.contrib.django import get_civi_client

async def contact_list(request):
    async with get_civi_client() as client:
        contacts = await client.Contact.all()
        return JsonResponse({"contacts": contacts})

Development

git clone https://github.com/JacobCoffee/civicrm-py.git
cd civicrm-py

make install  # dependencies
make test     # tests
make check    # lint + typecheck

Docs

civi.scriptr.dev

Alternatives

  • civipy - Another Python client for CiviCRM. No built-in web framework integrations.

License

MIT

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

civicrm_py-0.1.1.tar.gz (211.1 kB view details)

Uploaded Source

Built Distribution

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

civicrm_py-0.1.1-py3-none-any.whl (261.2 kB view details)

Uploaded Python 3

File details

Details for the file civicrm_py-0.1.1.tar.gz.

File metadata

  • Download URL: civicrm_py-0.1.1.tar.gz
  • Upload date:
  • Size: 211.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for civicrm_py-0.1.1.tar.gz
Algorithm Hash digest
SHA256 730de88597538980714644c2588f54e1a4dec6be60d472c8b1a6029bacb66626
MD5 11ad199764e353fa641a4d919a209d67
BLAKE2b-256 271f5a18202e0c62277ca829d2da5cdb47205dd284db9c8c1366a52b8d416455

See more details on using hashes here.

Provenance

The following attestation bundles were made for civicrm_py-0.1.1.tar.gz:

Publisher: publish.yml on JacobCoffee/civicrm-py

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

File details

Details for the file civicrm_py-0.1.1-py3-none-any.whl.

File metadata

  • Download URL: civicrm_py-0.1.1-py3-none-any.whl
  • Upload date:
  • Size: 261.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for civicrm_py-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 39ea15803b34d51f1f2ee79c4206f442e41bc9c6e7c430d2ade35a8677b09962
MD5 6c7739b30a9aafec372aedd439625f90
BLAKE2b-256 27bfeb651ccf05b2c056031f39dd5acdc08c899e4f0b17f28e0e27e310975dab

See more details on using hashes here.

Provenance

The following attestation bundles were made for civicrm_py-0.1.1-py3-none-any.whl:

Publisher: publish.yml on JacobCoffee/civicrm-py

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