2013-05-11 00:07:13 +08:00
|
|
|
from django.db import models
|
|
|
|
from django.db.models.loading import BaseAppCache
|
2013-05-19 18:35:17 +08:00
|
|
|
from django.db.models.options import DEFAULT_NAMES
|
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
|
|
|
|
|
|
|
|
|
|
|
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 {}
|
|
|
|
self.app_cache = None
|
|
|
|
|
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-05-11 00:07:13 +08:00
|
|
|
models = dict((k, v.copy()) for k, v in self.models.items())
|
|
|
|
)
|
|
|
|
|
|
|
|
def render(self):
|
|
|
|
"Turns the project state into actual models in a new AppCache"
|
|
|
|
if self.app_cache is None:
|
|
|
|
self.app_cache = BaseAppCache()
|
2013-05-19 00:30:34 +08:00
|
|
|
for model in self.models.values():
|
2013-05-11 00:07:13 +08:00
|
|
|
model.render(self.app_cache)
|
|
|
|
return self.app_cache
|
|
|
|
|
2013-05-18 17:48:46 +08:00
|
|
|
@classmethod
|
|
|
|
def from_app_cache(cls, app_cache):
|
|
|
|
"Takes in an AppCache and returns a ProjectState matching it"
|
2013-05-18 19:49:56 +08:00
|
|
|
models = {}
|
2013-05-18 17:48:46 +08:00
|
|
|
for model in app_cache.get_models():
|
2013-05-18 19:49:56 +08:00
|
|
|
model_state = ModelState.from_model(model)
|
|
|
|
models[(model_state.app_label, model_state.name.lower())] = model_state
|
|
|
|
return cls(models)
|
2013-05-18 17:48:46 +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-18 19:49:56 +08:00
|
|
|
def __init__(self, app_label, name, fields=None, options=None, bases=None):
|
2013-05-11 00:07:13 +08:00
|
|
|
self.app_label = app_label
|
|
|
|
self.name = name
|
|
|
|
self.fields = fields or []
|
|
|
|
self.options = options or {}
|
2013-05-18 19:49:56 +08:00
|
|
|
self.bases = bases or (models.Model, )
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_model(cls, model):
|
|
|
|
"""
|
|
|
|
Feed me a model, get a ModelState representing it out.
|
|
|
|
"""
|
|
|
|
# Deconstruct the fields
|
|
|
|
fields = []
|
|
|
|
for field in model._meta.local_fields:
|
|
|
|
name, path, args, kwargs = field.deconstruct()
|
|
|
|
field_class = import_by_path(path)
|
|
|
|
fields.append((name, field_class(*args, **kwargs)))
|
2013-05-19 18:35:17 +08:00
|
|
|
# Extract the options
|
|
|
|
options = {}
|
|
|
|
for name in DEFAULT_NAMES:
|
|
|
|
# Ignore some special options
|
|
|
|
if name in ["app_cache", "app_label"]:
|
|
|
|
continue
|
|
|
|
if name in model._meta.original_attrs:
|
|
|
|
options[name] = model._meta.original_attrs[name]
|
2013-05-18 19:49:56 +08:00
|
|
|
# Make our record
|
|
|
|
return cls(
|
|
|
|
model._meta.app_label,
|
|
|
|
model._meta.object_name,
|
|
|
|
fields,
|
2013-05-19 18:35:17 +08:00
|
|
|
options,
|
|
|
|
model.__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"
|
|
|
|
return self.__class__(
|
|
|
|
app_label = self.app_label,
|
|
|
|
name = self.name,
|
|
|
|
fields = self.fields,
|
|
|
|
options = self.options,
|
|
|
|
bases = self.bases,
|
|
|
|
)
|
|
|
|
|
|
|
|
def render(self, app_cache):
|
|
|
|
"Creates a Model object from our current state into the given app_cache"
|
|
|
|
# First, make a Meta object
|
|
|
|
meta_contents = {'app_label': self.app_label, "app_cache": app_cache}
|
|
|
|
meta_contents.update(self.options)
|
|
|
|
meta = type("Meta", tuple(), meta_contents)
|
|
|
|
# Then, work out our bases
|
|
|
|
# TODO: Use the actual bases
|
|
|
|
# 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-05-19 00:30:34 +08:00
|
|
|
tuple(self.bases),
|
2013-05-11 00:07:13 +08:00
|
|
|
body,
|
|
|
|
)
|