Skip to main content

A flexible plugin framework for Django with admin-managed plugin instances and registry-to-database sync.

Project description

Django Plugin System

A flexible plugin framework for Django that allows applications to expose extension points through Plugin Types, implement them through Plugin Items, and configure multiple runtime Plugin Instances through Django Admin.


Why use Django Plugin System?

As projects grow, it becomes increasingly common to support multiple implementations of the same functionality:

  • Multiple payment gateways
  • Different OTP providers
  • Various notification channels
  • AI providers from different vendors
  • Storage backends
  • Shipping providers
  • Authentication mechanisms

Without a plugin system, these integrations are often hardcoded into the application, making them difficult to replace, configure, or extend.

Django Plugin System provides a structured way to:

  • Define clear extension points through abstract interfaces
  • Register multiple independent implementations
  • Configure plugin instances directly from Django Admin
  • Switch providers without changing application code
  • Prioritize and enable/disable plugins at runtime
  • Allow third-party Django apps to extend your system cleanly

Instead of writing large chains of conditional logic:

if gateway == "zarinpal":
    ...
elif gateway == "stripe":
    ...
elif gateway == "paypal":
    ...

your application simply requests a plugin:

gateway = get_plugin_instance(PaymentGatewayType)
gateway.create_payment(amount)

The selected implementation is determined by configuration, priority, and availability—not by hardcoded business logic.

This approach keeps applications modular, maintainable, and open for extension without requiring modifications to the core codebase.

Django Plugin System helps you build extensible Django applications by separating what a service should do (Plugin Types) from how it is implemented (Plugin Items) and how it is configured (Plugin Instances).


Features

  • Interface-first plugin architecture
  • Multiple implementations per plugin type
  • Multiple configured instances per implementation
  • Django Admin integration
  • Configurable plugins using Django Forms
  • Registry-to-database synchronization
  • Automatic validation of plugin contracts
  • Priority-based plugin selection
  • Safe plugin discovery through AppConfig.ready()
  • Runtime plugin loading helpers

Installation

pip install django-plugin-system

Add the application:

INSTALLED_APPS = [
    ...
    "django_plugin_system",
]

Run migrations:

python manage.py migrate

Core Concepts

The system is built around three layers:

Plugin Type
    ↓
Plugin Item
    ↓
Plugin Instance

Plugin Type

Defines a contract (interface).

from abc import abstractmethod
from django_plugin_system.plugin_core import BasePluginType


class PaymentGatewayType(BasePluginType):
    name = "payment_gateway"

    @abstractmethod
    def create_payment(self, amount):
        pass

Plugin Item

Provides an implementation for a plugin type.

from django_plugin_system.plugin_core import BasePluginItem


class ZarinpalGateway(BasePluginItem):
    name = "zarinpal"
    plugin_type = PaymentGatewayType

    def create_payment(self, amount):
        ...

Plugin Instance

Represents a configured runtime instance of a plugin item.

Examples:

Zarinpal Production
Zarinpal Sandbox
Zarinpal Backup

All may use the same plugin item but different configuration values.


Typical Flow

Create Plugin Type > Create Plugin Item > Register Plugins > Run pluginsync > Create Plugin Instances > Load Plugin At Runtime


Registering Plugins

Registration should usually happen inside AppConfig.ready().

from django.apps import AppConfig

from django_plugin_system.register import (
    register_plugin_type,
    register_plugin_item,
)


class PaymentsConfig(AppConfig):
    name = "payments"

    def ready(self):
        register_plugin_type({
            "interface": PaymentGatewayType,
        })

        register_plugin_item({
            "plugin_class": ZarinpalGateway,
        })

Configurable Plugins

Plugins can expose configuration forms through a PluginConfiguration class.

from django import forms

from django_plugin_system.plugin_core import (
    PluginConfiguration,
)


class ZarinpalConfigForm(forms.Form):
    merchant_id = forms.CharField()


class ZarinpalConfiguration(PluginConfiguration):
    form_class = ZarinpalConfigForm

Attach it to a plugin item:

class ZarinpalGateway(BasePluginItem):
    name = "zarinpal"
    plugin_type = PaymentGatewayType
    configuration = ZarinpalConfiguration

    def create_payment(self, amount):
        merchant_id = self.config["merchant_id"]
        ...

Administrators can then create and manage configured plugin instances directly from Django Admin.

Accessing Configuration Values

When a plugin instance is loaded, its configuration is automatically made available through the config property.

For example, suppose an administrator creates:

Plugin Instance:
    Name: Zarinpal Production

Configuration:
    merchant_id = abc123

Inside the plugin:

def create_payment(self, amount):
    merchant_id = self.config["merchant_id"]

    print(merchant_id)

Output:

abc123

self.config is a standard Python dictionary, so the following patterns are also valid:

self.config["merchant_id"]
self.config.get("merchant_id")
self.config.get("sandbox", False)

Encryption Configuration

Django Plugin System stores plugin instance configuration using encrypted database fields.

Before using configurable plugins, make sure the required encryption settings are configured.

# settings.py

FIELD_ENCRYPTION_KEY = "your-field-encryption-key"
SALT_KEY = "your-salt-key"

