2013-12-29 03:13:08 +08:00
|
|
|
from django.apps import AppConfig
|
2013-12-24 19:25:17 +08:00
|
|
|
from django.apps.registry import Apps
|
2013-05-11 00:07:13 +08:00
|
|
|
from django.db import models
|
2013-12-06 22:05:12 +08:00
|
|
|
from django.db.models.options import DEFAULT_NAMES, normalize_unique_together
|
2013-09-07 01:14:09 +08:00
|
|
|
from django.utils import six
|
2013-05-18 19:49:56 +08:00
|
|
|
from django.utils.module_loading import import_by_path
|
2013-05-11 00:07:13 +08:00
|
|
|
|
|
|
|
|
2013-09-07 01:14:09 +08:00
|
|
|
class InvalidBasesError(ValueError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, models=None):
|
|
|
|
self.models = models or {}
|
2013-12-24 19:25:17 +08:00
|
|
|
self.apps = None
|
2013-05-11 00:07:13 +08:00
|
|
|
|
2013-05-19 00:30:34 +08:00
|
|
|
def add_model_state(self, model_state):
|
|
|
|
self.models[(model_state.app_label, model_state.name.lower())] = model_state
|
|
|
|
|
2013-05-11 00:07:13 +08:00
|
|
|
def clone(self):
|
|
|
|
"Returns an exact copy of this ProjectState"
|
2013-05-18 19:49:56 +08:00
|
|
|
return ProjectState(
|
2013-11-03 17:22:11 +08:00
|
|
|
models=dict((k, v.clone()) for k, v in self.models.items())
|
2013-05-11 00:07:13 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
def render(self):
|
2013-12-24 19:25:17 +08:00
|
|
|
"Turns the project state into actual models in a new Apps"
|
|
|
|
if self.apps is None:
|
2013-12-29 03:13:08 +08:00
|
|
|
# Populate the app registry with a stub for each application.
|
|
|
|
app_labels = set(model_state.app_label for model_state in self.models.values())
|
2013-12-30 23:03:06 +08:00
|
|
|
self.apps = Apps([AppConfigStub(label) for label in sorted(app_labels)])
|
2013-09-07 01:14:09 +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.
|
|
|
|
unrendered_models = list(self.models.values())
|
|
|
|
while unrendered_models:
|
|
|
|
new_unrendered_models = []
|
|
|
|
for model in unrendered_models:
|
|
|
|
try:
|
2013-12-24 19:25:17 +08:00
|
|
|
model.render(self.apps)
|
2013-09-07 01:14:09 +08:00
|
|
|
except InvalidBasesError:
|
|
|
|
new_unrendered_models.append(model)
|
|
|
|
if len(new_unrendered_models) == len(unrendered_models):
|
|
|
|
raise InvalidBasesError("Cannot resolve bases for %r" % new_unrendered_models)
|
|
|
|
unrendered_models = new_unrendered_models
|
2013-12-24 19:25:17 +08:00
|
|
|
return self.apps
|
2013-05-11 00:07:13 +08:00
|
|
|
|
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 = {}
|
2013-12-24 19:25:17 +08:00
|
|
|
for model in apps.get_models():
|
2013-05-18 19:49:56 +08:00
|
|
|
model_state = ModelState.from_model(model)
|
2013-10-10 23:07:48 +08:00
|
|
|
app_models[(model_state.app_label, model_state.name.lower())] = model_state
|
|
|
|
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
|
|
|
|
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):
|
|
|
|
"""
|
|
|
|
Stubs a Django AppConfig. Only provides a label and a dict of models.
|
|
|
|
"""
|
|
|
|
def __init__(self, label):
|
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
|
|
|
|
|
|
|
|
|
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
|
|
|
"""
|
|
|
|
|
2013-05-30 00:47:10 +08:00
|
|
|
def __init__(self, app_label, name, fields, options=None, bases=None):
|
2013-05-11 00:07:13 +08:00
|
|
|
self.app_label = app_label
|
|
|
|
self.name = 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, )
|
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.")
|
2013-05-18 19:49:56 +08:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_model(cls, model):
|
|
|
|
"""
|
|
|
|
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:
|
2013-05-18 19:49:56 +08:00
|
|
|
name, path, args, kwargs = field.deconstruct()
|
|
|
|
field_class = import_by_path(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,
|
|
|
|
))
|
2013-11-27 23:28:33 +08:00
|
|
|
for field in model._meta.local_many_to_many:
|
|
|
|
name, path, args, kwargs = field.deconstruct()
|
|
|
|
field_class = import_by_path(path)
|
2013-12-05 22:19:46 +08:00
|
|
|
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"]
|
|
|
|
options[name] = set(normalize_unique_together(ut))
|
2013-07-02 18:19:02 +08:00
|
|
|
else:
|
|
|
|
options[name] = model._meta.original_attrs[name]
|
2013-05-18 19:49:56 +08:00
|
|
|
# Make our record
|
2013-09-07 01:14:09 +08:00
|
|
|
bases = tuple(
|
2013-12-30 01:42:12 +08:00
|
|
|
("%s.%s" % (base._meta.app_label, base._meta.model_name) if hasattr(base, "_meta") else base)
|
2013-09-07 01:14:09 +08:00
|
|
|
for base in model.__bases__
|
|
|
|
if (not hasattr(base, "_meta") or not base._meta.abstract)
|
|
|
|
)
|
2013-06-23 00:15:51 +08:00
|
|
|
if not bases:
|
|
|
|
bases = (models.Model, )
|
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,
|
2013-05-18 19:49:56 +08:00
|
|
|
)
|
2013-05-11 00:07:13 +08:00
|
|
|
|
|
|
|
def clone(self):
|
|
|
|
"Returns an exact copy of this ModelState"
|
2013-06-20 22:12:59 +08:00
|
|
|
# We deep-clone the fields using deconstruction
|
|
|
|
fields = []
|
|
|
|
for name, field in self.fields:
|
|
|
|
_, path, args, kwargs = field.deconstruct()
|
|
|
|
field_class = import_by_path(path)
|
|
|
|
fields.append((name, field_class(*args, **kwargs)))
|
|
|
|
# Now make a copy
|
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,
|
|
|
|
fields=fields,
|
|
|
|
options=dict(self.options),
|
|
|
|
bases=self.bases,
|
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)
|
2013-07-02 18:19:02 +08:00
|
|
|
if "unique_together" in meta_contents:
|
|
|
|
meta_contents["unique_together"] = list(meta_contents["unique_together"])
|
2013-05-11 00:07:13 +08:00
|
|
|
meta = type("Meta", tuple(), meta_contents)
|
|
|
|
# Then, work out our bases
|
2013-12-28 21:55:54 +08:00
|
|
|
try:
|
|
|
|
bases = tuple(
|
|
|
|
(apps.get_model(*base.split(".", 1)) if isinstance(base, six.string_types) else base)
|
|
|
|
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
|
|
|
|
body = dict(self.fields)
|
|
|
|
body['Meta'] = meta
|
|
|
|
body['__module__'] = "__fake__"
|
|
|
|
# Then, make a Model object
|
|
|
|
return type(
|
|
|
|
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
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
return (
|
|
|
|
(self.app_label == other.app_label) and
|
|
|
|
(self.name == other.name) and
|
|
|
|
(len(self.fields) == len(other.fields)) and
|
|
|
|
all((k1 == k2 and (f1.deconstruct()[1:] == f2.deconstruct()[1:])) for (k1, f1), (k2, f2) in zip(self.fields, other.fields)) and
|
|
|
|
(self.options == other.options) and
|
|
|
|
(self.bases == other.bases)
|
|
|
|
)
|
|
|
|
|
|
|
|
def __ne__(self, other):
|
|
|
|
return not (self == other)
|