Make a start on operations and state (not sure if final layout)
This commit is contained in:
parent
eb5e50215a
commit
76d93a52cd
|
@ -0,0 +1 @@
|
|||
from .models import CreateModel, DeleteModel
|
|
@ -0,0 +1,38 @@
|
|||
class Operation(object):
|
||||
"""
|
||||
Base class for migration operations.
|
||||
|
||||
It's responsible for both mutating the in-memory model state
|
||||
(see db/migrations/state.py) to represent what it performs, as well
|
||||
as actually performing it against a live database.
|
||||
|
||||
Note that some operations won't modify memory state at all (e.g. data
|
||||
copying operations), and some will need their modifications to be
|
||||
optionally specified by the user (e.g. custom Python code snippets)
|
||||
"""
|
||||
|
||||
# If this migration can be run in reverse.
|
||||
# Some operations are impossible to reverse, like deleting data.
|
||||
reversible = True
|
||||
|
||||
def state_forwards(self, app, state):
|
||||
"""
|
||||
Takes the state from the previous migration, and mutates it
|
||||
so that it matches what this migration would perform.
|
||||
"""
|
||||
raise NotImplementedError()
|
||||
|
||||
def database_forwards(self, app, schema_editor, from_state, to_state):
|
||||
"""
|
||||
Performs the mutation on the database schema in the normal
|
||||
(forwards) direction.
|
||||
"""
|
||||
raise NotImplementedError()
|
||||
|
||||
def database_backwards(self, app, schema_editor, from_state, to_state):
|
||||
"""
|
||||
Performs the mutation on the database schema in the reverse
|
||||
direction - e.g. if this were CreateModel, it would in fact
|
||||
drop the model's table.
|
||||
"""
|
||||
raise NotImplementedError()
|
|
@ -0,0 +1,26 @@
|
|||
from .base import Operation
|
||||
from django.db.migrations.state import ModelState
|
||||
|
||||
|
||||
class CreateModel(Operation):
|
||||
"""
|
||||
Create a model's table.
|
||||
"""
|
||||
|
||||
def __init__(self, name):
|
||||
self.name = name
|
||||
|
||||
def state_forwards(self, app, state):
|
||||
state.models[app, self.name.lower()] = ModelState(state, app, self.name)
|
||||
|
||||
def database_forwards(self, app, schema_editor, from_state, to_state):
|
||||
app_cache = to_state.render()
|
||||
schema_editor.create_model(app_cache.get_model(app, self.name))
|
||||
|
||||
def database_backwards(self, app, schema_editor, from_state, to_state):
|
||||
"""
|
||||
Performs the mutation on the database schema in the reverse
|
||||
direction - e.g. if this were CreateModel, it would in fact
|
||||
drop the model's table.
|
||||
"""
|
||||
raise NotImplementedError()
|
|
@ -0,0 +1,81 @@
|
|||
from django.db import models
|
||||
from django.db.models.loading import BaseAppCache
|
||||
|
||||
|
||||
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
|
||||
|
||||
def clone(self):
|
||||
"Returns an exact copy of this ProjectState"
|
||||
ps = ProjectState(
|
||||
models = dict((k, v.copy()) for k, v in self.models.items())
|
||||
)
|
||||
for model in ps.models.values():
|
||||
model.project_state = ps
|
||||
return ps
|
||||
|
||||
def render(self):
|
||||
"Turns the project state into actual models in a new AppCache"
|
||||
if self.app_cache is None:
|
||||
self.app_cache = BaseAppCache()
|
||||
for model in self.model.values:
|
||||
model.render(self.app_cache)
|
||||
return self.app_cache
|
||||
|
||||
|
||||
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.
|
||||
"""
|
||||
|
||||
def __init__(self, project_state, app_label, name, fields=None, options=None, bases=None):
|
||||
self.project_state = project_state
|
||||
self.app_label = app_label
|
||||
self.name = name
|
||||
self.fields = fields or []
|
||||
self.options = options or {}
|
||||
self.bases = bases or None
|
||||
|
||||
def clone(self):
|
||||
"Returns an exact copy of this ModelState"
|
||||
return self.__class__(
|
||||
project_state = self.project_state,
|
||||
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
|
||||
if self.bases:
|
||||
raise NotImplementedError("Custom bases not quite done yet!")
|
||||
else:
|
||||
bases = [models.Model]
|
||||
# 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,
|
||||
tuple(bases),
|
||||
body,
|
||||
)
|
Loading…
Reference in New Issue