2014-05-06 01:50:51 +08:00
|
|
|
from __future__ import unicode_literals
|
2015-01-28 20:35:27 +08:00
|
|
|
|
2014-11-06 03:53:39 +08:00
|
|
|
import copy
|
2015-01-28 20:35:27 +08:00
|
|
|
from collections import OrderedDict
|
2014-05-06 01:50:51 +08:00
|
|
|
|
2013-12-29 03:13:08 +08:00
|
|
|
from django.apps import AppConfig
|
2014-05-01 03:25:12 +08:00
|
|
|
from django.apps.registry import Apps, apps as global_apps
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.conf import settings
|
2013-05-11 00:07:13 +08:00
|
|
|
from django.db import models
|
2014-06-16 05:55:44 +08:00
|
|
|
from django.db.models.fields.proxy import OrderWrt
|
2015-03-03 16:43:56 +08:00
|
|
|
from django.db.models.fields.related import RECURSIVE_RELATIONSHIP_CONSTANT
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.db.models.options import DEFAULT_NAMES, normalize_together
|
2015-03-03 16:43:56 +08:00
|
|
|
from django.db.models.utils import make_model_tuple
|
2013-09-07 01:14:09 +08:00
|
|
|
from django.utils import six
|
2014-08-07 20:33:42 +08:00
|
|
|
from django.utils.encoding import force_text, smart_text
|
2014-11-05 17:43:31 +08:00
|
|
|
from django.utils.functional import cached_property
|
2014-01-21 04:15:14 +08:00
|
|
|
from django.utils.module_loading import import_string
|
2014-12-25 20:30:37 +08:00
|
|
|
from django.utils.version import get_docs_version
|
2013-05-11 00:07:13 +08:00
|
|
|
|
|
|
|
|
2013-09-07 01:14:09 +08:00
|
|
|
class InvalidBasesError(ValueError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2015-02-09 08:11:25 +08:00
|
|
|
def _get_app_label_and_model_name(model, app_label=''):
|
|
|
|
if isinstance(model, six.string_types):
|
|
|
|
split = model.split('.', 1)
|
|
|
|
return (tuple(split) if len(split) == 2 else (app_label, split[0]))
|
|
|
|
else:
|
|
|
|
return model._meta.app_label, model._meta.model_name
|
|
|
|
|
|
|
|
|
|
|
|
def get_related_models_recursive(model):
|
|
|
|
"""
|
|
|
|
Returns all models that have a direct or indirect relationship
|
|
|
|
to the given model.
|
2015-04-01 05:35:41 +08:00
|
|
|
|
|
|
|
Relationships are either defined by explicit relational fields, like
|
|
|
|
ForeignKey, ManyToManyField or OneToOneField, or by inheriting from another
|
|
|
|
model (a superclass is related to its subclasses, but not vice versa). Note,
|
|
|
|
however, that a model inheriting from a concrete model is also related to
|
|
|
|
its superclass through the implicit *_ptr OneToOneField on the subclass.
|
2015-02-09 08:11:25 +08:00
|
|
|
"""
|
|
|
|
def _related_models(m):
|
|
|
|
return [
|
|
|
|
f.related_model for f in m._meta.get_fields(include_parents=True, include_hidden=True)
|
|
|
|
if f.is_relation and not isinstance(f.related_model, six.string_types)
|
|
|
|
] + [
|
|
|
|
subclass for subclass in m.__subclasses__()
|
|
|
|
if issubclass(subclass, models.Model)
|
|
|
|
]
|
|
|
|
|
|
|
|
seen = set()
|
|
|
|
queue = _related_models(model)
|
|
|
|
for rel_mod in queue:
|
|
|
|
rel_app_label, rel_model_name = rel_mod._meta.app_label, rel_mod._meta.model_name
|
|
|
|
if (rel_app_label, rel_model_name) in seen:
|
|
|
|
continue
|
|
|
|
seen.add((rel_app_label, rel_model_name))
|
|
|
|
queue.extend(_related_models(rel_mod))
|
|
|
|
return seen - {(model._meta.app_label, model._meta.model_name)}
|
|
|
|
|
|
|
|
|
2013-05-11 00:07:13 +08:00
|
|
|
class ProjectState(object):
|
|
|
|
"""
|
|
|
|
Represents the entire project's overall state.
|
|
|
|
This is the item that is passed around - we do it here rather than at the
|
|
|
|
app level so that cross-app FKs/etc. resolve properly.
|
|
|
|
"""
|
|
|
|
|
2014-05-01 03:25:12 +08:00
|
|
|
def __init__(self, models=None, real_apps=None):
|
2013-05-11 00:07:13 +08:00
|
|
|
self.models = models or {}
|
2014-05-01 03:25:12 +08:00
|
|
|
# Apps to include from main registry, usually unmigrated ones
|
|
|
|
self.real_apps = real_apps or []
|
2013-05-11 00:07:13 +08:00
|
|
|
|
2014-11-06 03:53:39 +08:00
|
|
|
def add_model(self, model_state):
|
2015-01-02 23:37:21 +08:00
|
|
|
app_label, model_name = model_state.app_label, model_state.name_lower
|
2014-11-06 03:53:39 +08:00
|
|
|
self.models[(app_label, model_name)] = model_state
|
|
|
|
if 'apps' in self.__dict__: # hasattr would cache the property
|
|
|
|
self.reload_model(app_label, model_name)
|
|
|
|
|
|
|
|
def remove_model(self, app_label, model_name):
|
|
|
|
del self.models[app_label, model_name]
|
|
|
|
if 'apps' in self.__dict__: # hasattr would cache the property
|
|
|
|
self.apps.unregister_model(app_label, model_name)
|
2015-04-04 05:36:35 +08:00
|
|
|
# Need to do this explicitly since unregister_model() doesn't clear
|
|
|
|
# the cache automatically (#24513)
|
|
|
|
self.apps.clear_cache()
|
2014-11-06 03:53:39 +08:00
|
|
|
|
|
|
|
def reload_model(self, app_label, model_name):
|
|
|
|
if 'apps' in self.__dict__: # hasattr would cache the property
|
|
|
|
try:
|
2015-02-09 08:11:25 +08:00
|
|
|
old_model = self.apps.get_model(app_label, model_name)
|
2014-11-06 03:53:39 +08:00
|
|
|
except LookupError:
|
2015-02-09 08:11:25 +08:00
|
|
|
related_models = set()
|
|
|
|
else:
|
|
|
|
# Get all relations to and from the old model before reloading,
|
|
|
|
# as _meta.apps may change
|
|
|
|
related_models = get_related_models_recursive(old_model)
|
|
|
|
|
|
|
|
# Get all outgoing references from the model to be rendered
|
|
|
|
model_state = self.models[(app_label, model_name)]
|
|
|
|
for name, field in model_state.fields:
|
|
|
|
if field.is_relation:
|
2015-02-26 22:19:17 +08:00
|
|
|
if field.remote_field.model == RECURSIVE_RELATIONSHIP_CONSTANT:
|
2015-02-09 08:11:25 +08:00
|
|
|
continue
|
2015-02-26 22:19:17 +08:00
|
|
|
rel_app_label, rel_model_name = _get_app_label_and_model_name(field.remote_field.model, app_label)
|
2015-02-09 08:11:25 +08:00
|
|
|
related_models.add((rel_app_label, rel_model_name.lower()))
|
|
|
|
|
2015-04-04 05:36:35 +08:00
|
|
|
# Include the model itself
|
|
|
|
related_models.add((app_label, model_name))
|
|
|
|
|
2015-02-09 08:11:25 +08:00
|
|
|
# Unregister all related models
|
|
|
|
for rel_app_label, rel_model_name in related_models:
|
|
|
|
self.apps.unregister_model(rel_app_label, rel_model_name)
|
2015-04-04 05:36:35 +08:00
|
|
|
# Need to do it once all models are unregistered to avoid corrupting
|
|
|
|
# existing models' _meta
|
|
|
|
self.apps.clear_cache()
|
2015-02-09 08:11:25 +08:00
|
|
|
|
2015-04-04 05:36:35 +08:00
|
|
|
states_to_be_rendered = []
|
2015-02-09 08:11:25 +08:00
|
|
|
# Gather all models states of those models that will be rerendered.
|
|
|
|
# This includes:
|
2015-04-04 05:36:35 +08:00
|
|
|
# 1. All related models of unmigrated apps
|
2015-02-09 08:11:25 +08:00
|
|
|
for model_state in self.apps.real_models:
|
|
|
|
if (model_state.app_label, model_state.name_lower) in related_models:
|
|
|
|
states_to_be_rendered.append(model_state)
|
|
|
|
|
2015-04-04 05:36:35 +08:00
|
|
|
# 2. All related models of migrated apps
|
2015-02-09 08:11:25 +08:00
|
|
|
for rel_app_label, rel_model_name in related_models:
|
|
|
|
try:
|
|
|
|
model_state = self.models[rel_app_label, rel_model_name]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
states_to_be_rendered.append(model_state)
|
|
|
|
|
|
|
|
# Render all models
|
|
|
|
self.apps.render_multiple(states_to_be_rendered)
|
2013-05-19 00:30:34 +08:00
|
|
|
|
2013-05-11 00:07:13 +08:00
|
|
|
def clone(self):
|
|
|
|
"Returns an exact copy of this ProjectState"
|
2014-11-06 03:53:39 +08:00
|
|
|
new_state = ProjectState(
|
2014-12-07 05:00:09 +08:00
|
|
|
models={k: v.clone() for k, v in self.models.items()},
|
2014-05-01 03:25:12 +08:00
|
|
|
real_apps=self.real_apps,
|
2013-05-11 00:07:13 +08:00
|
|
|
)
|
2014-11-06 03:53:39 +08:00
|
|
|
if 'apps' in self.__dict__:
|
|
|
|
new_state.apps = self.apps.clone()
|
|
|
|
return new_state
|
2013-05-11 00:07:13 +08:00
|
|
|
|
2014-11-05 17:43:31 +08:00
|
|
|
@cached_property
|
|
|
|
def apps(self):
|
|
|
|
return StateApps(self.real_apps, self.models)
|
2013-05-11 00:07:13 +08:00
|
|
|
|
2014-12-30 09:52:32 +08:00
|
|
|
@property
|
|
|
|
def concrete_apps(self):
|
|
|
|
self.apps = StateApps(self.real_apps, self.models, ignore_swappable=True)
|
|
|
|
return self.apps
|
|
|
|
|
2013-05-18 17:48:46 +08:00
|
|
|
@classmethod
|
2013-12-24 19:25:17 +08:00
|
|
|
def from_apps(cls, apps):
|
|
|
|
"Takes in an Apps and returns a ProjectState matching it"
|
2013-10-10 23:07:48 +08:00
|
|
|
app_models = {}
|
2014-06-23 04:00:49 +08:00
|
|
|
for model in apps.get_models(include_swapped=True):
|
2014-03-26 06:30:29 +08:00
|
|
|
model_state = ModelState.from_model(model)
|
2015-01-02 23:37:21 +08:00
|
|
|
app_models[(model_state.app_label, model_state.name_lower)] = model_state
|
2013-10-10 23:07:48 +08:00
|
|
|
return cls(app_models)
|
2013-05-18 17:48:46 +08:00
|
|
|
|
2013-09-25 20:47:46 +08:00
|
|
|
def __eq__(self, other):
|
|
|
|
if set(self.models.keys()) != set(other.models.keys()):
|
|
|
|
return False
|
2014-05-01 03:25:12 +08:00
|
|
|
if set(self.real_apps) != set(other.real_apps):
|
|
|
|
return False
|
2013-09-25 20:47:46 +08:00
|
|
|
return all(model == other.models[key] for key, model in self.models.items())
|
|
|
|
|
|
|
|
def __ne__(self, other):
|
|
|
|
return not (self == other)
|
|
|
|
|
2013-05-11 00:07:13 +08:00
|
|
|
|
2013-12-29 03:13:08 +08:00
|
|
|
class AppConfigStub(AppConfig):
|
|
|
|
"""
|
2014-01-28 04:28:53 +08:00
|
|
|
Stubs a Django AppConfig. Only provides a label, and a dict of models.
|
2013-12-29 03:13:08 +08:00
|
|
|
"""
|
2014-01-28 04:28:53 +08:00
|
|
|
# Not used, but required by AppConfig.__init__
|
|
|
|
path = ''
|
|
|
|
|
2013-12-29 03:13:08 +08:00
|
|
|
def __init__(self, label):
|
2014-09-13 04:11:09 +08:00
|
|
|
self.label = label
|
2014-09-13 05:45:08 +08:00
|
|
|
# App-label and app-name are not the same thing, so technically passing
|
|
|
|
# in the label here is wrong. In practice, migrations don't care about
|
|
|
|
# the app name, but we need something unique, and the label works fine.
|
2013-12-31 23:23:42 +08:00
|
|
|
super(AppConfigStub, self).__init__(label, None)
|
2013-12-29 03:13:08 +08:00
|
|
|
|
|
|
|
def import_models(self, all_models):
|
|
|
|
self.models = all_models
|
|
|
|
|
|
|
|
|
2014-11-05 17:43:31 +08:00
|
|
|
class StateApps(Apps):
|
|
|
|
"""
|
|
|
|
Subclass of the global Apps registry class to better handle dynamic model
|
|
|
|
additions and removals.
|
|
|
|
"""
|
2014-12-30 09:52:32 +08:00
|
|
|
def __init__(self, real_apps, models, ignore_swappable=False):
|
2014-11-05 17:43:31 +08:00
|
|
|
# Any apps in self.real_apps should have all their models included
|
|
|
|
# in the render. We don't use the original model instances as there
|
|
|
|
# are some variables that refer to the Apps object.
|
|
|
|
# FKs/M2Ms from real apps are also not included as they just
|
|
|
|
# mess things up with partial states (due to lack of dependencies)
|
2015-02-09 08:11:25 +08:00
|
|
|
self.real_models = []
|
2014-11-05 17:43:31 +08:00
|
|
|
for app_label in real_apps:
|
|
|
|
app = global_apps.get_app_config(app_label)
|
|
|
|
for model in app.get_models():
|
2015-02-09 08:11:25 +08:00
|
|
|
self.real_models.append(ModelState.from_model(model, exclude_rels=True))
|
2014-11-05 17:43:31 +08:00
|
|
|
# Populate the app registry with a stub for each application.
|
|
|
|
app_labels = {model_state.app_label for model_state in models.values()}
|
|
|
|
app_configs = [AppConfigStub(label) for label in sorted(real_apps + list(app_labels))]
|
|
|
|
super(StateApps, self).__init__(app_configs)
|
|
|
|
|
2015-02-09 08:11:25 +08:00
|
|
|
self.render_multiple(list(models.values()) + self.real_models)
|
|
|
|
|
2015-03-03 22:12:18 +08:00
|
|
|
# There shouldn't be any operations pending at this point.
|
|
|
|
pending_models = set(self._pending_operations)
|
|
|
|
if ignore_swappable:
|
|
|
|
pending_models -= {make_model_tuple(settings.AUTH_USER_MODEL)}
|
|
|
|
if pending_models:
|
|
|
|
msg = "Unhandled pending operations for models: %s"
|
|
|
|
labels = (".".join(model_key) for model_key in self._pending_operations)
|
|
|
|
raise ValueError(msg % ", ".join(labels))
|
2015-02-09 08:11:25 +08:00
|
|
|
|
|
|
|
def render_multiple(self, model_states):
|
2014-11-05 17:43:31 +08:00
|
|
|
# We keep trying to render the models in a loop, ignoring invalid
|
|
|
|
# base errors, until the size of the unrendered models doesn't
|
|
|
|
# decrease by at least one, meaning there's a base dependency loop/
|
|
|
|
# missing base.
|
2015-02-22 11:26:17 +08:00
|
|
|
if not model_states:
|
|
|
|
return
|
|
|
|
# Prevent that all model caches are expired for each render.
|
|
|
|
self.ready = False
|
2015-02-09 08:11:25 +08:00
|
|
|
unrendered_models = model_states
|
2014-11-05 17:43:31 +08:00
|
|
|
while unrendered_models:
|
|
|
|
new_unrendered_models = []
|
|
|
|
for model in unrendered_models:
|
|
|
|
try:
|
|
|
|
model.render(self)
|
|
|
|
except InvalidBasesError:
|
|
|
|
new_unrendered_models.append(model)
|
|
|
|
if len(new_unrendered_models) == len(unrendered_models):
|
2015-02-22 11:26:17 +08:00
|
|
|
self.ready = True
|
2014-11-05 17:43:31 +08:00
|
|
|
raise InvalidBasesError(
|
|
|
|
"Cannot resolve bases for %r\nThis can happen if you are inheriting models from an "
|
|
|
|
"app with migrations (e.g. contrib.auth)\n in an app with no migrations; see "
|
|
|
|
"https://docs.djangoproject.com/en/%s/topics/migrations/#dependencies "
|
|
|
|
"for more" % (new_unrendered_models, get_docs_version())
|
|
|
|
)
|
|
|
|
unrendered_models = new_unrendered_models
|
2015-02-22 11:26:17 +08:00
|
|
|
self.ready = True
|
|
|
|
self.clear_cache()
|
2014-11-05 17:43:31 +08:00
|
|
|
|
2014-11-06 03:53:39 +08:00
|
|
|
def clone(self):
|
|
|
|
"""
|
|
|
|
Return a clone of this registry, mainly used by the migration framework.
|
|
|
|
"""
|
|
|
|
clone = StateApps([], {})
|
|
|
|
clone.all_models = copy.deepcopy(self.all_models)
|
|
|
|
clone.app_configs = copy.deepcopy(self.app_configs)
|
2015-02-09 08:11:25 +08:00
|
|
|
# No need to actually clone them, they'll never change
|
|
|
|
clone.real_models = self.real_models
|
2014-11-06 03:53:39 +08:00
|
|
|
return clone
|
|
|
|
|
|
|
|
def register_model(self, app_label, model):
|
|
|
|
self.all_models[app_label][model._meta.model_name] = model
|
|
|
|
if app_label not in self.app_configs:
|
|
|
|
self.app_configs[app_label] = AppConfigStub(app_label)
|
|
|
|
self.app_configs[app_label].models = OrderedDict()
|
|
|
|
self.app_configs[app_label].models[model._meta.model_name] = model
|
2015-03-03 16:43:56 +08:00
|
|
|
self.do_pending_operations(model)
|
2014-11-06 03:53:39 +08:00
|
|
|
self.clear_cache()
|
|
|
|
|
|
|
|
def unregister_model(self, app_label, model_name):
|
|
|
|
try:
|
|
|
|
del self.all_models[app_label][model_name]
|
|
|
|
del self.app_configs[app_label].models[model_name]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
|
2014-11-05 17:43:31 +08:00
|
|
|
|
2013-05-11 00:07:13 +08:00
|
|
|
class ModelState(object):
|
|
|
|
"""
|
|
|
|
Represents a Django Model. We don't use the actual Model class
|
|
|
|
as it's not designed to have its options changed - instead, we
|
|
|
|
mutate this one and then render it into a Model as required.
|
2013-05-31 01:21:32 +08:00
|
|
|
|
|
|
|
Note that while you are allowed to mutate .fields, you are not allowed
|
|
|
|
to mutate the Field instances inside there themselves - you must instead
|
|
|
|
assign new ones, as these are not detached during a clone.
|
2013-05-11 00:07:13 +08:00
|
|
|
"""
|
|
|
|
|
2014-12-13 06:19:58 +08:00
|
|
|
def __init__(self, app_label, name, fields, options=None, bases=None, managers=None):
|
2013-05-11 00:07:13 +08:00
|
|
|
self.app_label = app_label
|
2014-05-06 02:39:26 +08:00
|
|
|
self.name = force_text(name)
|
2013-05-30 00:47:10 +08:00
|
|
|
self.fields = fields
|
2013-05-11 00:07:13 +08:00
|
|
|
self.options = options or {}
|
2013-05-18 19:49:56 +08:00
|
|
|
self.bases = bases or (models.Model, )
|
2014-12-13 06:19:58 +08:00
|
|
|
self.managers = managers or []
|
2013-05-30 00:47:10 +08:00
|
|
|
# Sanity-check that fields is NOT a dict. It must be ordered.
|
|
|
|
if isinstance(self.fields, dict):
|
|
|
|
raise ValueError("ModelState.fields cannot be a dict - it must be a list of 2-tuples.")
|
2014-05-23 06:51:11 +08:00
|
|
|
# Sanity-check that fields are NOT already bound to a model.
|
|
|
|
for name, field in fields:
|
|
|
|
if hasattr(field, 'model'):
|
|
|
|
raise ValueError(
|
|
|
|
'ModelState.fields cannot be bound to a model - "%s" is.' % name
|
|
|
|
)
|
2013-05-18 19:49:56 +08:00
|
|
|
|
2015-01-02 23:37:21 +08:00
|
|
|
@cached_property
|
|
|
|
def name_lower(self):
|
|
|
|
return self.name.lower()
|
|
|
|
|
2013-05-18 19:49:56 +08:00
|
|
|
@classmethod
|
2014-06-13 01:21:26 +08:00
|
|
|
def from_model(cls, model, exclude_rels=False):
|
2013-05-18 19:49:56 +08:00
|
|
|
"""
|
|
|
|
Feed me a model, get a ModelState representing it out.
|
|
|
|
"""
|
|
|
|
# Deconstruct the fields
|
|
|
|
fields = []
|
2013-09-01 05:18:44 +08:00
|
|
|
for field in model._meta.local_fields:
|
2015-02-26 22:19:17 +08:00
|
|
|
if getattr(field, "remote_field", None) and exclude_rels:
|
2014-06-13 01:21:26 +08:00
|
|
|
continue
|
2014-06-16 05:55:44 +08:00
|
|
|
if isinstance(field, OrderWrt):
|
|
|
|
continue
|
2013-05-18 19:49:56 +08:00
|
|
|
name, path, args, kwargs = field.deconstruct()
|
2014-01-21 04:15:14 +08:00
|
|
|
field_class = import_string(path)
|
2013-12-05 22:19:46 +08:00
|
|
|
try:
|
|
|
|
fields.append((name, field_class(*args, **kwargs)))
|
|
|
|
except TypeError as e:
|
2013-12-06 21:59:08 +08:00
|
|
|
raise TypeError("Couldn't reconstruct field %s on %s.%s: %s" % (
|
2013-12-05 22:19:46 +08:00
|
|
|
name,
|
2013-12-06 21:59:08 +08:00
|
|
|
model._meta.app_label,
|
2013-12-05 22:19:46 +08:00
|
|
|
model._meta.object_name,
|
|
|
|
e,
|
|
|
|
))
|
2014-06-13 01:21:26 +08:00
|
|
|
if not exclude_rels:
|
|
|
|
for field in model._meta.local_many_to_many:
|
|
|
|
name, path, args, kwargs = field.deconstruct()
|
|
|
|
field_class = import_string(path)
|
|
|
|
try:
|
|
|
|
fields.append((name, field_class(*args, **kwargs)))
|
|
|
|
except TypeError as e:
|
|
|
|
raise TypeError("Couldn't reconstruct m2m field %s on %s: %s" % (
|
|
|
|
name,
|
|
|
|
model._meta.object_name,
|
|
|
|
e,
|
|
|
|
))
|
2013-05-19 18:35:17 +08:00
|
|
|
# Extract the options
|
|
|
|
options = {}
|
|
|
|
for name in DEFAULT_NAMES:
|
|
|
|
# Ignore some special options
|
2013-12-24 19:25:17 +08:00
|
|
|
if name in ["apps", "app_label"]:
|
2013-05-19 18:35:17 +08:00
|
|
|
continue
|
2013-07-02 18:19:02 +08:00
|
|
|
elif name in model._meta.original_attrs:
|
|
|
|
if name == "unique_together":
|
2013-12-06 22:05:12 +08:00
|
|
|
ut = model._meta.original_attrs["unique_together"]
|
2014-03-02 03:06:15 +08:00
|
|
|
options[name] = set(normalize_together(ut))
|
|
|
|
elif name == "index_together":
|
|
|
|
it = model._meta.original_attrs["index_together"]
|
|
|
|
options[name] = set(normalize_together(it))
|
2013-07-02 18:19:02 +08:00
|
|
|
else:
|
|
|
|
options[name] = model._meta.original_attrs[name]
|
2014-08-07 10:13:37 +08:00
|
|
|
# Force-convert all options to text_type (#23226)
|
|
|
|
options = cls.force_text_recursive(options)
|
2014-06-13 02:12:07 +08:00
|
|
|
# If we're ignoring relationships, remove all field-listing model
|
|
|
|
# options (that option basically just means "make a stub model")
|
|
|
|
if exclude_rels:
|
|
|
|
for key in ["unique_together", "index_together", "order_with_respect_to"]:
|
|
|
|
if key in options:
|
|
|
|
del options[key]
|
2014-03-05 04:17:07 +08:00
|
|
|
|
|
|
|
def flatten_bases(model):
|
|
|
|
bases = []
|
|
|
|
for base in model.__bases__:
|
|
|
|
if hasattr(base, "_meta") and base._meta.abstract:
|
|
|
|
bases.extend(flatten_bases(base))
|
|
|
|
else:
|
|
|
|
bases.append(base)
|
|
|
|
return bases
|
|
|
|
|
|
|
|
# We can't rely on __mro__ directly because we only want to flatten
|
|
|
|
# abstract models and not the whole tree. However by recursing on
|
|
|
|
# __bases__ we may end up with duplicates and ordering issues, we
|
|
|
|
# therefore discard any duplicates and reorder the bases according
|
|
|
|
# to their index in the MRO.
|
2014-03-05 07:13:15 +08:00
|
|
|
flattened_bases = sorted(set(flatten_bases(model)), key=lambda x: model.__mro__.index(x))
|
2014-03-05 04:17:07 +08:00
|
|
|
|
2013-05-18 19:49:56 +08:00
|
|
|
# Make our record
|
2013-09-07 01:14:09 +08:00
|
|
|
bases = tuple(
|
2014-01-20 02:09:52 +08:00
|
|
|
(
|
|
|
|
"%s.%s" % (base._meta.app_label, base._meta.model_name)
|
|
|
|
if hasattr(base, "_meta") else
|
|
|
|
base
|
|
|
|
)
|
2014-03-05 04:17:07 +08:00
|
|
|
for base in flattened_bases
|
2013-09-07 01:14:09 +08:00
|
|
|
)
|
2014-01-20 02:09:52 +08:00
|
|
|
# Ensure at least one base inherits from models.Model
|
|
|
|
if not any((isinstance(base, six.string_types) or issubclass(base, models.Model)) for base in bases):
|
2014-03-05 04:17:07 +08:00
|
|
|
bases = (models.Model,)
|
2014-12-13 06:19:58 +08:00
|
|
|
|
|
|
|
# Constructs all managers on the model
|
|
|
|
managers = {}
|
|
|
|
|
|
|
|
def reconstruct_manager(mgr):
|
|
|
|
as_manager, manager_path, qs_path, args, kwargs = mgr.deconstruct()
|
|
|
|
if as_manager:
|
|
|
|
qs_class = import_string(qs_path)
|
|
|
|
instance = qs_class.as_manager()
|
|
|
|
else:
|
|
|
|
manager_class = import_string(manager_path)
|
|
|
|
instance = manager_class(*args, **kwargs)
|
|
|
|
# We rely on the ordering of the creation_counter of the original
|
|
|
|
# instance
|
|
|
|
managers[mgr.name] = (mgr.creation_counter, instance)
|
|
|
|
|
2015-02-06 21:24:51 +08:00
|
|
|
if hasattr(model, "_default_manager"):
|
|
|
|
default_manager_name = model._default_manager.name
|
|
|
|
# Make sure the default manager is always the first
|
|
|
|
if model._default_manager.use_in_migrations:
|
|
|
|
reconstruct_manager(model._default_manager)
|
|
|
|
else:
|
|
|
|
# Force this manager to be the first and thus default
|
|
|
|
managers[default_manager_name] = (0, models.Manager())
|
|
|
|
# Sort all managers by their creation counter
|
|
|
|
for _, manager, _ in sorted(model._meta.managers):
|
|
|
|
if manager.name == "_base_manager" or not manager.use_in_migrations:
|
|
|
|
continue
|
|
|
|
reconstruct_manager(manager)
|
|
|
|
# Sort all managers by their creation counter but take only name and
|
|
|
|
# instance for further processing
|
|
|
|
managers = [
|
|
|
|
(name, instance) for name, (cc, instance) in
|
|
|
|
sorted(managers.items(), key=lambda v: v[1])
|
|
|
|
]
|
|
|
|
if managers == [(default_manager_name, models.Manager())]:
|
|
|
|
managers = []
|
2014-12-13 06:19:58 +08:00
|
|
|
else:
|
|
|
|
managers = []
|
|
|
|
|
|
|
|
# Construct the new ModelState
|
2013-05-18 19:49:56 +08:00
|
|
|
return cls(
|
|
|
|
model._meta.app_label,
|
|
|
|
model._meta.object_name,
|
|
|
|
fields,
|
2013-05-19 18:35:17 +08:00
|
|
|
options,
|
2013-06-23 00:15:51 +08:00
|
|
|
bases,
|
2014-12-13 06:19:58 +08:00
|
|
|
managers,
|
2013-05-18 19:49:56 +08:00
|
|
|
)
|
2013-05-11 00:07:13 +08:00
|
|
|
|
2014-08-07 10:13:37 +08:00
|
|
|
@classmethod
|
|
|
|
def force_text_recursive(cls, value):
|
|
|
|
if isinstance(value, six.string_types):
|
2014-08-07 20:33:42 +08:00
|
|
|
return smart_text(value)
|
2014-08-07 10:13:37 +08:00
|
|
|
elif isinstance(value, list):
|
|
|
|
return [cls.force_text_recursive(x) for x in value]
|
|
|
|
elif isinstance(value, tuple):
|
|
|
|
return tuple(cls.force_text_recursive(x) for x in value)
|
|
|
|
elif isinstance(value, set):
|
|
|
|
return set(cls.force_text_recursive(x) for x in value)
|
|
|
|
elif isinstance(value, dict):
|
2014-12-07 05:00:09 +08:00
|
|
|
return {
|
|
|
|
cls.force_text_recursive(k): cls.force_text_recursive(v)
|
2014-08-07 10:13:37 +08:00
|
|
|
for k, v in value.items()
|
2014-12-07 05:00:09 +08:00
|
|
|
}
|
2014-08-07 10:13:37 +08:00
|
|
|
return value
|
|
|
|
|
2014-05-23 06:51:11 +08:00
|
|
|
def construct_fields(self):
|
|
|
|
"Deep-clone the fields using deconstruction"
|
2013-06-20 22:12:59 +08:00
|
|
|
for name, field in self.fields:
|
|
|
|
_, path, args, kwargs = field.deconstruct()
|
2014-01-21 04:15:14 +08:00
|
|
|
field_class = import_string(path)
|
2014-05-23 06:51:11 +08:00
|
|
|
yield name, field_class(*args, **kwargs)
|
|
|
|
|
2015-01-14 01:18:23 +08:00
|
|
|
def construct_managers(self):
|
|
|
|
"Deep-clone the managers using deconstruction"
|
|
|
|
# Sort all managers by their creation counter
|
|
|
|
sorted_managers = sorted(self.managers, key=lambda v: v[1].creation_counter)
|
|
|
|
for mgr_name, manager in sorted_managers:
|
|
|
|
as_manager, manager_path, qs_path, args, kwargs = manager.deconstruct()
|
|
|
|
if as_manager:
|
|
|
|
qs_class = import_string(qs_path)
|
|
|
|
yield mgr_name, qs_class.as_manager()
|
|
|
|
else:
|
|
|
|
manager_class = import_string(manager_path)
|
|
|
|
yield mgr_name, manager_class(*args, **kwargs)
|
|
|
|
|
2014-05-23 06:51:11 +08:00
|
|
|
def clone(self):
|
|
|
|
"Returns an exact copy of this ModelState"
|
2013-05-11 00:07:13 +08:00
|
|
|
return self.__class__(
|
2013-11-03 17:22:11 +08:00
|
|
|
app_label=self.app_label,
|
|
|
|
name=self.name,
|
2014-05-23 06:51:11 +08:00
|
|
|
fields=list(self.construct_fields()),
|
2013-11-03 17:22:11 +08:00
|
|
|
options=dict(self.options),
|
|
|
|
bases=self.bases,
|
2015-01-14 01:18:23 +08:00
|
|
|
managers=list(self.construct_managers()),
|
2013-05-11 00:07:13 +08:00
|
|
|
)
|
|
|
|
|
2013-12-24 19:25:17 +08:00
|
|
|
def render(self, apps):
|
|
|
|
"Creates a Model object from our current state into the given apps"
|
2013-05-11 00:07:13 +08:00
|
|
|
# First, make a Meta object
|
2013-12-24 19:25:17 +08:00
|
|
|
meta_contents = {'app_label': self.app_label, "apps": apps}
|
2013-05-11 00:07:13 +08:00
|
|
|
meta_contents.update(self.options)
|
2014-05-06 01:50:51 +08:00
|
|
|
meta = type(str("Meta"), tuple(), meta_contents)
|
2013-05-11 00:07:13 +08:00
|
|
|
# Then, work out our bases
|
2013-12-28 21:55:54 +08:00
|
|
|
try:
|
|
|
|
bases = tuple(
|
2014-01-26 19:57:08 +08:00
|
|
|
(apps.get_model(base) if isinstance(base, six.string_types) else base)
|
2013-12-28 21:55:54 +08:00
|
|
|
for base in self.bases
|
|
|
|
)
|
|
|
|
except LookupError:
|
2013-12-06 07:55:31 +08:00
|
|
|
raise InvalidBasesError("Cannot resolve one or more bases from %r" % (self.bases,))
|
2013-05-11 00:07:13 +08:00
|
|
|
# Turn fields into a dict for the body, add other bits
|
2014-05-23 06:51:11 +08:00
|
|
|
body = dict(self.construct_fields())
|
2013-05-11 00:07:13 +08:00
|
|
|
body['Meta'] = meta
|
|
|
|
body['__module__'] = "__fake__"
|
2014-12-13 06:19:58 +08:00
|
|
|
|
|
|
|
# Restore managers
|
2015-01-14 01:18:23 +08:00
|
|
|
body.update(self.construct_managers())
|
2014-12-13 06:19:58 +08:00
|
|
|
|
2014-11-06 03:53:39 +08:00
|
|
|
# Then, make a Model object (apps.register_model is called in __new__)
|
2013-05-11 00:07:13 +08:00
|
|
|
return type(
|
2014-04-01 03:25:08 +08:00
|
|
|
str(self.name),
|
2013-09-07 01:14:09 +08:00
|
|
|
bases,
|
2013-05-11 00:07:13 +08:00
|
|
|
body,
|
|
|
|
)
|
2013-06-20 22:19:30 +08:00
|
|
|
|
|
|
|
def get_field_by_name(self, name):
|
|
|
|
for fname, field in self.fields:
|
|
|
|
if fname == name:
|
|
|
|
return field
|
|
|
|
raise ValueError("No field called %s on model %s" % (name, self.name))
|
2013-09-25 20:47:46 +08:00
|
|
|
|
2014-07-06 14:57:23 +08:00
|
|
|
def __repr__(self):
|
|
|
|
return "<ModelState: '%s.%s'>" % (self.app_label, self.name)
|
|
|
|
|
2013-09-25 20:47:46 +08:00
|
|
|
def __eq__(self, other):
|
|
|
|
return (
|
|
|
|
(self.app_label == other.app_label) and
|
|
|
|
(self.name == other.name) and
|
|
|
|
(len(self.fields) == len(other.fields)) and
|
2014-10-25 12:42:44 +08:00
|
|
|
all((k1 == k2 and (f1.deconstruct()[1:] == f2.deconstruct()[1:]))
|
|
|
|
for (k1, f1), (k2, f2) in zip(self.fields, other.fields)) and
|
2013-09-25 20:47:46 +08:00
|
|
|
(self.options == other.options) and
|
2014-12-13 06:19:58 +08:00
|
|
|
(self.bases == other.bases) and
|
|
|
|
(self.managers == other.managers)
|
2013-09-25 20:47:46 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
def __ne__(self, other):
|
|
|
|
return not (self == other)
|