Opened 7 years ago
Last modified 16 months ago
#28438 new Bug
Initial migration creates fields not listed in the migration if mixin class changes
Reported by: | Michal Dabski | Owned by: | nobody |
---|---|---|---|
Component: | Migrations | Version: | 1.11 |
Severity: | Normal | Keywords: | migration, models, mixin, postgresql |
Cc: | Triage Stage: | Accepted | |
Has patch: | no | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description (last modified by )
Consider the sample model with a mixin class:
from django.db import models class TestMixin(object): pass class TestModel(TestMixin, models.Model): sample_field = models.CharField(max_length=20)
When running makemigrations, django creates a perfectly good and valid migration:
# -*- coding: utf-8 -*- # Generated by Django 1.11.3 on 2017-07-26 17:03 from __future__ import unicode_literals from django.db import migrations, models import migration_test.models class Migration(migrations.Migration): initial = True dependencies = [ ] operations = [ migrations.CreateModel( name='TestModel', fields=[ ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), ('sample_field', models.CharField(max_length=20)), ], bases=(migration_test.models.TestMixin, models.Model), ), ]
SQL:
BEGIN; -- -- Create model TestModel -- CREATE TABLE "migration_test_testmodel" ("id" serial NOT NULL PRIMARY KEY, "sample_field" varchar(20) NOT NULL); COMMIT;
Next, refactor mixin class to add a field to it - need to change mixin's base class to models.Model
, otherwise field will not be correctly inherited by models:
class TestMixin(models.Model): mixin_field = models.CharField(max_length=20, default='test') class Meta: abstract = True
This creates a new migration which adds mixin_field
to it - nothing special. However, when applying both migrations after the model changes, second migration fails with the following error django.db.utils.ProgrammingError: column "mixin_field" of relation "migration_test_testmodel" already exists
. as it turns out, the first migration's SQL has now changed to include mixin_field
:
BEGIN; -- -- Create model TestModel -- CREATE TABLE "migration_test_testmodel" ("id" serial NOT NULL PRIMARY KEY, "mixin_field" varchar(20) NOT NULL, "sample_field" varchar(20) NOT NULL); COMMIT;
The python code of the migration obviously has not changed, but the resulting SQL did, and it includes a field not explicitly listed in the migration's fields
.
Note:
- this does not happen if the mixin class extends
models.Model
to begin with. - if model extends a mixin that extends
object
, it ends up in model'sbases
, however if mixin extendsmodel.Model
it does not. - when mixin's base class changes, schema migration does not reflect this change in model's
bases
Tested with Django 1.11.3, Python 2.7
Proposed solution:
migration should not create database fields for model fields not explicitly listed in fields
Change History (11)
comment:1 by , 7 years ago
Description: | modified (diff) |
---|
comment:2 by , 7 years ago
Keywords: | migration models mixin sql added |
---|
comment:3 by , 7 years ago
Description: | modified (diff) |
---|
comment:4 by , 7 years ago
Component: | Uncategorized → Migrations |
---|---|
Type: | Uncategorized → Bug |
comment:5 by , 7 years ago
Keywords: | postgresql added; sql removed |
---|
comment:6 by , 7 years ago
At this point my code has been deployed to production with mixin inheriting from object. I use mixin to reuse code across two models that have separete hierarchy trees, so common base class was not an option.
At the moment my best workaround is to keep the mixin as-is, and add a field to each subclass as needed, but that violates separation of concerns principle and DRY.
I have seen mixin pattern being used in models in Django itself, so I think it should be OK to use mixins. For example, PermissionMixin
class. The only difference is that my mixin used to inherit from object
originally, which was recorded by the initial migration.
comment:7 by , 7 years ago
Ah, I see, that PermissionMixin
mixin is made using an abstract model as I thought, cool, thanks.
Ok, in that case maybe you could ask in django-users
or ask in IRC for help? Some ideas of how to overcome that:
- fix code as you suggested by using abstract and fake that 2nd migration to run, since that won't happen again for that same model (but other django code might rerun migrations, not sure? e.g. tests? but can be turned off) ; so next time you add another field to that mixin it won't change migration 0001 since you already ran it, so the newly generated migration to
AddField
will run fine. Might also help to squash all migrations so it'll generate proper new tables?
- or, if possible, create a new model and extend from proper mixin and copy all records over? But this seems more complicated in an environment with lots of data, probably there's a better workaround like the one above if that works.
comment:8 by , 7 years ago
Triage Stage: | Unreviewed → Accepted |
---|
I guess it's related and/or duplicate of #23521.
comment:9 by , 7 years ago
My first reaction, when I saw this ticket, was to think that it's invalid: Once you have a migration that uses some code construct -- a function or a base-class -- that code should not change, as changing it in any significant way undermines the idea that migrations recreate the models as they were at the time the migration was generated.
However, that's not what our docs say -- the docs only say that the function or class needs "to be kept around", not that it needs to be kept unchanged.
That said, I still don't think Django should support such changes, other than to flag them as errors.
That said, I'm not sure if there's any reasonable way to detect these situations automatically.
Either way, I think we should fix the docs.
#23521 may be related, but I don't think it's a duplicate, as the use-case described there is an intended one.
comment:10 by , 6 years ago
I hit this one today: without testing (I'll try to get there), it seems likely that adding fields to an abstract model will have a similar result.
comment:11 by , 16 months ago
This seems to be still an issue. Here's couple of things that I am noticing:
- There is, of course, no problem if you run
makemigrations
once in the end, after converting the Mixin into an abstract Model.
- There is also no problem if you run
migrate
before changing the Mixin, and thenmakemigrations
andmigrate
again.
That actually results to the exact same migration as in the problematic case.
Here is a test I wrote that can reproduce the issue, in case it can be helpful:
# <In tests.migrations.test_operations.OperationTests> def test_field_can_be_added_to_mixin_bases(self): # Replicating the scenario where a Mixin class has initially no fields # and does not extend Model; and then gets refactored to an abstract Model # with fields specified. class SomeMixin(models.Model): cuteness = models.IntegerField(null=True) class Meta: abstract = True operation = migrations.CreateModel( name='CutePony', fields=[ ('id', models.AutoField(primary_key=True)), ('pink', models.IntegerField(null=True)), ], bases=(SomeMixin, models.Model), ) app_label = 'test_28438' project_state, new_state = self.make_test_state(app_label, operation) with connection.schema_editor() as editor: operation.database_forwards(app_label, editor, project_state, new_state) project_state = new_state new_state = project_state.clone() operation = migrations.AddField( model_name='CutePony', name='cuteness', field=models.IntegerField(null=True) ) operation.state_forwards(app_label, new_state) with connection.schema_editor() as editor: operation.database_forwards(app_label, editor, project_state, new_state) self.assertTableExists(f'{app_label}_cutepony')
(This raises: django.db.utils.OperationalError: duplicate column name: cuteness
)
Hi Michal,
I'm not sure why the
bases=...
is added when you have a non-abstract base class that doesn't extend frommodels.Model
. Maybe someone more knowledgable could answer that. I can't find any mention of mixin in inheritance documentation, so I'd probably just use abstract models as suggested in this SO post to avoid going in undocumented territory.Maybe documentation not to use
object
base class with models could be added? Or trigger an error if you tried doing that to not shoot yourself in the foot?I could reproduce this for
postgres
(worked forsqlite
since it generates a new table in 2nd migration and copies every row, instead of adding a new column; probably inefficient, not the way to go anyway), so assuming that's what you used.Also found this github package, but I'd stay away from it unless I understood what it did, which I don't haha.