Skip to main content

Multi-backend caching system for Lexigram Framework - Redis, Memcached, and in-memory caching

Project description

lexigram-cache

Multi-backend caching system for Lexigram Framework — Redis, Memcached, and in-memory caching.


Overview

Multi-backend async caching for the Lexigram Framework. Supports Redis, in-memory, and Memcached backends with stampede protection, circuit breaker, Result-aware caching, and domain model round-trip serialization.

Configure backends via CacheModule.configure() and inject CacheBackendProtocol into any service. The @cacheable decorator provides cache-aside logic with automatic key generation and Result support.

Install

uv add lexigram-cache
# Optional extras
uv add "lexigram-cache[redis,memcached,semantic]"

Quick Start

from lexigram import Application
from lexigram.di.module import Module, module
from lexigram.cache import CacheModule, CacheConfig
from lexigram.cache.config import CacheBackendConfig
from lexigram.cache.types import BackendType
from lexigram.contracts.infra.cache import CacheBackendProtocol

@module(imports=[
    CacheModule.configure(
        CacheConfig(
            backends=[
                CacheBackendConfig(
                    name="default",
                    type=BackendType.REDIS,
                    default=True,
                    redis_url="redis://localhost:6379/0",
                )
            ]
        )
    )
])
class AppModule(Module):
    pass

async def main():
    async with Application.boot(modules=[AppModule]) as app:
        cache = await app.container.resolve(CacheBackendProtocol)
        await cache.set("greeting", "hello", ttl=60)
        value = await cache.get("greeting")
        print(value)  # "hello"

if __name__ == "__main__":
    import asyncio
    asyncio.run(main())

Configuration

Zero-config usage: Call CacheModule.configure() with no arguments to use defaults (in-memory backend).

Option 1 — YAML file

# application.yaml
cache:
  enabled: true
  backends:
    - name: redis
      type: redis
      default: true
      redis_url: "redis://localhost:6379/0"
      default_ttl: 300
  service:
    enable_protection: true

Option 2 — Profiles + Environment Variables (recommended)

export LEX_CACHE__ENABLED=true
export LEX_CACHE__BACKENDS__0__TYPE=redis
export LEX_CACHE__BACKENDS__0__REDIS_URL=redis://localhost:6379/0

Option 3 — Python

from lexigram.cache import CacheModule, CacheConfig
from lexigram.cache.config import CacheBackendConfig
from lexigram.cache.types import BackendType

CacheModule.configure(
    CacheConfig(
        backends=[
            CacheBackendConfig(
                name="redis",
                type=BackendType.REDIS,
                default=True,
                redis_url="redis://localhost:6379/0",
            )
        ]
    )
)

Config reference

Field Default Env var Description
enabled True LEX_CACHE__ENABLED Enable the cache module
backends[].name LEX_CACHE__BACKENDS__0__NAME Unique backend name
backends[].type LEX_CACHE__BACKENDS__0__TYPE Backend type: redis, memory, memcached
backends[].redis_url LEX_CACHE__BACKENDS__0__REDIS_URL Redis connection URL
backends[].default_ttl null LEX_CACHE__BACKENDS__0__DEFAULT_TTL Default TTL in seconds
service.enable_protection True LEX_CACHE__SERVICE__ENABLE_PROTECTION Stampede protection
service.circuit_breaker_enabled False LEX_CACHE__SERVICE__CIRCUIT_BREAKER_ENABLED Circuit breaker on backend failures

Module Factory Methods

Method Description
CacheModule.configure(...) Configure with explicit CacheConfig
CacheModule.stub() In-memory backend for unit testing

Key Features

  • Multi-backend — Redis, in-memory, and Memcached with a unified protocol
  • Stampede protection — Distributed lock on cold reads prevents thundering herd
  • Circuit breaker — Opens on repeated backend failures, falls through to origin
  • @cacheable decorator — Cache-aside with automatic key generation and Result support
  • Domain model serialization — Type-tagged JSON envelope preserves type identity round-trip
  • Production security — Blocks insecure Redis passwords when LEX_ENV=production

Testing

async with Application.boot(modules=[CacheModule.stub()]) as app:
    cache = await app.container.resolve(CacheBackendProtocol)
    await cache.set("key", "value", ttl=60)
    assert await cache.get("key") == "value"

Key Source Files

File What it contains
src/lexigram/cache/module.py CacheModule with configure() and stub()
src/lexigram/cache/config.py CacheConfig, CacheBackendConfig, CacheServiceConfig
src/lexigram/cache/di/provider.py CacheProvider boot and registration
src/lexigram/cache/decorators.py @cacheable decorator
src/lexigram/cache/backends/ Redis, in-memory, and Memcached implementations

Project details


Download files

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

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distribution

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

lexigram_cache-0.1.1-py3-none-any.whl (133.2 kB view details)

Uploaded Python 3

File details

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

File metadata

File hashes

Hashes for lexigram_cache-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 6cd6e1fd690fd927e9fcf35150d845ae117707bf491920866c63962c0ad6adbc
MD5 291e75367b6b70c91b7c4fd0b80e2c6f
BLAKE2b-256 d2acd35f90e4c46d94412ad7d4e3753ff18ba7b5fcf443b014f0dd7192d8dba1

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