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:
- Status
- 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:
- Select plugin item
- Configure plugin-specific form
- Set instance metadata
- 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:
BasePluginTypeBasePluginItemPluginConfiguration
License
MIT License
Copyright (c) Alireza Tabatabaeian
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file django_plugin_system-2.0.0.tar.gz.
File metadata
- Download URL: django_plugin_system-2.0.0.tar.gz
- Upload date:
- Size: 24.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
50473eceba7ede373d9266b46830c830dfade12a1416339c3a4d27de742fa5cc
|
|
| MD5 |
1149cfa39e6764a9d20a6f18967a4a8a
|
|
| BLAKE2b-256 |
607828e3fbb80858d763623b8d93cd435e422b8adfe3e1018db1d8ce75a52393
|
File details
Details for the file django_plugin_system-2.0.0-py3-none-any.whl.
File metadata
- Download URL: django_plugin_system-2.0.0-py3-none-any.whl
- Upload date:
- Size: 22.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
cdbf5a7107e944cb7efd9eb5592d3546cba0439678e9acaaed3999ab526327c3
|
|
| MD5 |
4488e087378fe696403167eabfc03929
|
|
| BLAKE2b-256 |
6789479e7b102f51c84989fff8e38a7e57853422f1c55110995c2655d101d037
|