Skip to main content

OpenFrame Microservice Suite — MongoDB document store adapter.

Project description

openframe-adapters-db-mongo

MongoDB document store adapter for the OpenFrame Microservice Suite.

Part of the openframe-adapters monorepo. Implements BaseRepository[T] and HealthCheck from openframe-core using motor (AsyncIOMotorClient).


Installation

pip install openframe-adapters-db-mongo

Required env vars:

MONGO_URL=mongodb://user:password@host:27017
MONGO_DATABASE=mydb

Atlas SRV URIs work too:

MONGO_URL=mongodb+srv://user:password@cluster.mongodb.net

Quick start

Raw dict mode

from openframe.adapters.db.mongo import MongoSettings, MongoRepository

settings = MongoSettings()  # reads MONGO_URL and MONGO_DATABASE from env
repo = MongoRepository(settings, collection="artifacts")

doc = await repo.get("507f1f77bcf86cd799439011")   # dict | None
docs, total = await repo.list(10, 0)                # ([dict, ...], int)
created = await repo.create({"title": "paper"})
updated = await repo.update({"_id": "...", "title": "updated"})
deleted = await repo.delete("507f1f77bcf86cd799439011")  # bool

Typed domain mode

from dataclasses import dataclass
from openframe.adapters.db.mongo import MongoSettings, MongoRepository

@dataclass
class Artifact:
    id: str
    title: str
    artifact_type: str

class ArtifactRepository(MongoRepository[Artifact]):
    _collection = "artifacts"

    def _doc_to_entity(self, doc: dict) -> Artifact:
        return Artifact(
            id=doc["_id"],
            title=doc["title"],
            artifact_type=doc["artifact_type"],
        )

    def _entity_to_doc(self, entity: Artifact) -> dict:
        return {
            "_id": entity.id,
            "title": entity.title,
            "artifact_type": entity.artifact_type,
        }

settings = MongoSettings()
repo = ArtifactRepository(settings)
artifact: Artifact | None = await repo.get("507f1f77bcf86cd799439011")

_id handling

MongoDB uses _id as the document identifier; BaseRepository uses entity_id: str. The adapter bridges this transparently:

  • Reading: _id is always returned as strObjectId is never exposed to callers. A convenience id key is also added mirroring _id.
  • Writing: If an entity dict has an id key but no _id key, the adapter maps id_id before insertion.
  • Filtering: String entity IDs that are valid 24-char hex are parsed as ObjectId for indexed lookups; other strings are used as plain string _id values.

Configuration

All settings are read from environment variables.

Env var Type Default Description
MONGO_URL str required Motor/pymongo connection string
MONGO_DATABASE str required Database name
MONGO_MIN_POOL_SIZE int 5 Minimum pool connections
MONGO_MAX_POOL_SIZE int 20 Maximum pool connections
MONGO_SERVER_SELECTION_TIMEOUT_MS int 5000 Server selection timeout (ms)
MONGO_TLS bool False Enable TLS/SSL
MONGO_TLS_ALLOW_INVALID_CERTS bool False Skip cert validation (dev only)
CONNECTION_TIMEOUT float 30.0 Pool creation timeout (s)
OPERATION_TIMEOUT float 10.0 Per-operation timeout (s)
MAX_RETRIES int 3 Max retry attempts

Timeout strategy

Two layers of timeout on every operation:

  1. asyncio.timeout(settings.operation_timeout) — cancels the Python coroutine.
  2. max_time_ms=int(settings.operation_timeout * 1000) passed to motor query methods — instructs the MongoDB server to abort the query.

Health checks

MongoRepository implements the HealthCheck protocol from openframe-core.

alive = await repo.ping()       # admin.command("ping") — fast liveness check
ready = await repo.is_ready()   # list_collection_names() — full readiness check

Both methods return False on any failure and never raise.


Exception hierarchy

All exceptions are AdapterError subclasses from openframe.core.exceptions. Raw motor/pymongo exceptions never escape the adapter.

Situation Exception
Cannot connect to MongoDB AdapterConnectionError
Invalid MONGO_URL syntax AdapterConfigurationError
Query failed (constraint, auth, etc.) AdapterQueryError
Operation exceeded timeout AdapterTimeoutError

Development

# from the package directory
pip install -e ".[dev]"
pytest tests/ -v

Protocol conformance

from openframe.core.ports import BaseRepository
from openframe.core.health import HealthCheck

repo = MongoRepository(settings, collection="artifacts")
assert isinstance(repo, BaseRepository)   # True — structural check
assert isinstance(repo, HealthCheck)      # True — structural check

No inheritance from either Protocol is required or used.


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

openframe_adapters_db_mongo-1.1.0.tar.gz (18.5 kB view details)

Uploaded Source

Built Distribution

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

openframe_adapters_db_mongo-1.1.0-py3-none-any.whl (13.0 kB view details)

Uploaded Python 3

File details

Details for the file openframe_adapters_db_mongo-1.1.0.tar.gz.

File metadata

File hashes

Hashes for openframe_adapters_db_mongo-1.1.0.tar.gz
Algorithm Hash digest
SHA256 0ed52714dd147de73112e72e27f7e9ef99268877ff35cda09771a2c18a9d38d2
MD5 6e9e3a782973fcb4433f0860d2ea1d29
BLAKE2b-256 95375ac1244205d94acfae49961f1f123ac9ed605bfb9d7549a6b2cad11f7687

See more details on using hashes here.

File details

Details for the file openframe_adapters_db_mongo-1.1.0-py3-none-any.whl.

File metadata

File hashes

Hashes for openframe_adapters_db_mongo-1.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 afe2a7892ed0be06d51b6fd00301df8a46c8fc53fdf23346b23b32bccbe8e30a
MD5 067cfcfdeab270ad83ad68c0a87c4181
BLAKE2b-256 864ee1af42ca02eccfd860421acf506f7966ffd5dde59109821e8314a660dbd3

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