Skip to main content

Extandable field that enables Django apps to store their data on your models.

Project description

Extandable field and related tools that enable Django apps to extend your reusable app.

IMPORTANT NOTE

This app is fork of the original Django appdata. Fork was created because of original application does not work correctly with the latest version of Django and reactions on pull requests are very slow. However I recommend use the original application if you can.

Motivation

When working with reusable django apps we often found that we needed to add something extra to the model or form the app provided. Some apps try to solve this by providing a flexible model definition and a pluggable form (see django.contrib.comments for an exmple of this approach) but even then it leads to some duplication of efforts.

django-appdata app tries, through AppDataField, MultiForm and AppDataModelAdmin, to provide a standardised approach to extending existing apps.

Extending Models

When you have an extendable django app using the AppDataField:

from django.db import models
from app_data import AppDataField

class BlogPost(models.Model):
    text = models.TextField()
    app_data = AppDataField()

your code can register a namespace on any (or all) AppDataField and store it’s own data there by registering a container (subclass of AppDataContainer). To define the data you use django’s form framework:

from django.forms.models import ModelMultipleChoiceField
from app_data import app_registry, AppDataForm, AppDataContainer

from .models import Tag

class TaggingAppDataForm(AppDataForm):
    public_tags = ModelMultipleChoiceField(Tag.objects.all())
    admin_tags = ModelMultipleChoiceField(Tag.objects.all())

class TaggingAppDataContainer(AppDataContainer):
    form_class = TaggingAppDataForm

    def tag_string(self):
        print ', '.join(t.name for t in self.public_tags)

app_registry.register('tagging', TaggingAppDataContainer)

This should give you access to ‘tagging’ namespace in any defined AppDataField:

from blog_app.models import BlogPost

bp = BlogPost()
assert bp.app_data.tagging.tag_string() == ""

Additional Options

Note that if you don’t need to add custom methods to your container you can just use a factory to create the subclass:

app_registry.register('tagging', AppDataContainer.from_form(TaggingAppDataForm))

Additionaly you can restrict the registration to a given model:

from blog_app.models import BlogPost

app_registry.register('tagging', TaggingAppDataContainer, BlogPost)

Extending Forms

django-appdata supplies a MultiForm class - a wrapper around django’s ModelForm with optional added sub-forms that corresponds to namespaces registered in the model’s AppDataField, typically the extendable app would create and use a MultiForm instead of a regular ModelForm:

from app_data.forms import multiform_factory
from .models import BlogPost

BlogPostMultiForm = multiform_factory(BlogPost)

And when using that app any project can add additional sub-forms to that MultiForm:

from blog_app.forms import BlogPostMultiForm

BlogPostMultiForm.add_form('tagging', {'fields': ['public_tags']})

This way when the reusable app’s code can remain unchanged and we can inject additional form logic to its processing.

Additional Options

Any arguments and keyword arguments are passed without change to the ModelForm class the MultiForm is wrapping so even if you have custom args for your ModelForm everything will still work:

from django.forms.models import BaseModelForm

class ModelFormWithUser(ModelForm):
    def __init__(self, user, *args, **kwargs):
        self.user = user
        super(ModelFormWithUser, self).__init__(*args, **kwargs)

BlogPostMultiForm = multiform_factory(BlogPost, form=ModelFormWithUser)

And of course you are not limited to the use of a factory function:

from app_data import MultiForm

class MyMultiForm(MultiForm):
    ModelForm = BlogPostModelForm

MultiForms in Admin

If you wish to add your own code to the admin interface, just use AppDataModelAdmin:

from django.contrib import admin
from app_data.admin import AppDataModelAdmin
from blog_app.models import BlogPost

class BlogPostAdmin(AppDataModelAdmin):
    # due to bug in django's admin validation we need to use
    # declared_fieldsets instead of just fieldsets
    declared_fieldsets = [
        (None, {'fields': ['text', ]}),
        ('Tagging', {'fields': [('tagging.public_tags', 'tagging.admin_tags')]})
    ]
admin.site.register(BlogPost, BlogPostAdmin)

Additional Options

As with django’s admin and forms you can supply your own MultiForm class by using the multiform attribute of AppDataModelAdmin.

Behind the scenes

django-appdata uses a TextField to store the data on the model using JSON and django’s forms framework for (de)serialization and validation of the data.

When accessing the containers in the field we will try to locate the appropriate container in the registry. If none is found, plain data will be returned if present (dict). To assure everything working properly we recommend putting some sort of init code in place for your project that will make sure all the registration is done before any actual code is run. We are using a module called register in our apps and then a piece of code similar to admin’s autodiscover to iterate through installed apps and load this module.

Build status

Master branch:
Travis CI - Distributed build platform for the open source community

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-appdata-mam-0.1.7.tar.gz (18.5 kB view details)

Uploaded Source

Built Distribution

django_appdata_mam-0.1.7-py2.py3-none-any.whl (15.2 kB view details)

Uploaded Python 2 Python 3

File details

Details for the file django-appdata-mam-0.1.7.tar.gz.

File metadata

File hashes

Hashes for django-appdata-mam-0.1.7.tar.gz
Algorithm Hash digest
SHA256 f0d83c9a71208b351324e28a520518c9f96810f82d0700c2f3fc559aa0daa109
MD5 667f5263346da54da3927c660369f6c6
BLAKE2b-256 9d57e83ecc144691a788b051b6e68fbf6d61775db41a7075b663a4c0853ef45a

See more details on using hashes here.

File details

Details for the file django_appdata_mam-0.1.7-py2.py3-none-any.whl.

File metadata

File hashes

Hashes for django_appdata_mam-0.1.7-py2.py3-none-any.whl
Algorithm Hash digest
SHA256 f3d66e8863006ff1db2a4390774aa5049797265d0ceabe6444e860113a7860e8
MD5 b9e08a9f6adb8a9c5329e9f308386c40
BLAKE2b-256 3b45abdbe20ff0bc1f7de54ddedbf60ee8ef49734aca0c0d4b79b54494cbf6cb

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page