For production environments, these values should be generated securely and stored outside source control (for example using environment variables).

import os

FIELD_ENCRYPTION_KEY = os.environ["FIELD_ENCRYPTION_KEY"]
SALT_KEY = os.environ["SALT_KEY"]

For more information about key generation and encryption settings, please refer to the documentation of the underlying encryption package used by Django Plugin System.


Synchronizing Plugins

After registering plugins, synchronize them into the database:

python manage.py pluginsync

Or:

python manage.py pluginsync --prune

The sync process:

  • Creates missing plugin types
  • Creates missing plugin items
  • Creates default instances for non-configurable plugins
  • Optionally removes stale records

Loading Plugins

Retrieve the selected plugin instance:

from django_plugin_system.helpers import get_plugin_instance

gateway = get_plugin_instance(
    PaymentGatewayType
)

gateway.create_payment(1000)

Load a specific instance:

gateway = get_plugin_instance_by_id(
    PaymentGatewayType,
    plugin_id=instance_id,
)

Load all active instances

In some scenarios you may want to present all available plugins to a user and let them choose which one to use, or maybe try instances on by one until the result is successful.

instances : List[PluginInstance] = get_active_plugins(
    PaymentGatewayType
)

for instance in instances:
    print(instance.name)

Each returned object is a PluginInstance.

To load the actual plugin implementation:

selected_instance = instances[0]

gateway = selected_instance.load_instance() # loads actual item with methods, like create_payment
gateway.create_payment(1000)

Reserve plugins can be retrieved using:

reserve_instances = get_reserve_plugins(
    PaymentGatewayType
)

This can be useful when implementing custom fallback or failover logic.


Plugin Selection

By default, each plugin type selects a single plugin instance according to:

  1. Status
  2. Priority

Selection order:

ACTIVE
    ↓
RESERVED
    ↓
DISABLED

Within the same status, lower priority values are preferred.

For example:

Instance Status Priority
Stripe Production ACTIVE 10
Stripe Backup ACTIVE 20
Stripe Sandbox RESERVED 1

The selected plugin would be: Strip Production

because ACTIVE instances are preferred over RESERVED instances regardless of priority.

Customizing Plugin Selection

The default selection strategy works well for most applications, but some projects may require custom logic.

Examples:

  • Selecting plugins based on geographic region
  • Selecting plugins based on user preferences
  • Load balancing between multiple providers
  • Implementing A/B testing
  • Selecting plugins according to business rules

Provide a custom get_plugin function when registering the plugin type.

def custom_get_plugin(plugin_type):
    return(
            PluginInstance.objects
            .select_related("item", "item__plugin_type")
            .filter(item__plugin_type=plugin_type, status=PluginStatus.Active)
            .order_by('name')
            .first()
        )

Register the type:

register_plugin_type({
    "interface": TestPluginType,
    "get_plugin": custom_get_plugin
})

The function must return either:

  • a PluginInstance
  • or None

Django Admin

The admin interface allows:

  • Creating plugin instances
  • Editing plugin configuration
  • Enabling/disabling instances
  • Changing priorities
  • Verifying plugin loading

For configurable plugins:

  1. Select plugin item
  2. Configure plugin-specific form
  3. Set instance metadata
  4. Save

Validation

The framework validates registrations during startup.

Examples:

  • Plugin types must define a name.
  • Plugin types must contain abstract methods.
  • Plugin items must implement required methods.
  • Configuration classes must inherit from PluginConfiguration.
  • Configuration forms must inherit from Django Form.

Invalid plugins fail fast during application startup.


Legacy Support

Older registration APIs remain supported for backward compatibility, but new projects should use:

  • BasePluginType
  • BasePluginItem
  • PluginConfiguration

License

MIT License

Copyright (c) Alireza Tabatabaeian

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

django_plugin_system-2.0.1.tar.gz (24.3 kB view details)

Uploaded Source

Built Distribution

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

django_plugin_system-2.0.1-py3-none-any.whl (22.4 kB view details)

Uploaded Python 3

File details

Details for the file django_plugin_system-2.0.1.tar.gz.

File metadata

  • Download URL: django_plugin_system-2.0.1.tar.gz
  • Upload date:
  • Size: 24.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.5

File hashes

Hashes for django_plugin_system-2.0.1.tar.gz
Algorithm Hash digest
SHA256 a028d617e5277268400db40ca3b6c9cf8bbaa8ffb5a4caab931c4769b4e6fb27
MD5 c77858db501f09f49998782ebdf6e1dc
BLAKE2b-256 f61c55c5d4cb4e3f5bc9b0f760e3a636bc619c0ed8421a8f4fb0851af9780be3

See more details on using hashes here.

File details

Details for the file django_plugin_system-2.0.1-py3-none-any.whl.

File metadata

File hashes

Hashes for django_plugin_system-2.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 a49b4f2eadff7f77dead6910f8d71f34acec1cb1cda7293e7f6ce3d0e7ca3b78
MD5 7ed05f8f924565b69dcc32073e71c349
BLAKE2b-256 76dc85ec187deadf631a496f872de92ee06612b35a46a8e02c9a5a6005603f8c

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