Skip to main content

Cullinan — A pluggable IoC/DI web framework

Project description

Python version PyPI version PyPI downloads Ask DeepWiki GitHub stars License

   _____      _ _ _
  / ____|    | | (_)
 | |    _   _| | |_ _ __   __ _ _ __
 | |   | | | | | | | '_ \ / _` | '_ \
 | |___| |_| | | | | | | | (_| | | | |
  \_____\__,_|_|_|_|_| |_|\__,_|_| |_|

Cullinan

A business-first Python web framework for building applications through decorators, module boundaries, and built-in IoC/DI.

Cullinan is a Python Web Framework for developers who want to organize applications around business services, controllers, and methods instead of manually wiring an app object. Its default path is decorator-first discovery plus configure(root_module=...) and run(), with @module used as the application boundary when ownership, runtime structure, and stability matter.


What is Cullinan?

Cullinan is designed to make the framework work around your business architecture, not the other way around.

  • Business-first application model - write services, controllers, middleware, and methods first
  • Decorator-first discovery - let Python imports and decorator metadata assemble the runtime
  • Structured module boundaries - use @module to express ownership and runtime boundaries
  • Built-in IoC/DI and lifecycle - keep application wiring inside the framework model
  • Public semantic path - build, run, test, and extend applications through stable public APIs

Cullinan is not centered on treating an app object as a manual registration hub. The recommended development flow is to declare business components, define a root module when structure matters, and let the framework assemble the application runtime.


Why Cullinan?

1. Framework semantics before manual wiring

Cullinan gives you a clear application model: declare components with decorators, let the framework discover them, and keep explicit runtime internals as an advanced path rather than the default onboarding path.

2. Application boundaries that scale past toy examples

@module is more than a naming convention. It is the structured boundary for owned packages, runtime composition, and higher-level lifecycle behavior when your application grows.

3. Web, DI, lifecycle, and testing in one model

Routing, parameter binding, dependency injection, lifecycle hooks, middleware, and test flow all sit inside one framework vocabulary instead of forcing developers to stitch together multiple unrelated patterns.

4. A Pythonic path for business applications

Cullinan is designed to keep developers focused on business architecture and business methods. You can stay on the public path for most application work and only move into internals when you intentionally need advanced extension behavior.


Framework capabilities

Capability layer What Cullinan provides
Application composition configure(root_module=...), decorator-driven discovery, structured runtime assembly
Web API model @controller, RESTful decorators, WebResponse, middleware pipeline, WebSocket support
Parameter system Typed Path, Query, Body, validation, conversion, and controller-method binding
IoC/DI and lifecycle Inject(), InjectByName(), request scope, startup/shutdown hooks, component lifecycle
Runtime boundaries @module ownership, clearer package structure, better runtime organization
Testing and delivery get_asgi_app(), resettable registries, packaging-friendly runtime, cross-platform support

Cullinan's goal is not to expose every internal knob on the README homepage. The goal is to provide a coherent framework model that remains readable from first contact through production use.


Install

Cullinan is published on PyPI and currently supports Python 3.7+.

pip install -U pip
pip install cullinan

After installation, start from the minimal example in examples/minimal_app/ or the repository guide in docs/examples.md.


Quick Start

The recommended starting point is: define business components, declare the root boundary, configure the app, then run it.

from cullinan import Inject, configure, controller, get_api, module, run, service


@service
class GreetingService:
    def greet(self) -> str:
        return "Hello from Cullinan!"


@controller(url="/hello")
class HelloController:
    greeting_service: "GreetingService" = Inject()

    @get_api(url="")
    def hello(self):
        return {"message": self.greeting_service.greet()}


@module
class RootModule:
    """Declare the application boundary when structure matters."""


def create_app():
    return configure(root_module=RootModule)


if __name__ == "__main__":
    create_app()
    run()

Run it:

python minimal_app.py

Then open:

http://localhost:4080/hello

This example shows the default Cullinan path:

  • business service first
  • controller methods as the public web surface
  • Inject() for framework-managed wiring
  • @module as the runtime boundary
  • configure(root_module=...) plus run() as the startup flow

Learning path

If you are new to Cullinan, use this order:

  1. examples/minimal_app/ - shortest public entrypoint
  2. examples/controller_service_inject/ - service/controller layering with Inject()
  3. examples/middleware_and_module/ - @module boundaries and middleware semantics
  4. examples/parameter_handling/ - method-level Path, Query, and Body
  5. examples/testing_flow/ - public-API testing with get_asgi_app()

Then move into the documentation knowledge base:


Documentation

English

中文


Current series

v0.93a6.post1 focuses the public application path around:

  • configure(root_module=...) and run()
  • decorator-first discovery through Python imports
  • @module as the structured boundary for runtime ownership
  • built-in IoC/DI, lifecycle, and semantic diagnostics
  • unified example and documentation guidance around the current public API
  • consistency fixes for high-level public runtime entrypoints

Version-specific details and migration notes live in the documentation knowledge base rather than in the homepage narrative.


Project links


License

MIT License - see LICENSE for details.

Project details


Release history Release notifications | RSS feed

Download files

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

Source Distribution

cullinan-0.93a6.post1.tar.gz (206.2 kB view details)

Uploaded Source

Built Distribution

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

cullinan-0.93a6.post1-py3-none-any.whl (257.1 kB view details)

Uploaded Python 3

File details

Details for the file cullinan-0.93a6.post1.tar.gz.

File metadata

  • Download URL: cullinan-0.93a6.post1.tar.gz
  • Upload date:
  • Size: 206.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.5

File hashes

Hashes for cullinan-0.93a6.post1.tar.gz
Algorithm Hash digest
SHA256 070a01848919495f1a96d25d73fdd1ed8267a3cb142a002876c89d0ed0bc83d0
MD5 e18494177df64036fea38166306a7a34
BLAKE2b-256 e474f3359ee7cb8921080b61cf3bb2fe7d4f7d8210021002880cdd95373a34cb

See more details on using hashes here.

File details

Details for the file cullinan-0.93a6.post1-py3-none-any.whl.

File metadata

File hashes

Hashes for cullinan-0.93a6.post1-py3-none-any.whl
Algorithm Hash digest
SHA256 847ad74e9ed964ce21f1876a88b45bc26a115fefd1635e3489902324892da4e0
MD5 79e05ce4933f14e477c7b52f3f062cd1
BLAKE2b-256 a986708a58e446351b17ce6c47aa1ba3b6c0fcde360ec000d7bb6c75a729cd2a

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