Reputation: 8467
tl;dr
Django throws an AttributeError
when I try to render admin class with ManyToMany
relation through custom form using fields
or fieldsets
attributes, apart from form
attribute:
AttributeError: 'ManyToManyField' object has no attribute 'through'
Models:
I have two models binded with ManyToMany
relation:
class Project(models.Model):
title = models.CharField(max_length=200, unique=True)
class Centre(models.Model):
projects = models.ManyToManyField(
Project, blank=True, related_name='centres',
)
Form:
Also, I've created a custom form for a Project
model to add FilteredSelectMultiple
field for Centre
:
class ProjectAdminForm(forms.ModelForm):
centres = forms.ModelMultipleChoiceField(
Centre.objects.all(),
widget=admin.widgets.FilteredSelectMultiple('Centres', False),
required=False,
)
def __init__(self, *args, **kwargs):
super(ProjectAdminForm, self).__init__(*args, **kwargs)
if self.instance.pk:
self.initial['centres'] = self.instance.centres.values_list(
'pk', flat=True
)
def save(self, *args, **kwargs):
instance = super(ProjectAdminForm, self).save(*args, **kwargs)
if instance.pk:
instance.centres.clear()
instance.centres.add(*self.cleaned_data['centres'])
return instance
Admin:
The thing is, when I define admin class like this:
@admin.register(Project, site=admin_site)
class ProjectAdmin(admin.ModelAdmin):
form = ProjectAdminForm
It works fine and renders a correct form for title
and centres
fields.
Problem
However, If I define fields as such:
@admin.register(Project, site=admin_site)
class ProjectAdmin(admin.ModelAdmin):
form = ProjectAdminForm
fields = ('title', 'centres', )
This throws an AttributeError
:
AttributeError: 'ManyToManyField' object has no attribute 'through'
I've tried setting through
attribute on models.ManyToManyField
in Project
model, defining intermediary ProjectCentresModel
, but I still can't understand what is going on inside Django and what should I tweak to make it work.
Traceback:
Unhandled exception in thread started by <function check_errors.<locals>.wrapper at 0x7f88ac6e5bf8>
Traceback (most recent call last):
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/utils/autoreload.py", line 226, in wrapper
fn(*args, **kwargs)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/core/management/commands/runserver.py", line 113, in inner_run
autoreload.raise_last_exception()
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/utils/autoreload.py", line 249, in raise_last_exception
six.reraise(*_exception)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/utils/six.py", line 685, in reraise
raise value.with_traceback(tb)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/utils/autoreload.py", line 226, in wrapper
fn(*args, **kwargs)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/__init__.py", line 27, in setup
apps.populate(settings.INSTALLED_APPS)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/apps/registry.py", line 115, in populate
app_config.ready()
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/apps.py", line 23, in ready
self.module.autodiscover()
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/__init__.py", line 26, in autodiscover
autodiscover_modules('admin', register_to=site)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/utils/module_loading.py", line 50, in autodiscover_modules
import_module('%s.%s' % (app_config.name, module_to_search))
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/importlib/__init__.py", line 126, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 986, in _gcd_import
File "<frozen importlib._bootstrap>", line 969, in _find_and_load
File "<frozen importlib._bootstrap>", line 958, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 673, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 665, in exec_module
File "<frozen importlib._bootstrap>", line 222, in _call_with_frames_removed
File "/media/kidbinary/Data/web/PYTHON/work_project/work/website/admin.py", line 165, in <module>
class ProjectAdmin(DefaultOrderingModelAdmin):
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/decorators.py", line 28, in _model_admin_wrapper
admin_site.register(models, admin_class=admin_class)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/sites.py", line 110, in register
system_check_errors.extend(admin_obj.check())
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/options.py", line 117, in check
return self.checks_class().check(self, **kwargs)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 515, in check
errors = super(ModelAdminChecks, self).check(admin_obj)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 71, in check
errors.extend(self._check_fieldsets(admin_obj))
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 155, in _check_fieldsets
for index, fieldset in enumerate(obj.fieldsets)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 155, in <listcomp>
for index, fieldset in enumerate(obj.fieldsets)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 190, in _check_fieldsets_item
for fieldset_fields in fieldset[1]['fields']
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 190, in <listcomp>
for fieldset_fields in fieldset[1]['fields']
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 204, in _check_field_spec
return self._check_field_spec_item(obj, model, fields, label)
File "/media/kidbinary/Data/web/PYTHON/.virtualenvs/work/lib/python3.5/site-packages/django/contrib/admin/checks.py", line 220, in _check_field_spec_item
if field.many_to_many and not field.remote_field.through._meta.auto_created:
AttributeError: 'ManyToManyField' object has no attribute 'through'
Upvotes: 2
Views: 2022
Reputation: 2157
class ProjectAdmin(admin.ModelAdmin):
model = Project
form = ProjectAdminForm
fields = ('title', 'centres',)
class CentreAdmin(admin.ModelAdmin):
model = Centre
list_display = ['id', 'get_projects']
def get_projects(self, obj):
#lists all the projects under a center without having to use inlines
return "\n".join([proj.title for proj in obj.projects.all()])
I honestly can't say why it was throwing that error, adding the model was just a guess work. I checked the source file of the error in django but couldn't really understand what was happening there.
Upvotes: 1