[1.7.x] Factored create_fk_sql, create_unique_sql, and delete_constraint_sql.
Backport of 0bea6c8749
from master
This commit is contained in:
parent
1c1f418802
commit
09b5ff64b8
|
@ -232,15 +232,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
to_table = field.rel.to._meta.db_table
|
to_table = field.rel.to._meta.db_table
|
||||||
to_column = field.rel.to._meta.get_field(field.rel.field_name).column
|
to_column = field.rel.to._meta.get_field(field.rel.field_name).column
|
||||||
if self.connection.features.supports_foreign_keys:
|
if self.connection.features.supports_foreign_keys:
|
||||||
self.deferred_sql.append(
|
self.deferred_sql.append(self._create_fk_sql(model, field, "_fk_%(to_table)s_%(to_column)s"))
|
||||||
self.sql_create_fk % {
|
|
||||||
"name": self._create_index_name(model, [field.column], suffix="_fk_%s_%s" % (to_table, to_column)),
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"column": self.quote_name(field.column),
|
|
||||||
"to_table": self.quote_name(to_table),
|
|
||||||
"to_column": self.quote_name(to_column),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
elif self.sql_create_inline_fk:
|
elif self.sql_create_inline_fk:
|
||||||
definition += " " + self.sql_create_inline_fk % {
|
definition += " " + self.sql_create_inline_fk % {
|
||||||
"to_table": self.quote_name(to_table),
|
"to_table": self.quote_name(to_table),
|
||||||
|
@ -314,20 +306,11 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
model._meta.db_table,
|
model._meta.db_table,
|
||||||
", ".join(columns),
|
", ".join(columns),
|
||||||
))
|
))
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_names[0]))
|
||||||
self.sql_delete_unique % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": constraint_names[0],
|
|
||||||
},
|
|
||||||
)
|
|
||||||
# Created uniques
|
# Created uniques
|
||||||
for fields in news.difference(olds):
|
for fields in news.difference(olds):
|
||||||
columns = [model._meta.get_field_by_name(field)[0].column for field in fields]
|
columns = [model._meta.get_field_by_name(field)[0].column for field in fields]
|
||||||
self.execute(self.sql_create_unique % {
|
self.execute(self._create_unique_sql(model, columns))
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": self._create_index_name(model, columns, suffix="_uniq"),
|
|
||||||
"columns": ", ".join(self.quote_name(column) for column in columns),
|
|
||||||
})
|
|
||||||
|
|
||||||
def alter_index_together(self, model, old_index_together, new_index_together):
|
def alter_index_together(self, model, old_index_together, new_index_together):
|
||||||
"""
|
"""
|
||||||
|
@ -347,12 +330,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
model._meta.db_table,
|
model._meta.db_table,
|
||||||
", ".join(columns),
|
", ".join(columns),
|
||||||
))
|
))
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_index, model, constraint_names[0]))
|
||||||
self.sql_delete_index % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": constraint_names[0],
|
|
||||||
},
|
|
||||||
)
|
|
||||||
# Created indexes
|
# Created indexes
|
||||||
for fields in news.difference(olds):
|
for fields in news.difference(olds):
|
||||||
columns = [model._meta.get_field_by_name(field)[0].column for field in fields]
|
columns = [model._meta.get_field_by_name(field)[0].column for field in fields]
|
||||||
|
@ -431,17 +409,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
)
|
)
|
||||||
# Add any FK constraints later
|
# Add any FK constraints later
|
||||||
if field.rel and self.connection.features.supports_foreign_keys and field.db_constraint:
|
if field.rel and self.connection.features.supports_foreign_keys and field.db_constraint:
|
||||||
to_table = field.rel.to._meta.db_table
|
self.deferred_sql.append(self._create_fk_sql(model, field, "_fk_%(to_table)s_%(to_column)s"))
|
||||||
to_column = field.rel.to._meta.get_field(field.rel.field_name).column
|
|
||||||
self.deferred_sql.append(
|
|
||||||
self.sql_create_fk % {
|
|
||||||
"name": self._create_index_name(model, [field.column], suffix="_fk_%s_%s" % (to_table, to_column)),
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"column": self.quote_name(field.column),
|
|
||||||
"to_table": self.quote_name(to_table),
|
|
||||||
"to_column": self.quote_name(to_column),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Reset connection if required
|
# Reset connection if required
|
||||||
if self.connection.features.connection_persists_old_columns:
|
if self.connection.features.connection_persists_old_columns:
|
||||||
self.connection.close()
|
self.connection.close()
|
||||||
|
@ -461,12 +429,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
if field.rel:
|
if field.rel:
|
||||||
fk_names = self._constraint_names(model, [field.column], foreign_key=True)
|
fk_names = self._constraint_names(model, [field.column], foreign_key=True)
|
||||||
for fk_name in fk_names:
|
for fk_name in fk_names:
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_fk, model, fk_name))
|
||||||
self.sql_delete_fk % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": fk_name,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Delete the column
|
# Delete the column
|
||||||
sql = self.sql_delete_column % {
|
sql = self.sql_delete_column % {
|
||||||
"table": self.quote_name(model._meta.db_table),
|
"table": self.quote_name(model._meta.db_table),
|
||||||
|
@ -522,12 +485,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
old_field.column,
|
old_field.column,
|
||||||
))
|
))
|
||||||
for constraint_name in constraint_names:
|
for constraint_name in constraint_names:
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_name))
|
||||||
self.sql_delete_unique % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": constraint_name,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
# Drop any FK constraints, we'll remake them later
|
# Drop any FK constraints, we'll remake them later
|
||||||
fks_dropped = set()
|
fks_dropped = set()
|
||||||
if old_field.rel and old_field.db_constraint:
|
if old_field.rel and old_field.db_constraint:
|
||||||
|
@ -540,24 +498,14 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
))
|
))
|
||||||
for fk_name in fk_names:
|
for fk_name in fk_names:
|
||||||
fks_dropped.add((old_field.column,))
|
fks_dropped.add((old_field.column,))
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_fk, model, fk_name))
|
||||||
self.sql_delete_fk % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": fk_name,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Drop incoming FK constraints if we're a primary key and things are going
|
# Drop incoming FK constraints if we're a primary key and things are going
|
||||||
# to change.
|
# to change.
|
||||||
if old_field.primary_key and new_field.primary_key and old_type != new_type:
|
if old_field.primary_key and new_field.primary_key and old_type != new_type:
|
||||||
for rel in new_field.model._meta.get_all_related_objects():
|
for rel in new_field.model._meta.get_all_related_objects():
|
||||||
rel_fk_names = self._constraint_names(rel.model, [rel.field.column], foreign_key=True)
|
rel_fk_names = self._constraint_names(rel.model, [rel.field.column], foreign_key=True)
|
||||||
for fk_name in rel_fk_names:
|
for fk_name in rel_fk_names:
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_fk, rel.model, fk_name))
|
||||||
self.sql_delete_fk % {
|
|
||||||
"table": self.quote_name(rel.model._meta.db_table),
|
|
||||||
"name": fk_name,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Removed an index?
|
# Removed an index?
|
||||||
if old_field.db_index and not new_field.db_index and not old_field.unique and not (not new_field.unique and old_field.unique):
|
if old_field.db_index and not new_field.db_index and not old_field.unique and not (not new_field.unique and old_field.unique):
|
||||||
# Find the index for this field
|
# Find the index for this field
|
||||||
|
@ -569,12 +517,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
old_field.column,
|
old_field.column,
|
||||||
))
|
))
|
||||||
for index_name in index_names:
|
for index_name in index_names:
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_index, model, index_name))
|
||||||
self.sql_delete_index % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": index_name,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Change check constraints?
|
# Change check constraints?
|
||||||
if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:
|
if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:
|
||||||
constraint_names = self._constraint_names(model, [old_field.column], check=True)
|
constraint_names = self._constraint_names(model, [old_field.column], check=True)
|
||||||
|
@ -585,12 +528,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
old_field.column,
|
old_field.column,
|
||||||
))
|
))
|
||||||
for constraint_name in constraint_names:
|
for constraint_name in constraint_names:
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_check, model, constraint_name))
|
||||||
self.sql_delete_check % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": constraint_name,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Have they renamed the column?
|
# Have they renamed the column?
|
||||||
if old_field.column != new_field.column:
|
if old_field.column != new_field.column:
|
||||||
self.execute(self.sql_rename_column % {
|
self.execute(self.sql_rename_column % {
|
||||||
|
@ -675,13 +613,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
self.execute(sql, params)
|
self.execute(sql, params)
|
||||||
# Added a unique?
|
# Added a unique?
|
||||||
if not old_field.unique and new_field.unique:
|
if not old_field.unique and new_field.unique:
|
||||||
self.execute(
|
self.execute(self._create_unique_sql(model, [new_field.column]))
|
||||||
self.sql_create_unique % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": self._create_index_name(model, [new_field.column], suffix="_uniq"),
|
|
||||||
"columns": self.quote_name(new_field.column),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Added an index?
|
# Added an index?
|
||||||
if not old_field.db_index and new_field.db_index and not new_field.unique and not (not old_field.unique and new_field.unique):
|
if not old_field.db_index and new_field.db_index and not new_field.unique and not (not old_field.unique and new_field.unique):
|
||||||
self.execute(
|
self.execute(
|
||||||
|
@ -709,12 +641,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
model._meta.db_table,
|
model._meta.db_table,
|
||||||
))
|
))
|
||||||
for constraint_name in constraint_names:
|
for constraint_name in constraint_names:
|
||||||
self.execute(
|
self.execute(self._delete_constraint_sql(self.sql_delete_pk, model, constraint_name))
|
||||||
self.sql_delete_pk % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": constraint_name,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
# Make the new one
|
# Make the new one
|
||||||
self.execute(
|
self.execute(
|
||||||
self.sql_create_pk % {
|
self.sql_create_pk % {
|
||||||
|
@ -742,29 +669,11 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
if new_field.rel and \
|
if new_field.rel and \
|
||||||
(fks_dropped or (old_field.rel and not old_field.db_constraint)) and \
|
(fks_dropped or (old_field.rel and not old_field.db_constraint)) and \
|
||||||
new_field.db_constraint:
|
new_field.db_constraint:
|
||||||
to_table = new_field.rel.to._meta.db_table
|
self.execute(self._create_fk_sql(model, new_field, "_fk_%(to_table)s_%(to_column)s"))
|
||||||
to_column = new_field.rel.get_related_field().column
|
|
||||||
self.execute(
|
|
||||||
self.sql_create_fk % {
|
|
||||||
"table": self.quote_name(model._meta.db_table),
|
|
||||||
"name": self._create_index_name(model, [new_field.column], suffix="_fk_%s_%s" % (to_table, to_column)),
|
|
||||||
"column": self.quote_name(new_field.column),
|
|
||||||
"to_table": self.quote_name(to_table),
|
|
||||||
"to_column": self.quote_name(to_column),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Rebuild FKs that pointed to us if we previously had to drop them
|
# Rebuild FKs that pointed to us if we previously had to drop them
|
||||||
if old_field.primary_key and new_field.primary_key and old_type != new_type:
|
if old_field.primary_key and new_field.primary_key and old_type != new_type:
|
||||||
for rel in new_field.model._meta.get_all_related_objects():
|
for rel in new_field.model._meta.get_all_related_objects():
|
||||||
self.execute(
|
self.execute(self._create_fk_sql(rel.model, rel.field, "_fk"))
|
||||||
self.sql_create_fk % {
|
|
||||||
"table": self.quote_name(rel.model._meta.db_table),
|
|
||||||
"name": self._create_index_name(rel.model, [rel.field.column], suffix="_fk"),
|
|
||||||
"column": self.quote_name(rel.field.column),
|
|
||||||
"to_table": self.quote_name(model._meta.db_table),
|
|
||||||
"to_column": self.quote_name(new_field.column),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
# Does it have check constraints we need to add?
|
# Does it have check constraints we need to add?
|
||||||
if old_db_params['check'] != new_db_params['check'] and new_db_params['check']:
|
if old_db_params['check'] != new_db_params['check'] and new_db_params['check']:
|
||||||
self.execute(
|
self.execute(
|
||||||
|
@ -856,7 +765,49 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
index_name = "D%s" % index_name[:-1]
|
index_name = "D%s" % index_name[:-1]
|
||||||
return index_name
|
return index_name
|
||||||
|
|
||||||
def _constraint_names(self, model, column_names=None, unique=None, primary_key=None, index=None, foreign_key=None, check=None):
|
def _create_index_sql(self, model, fields, suffix=""):
|
||||||
|
columns = [field.column for field in fields]
|
||||||
|
return self.sql_create_index % {
|
||||||
|
"table": self.quote_name(model._meta.db_table),
|
||||||
|
"name": self._create_index_name(model, columns, suffix=suffix),
|
||||||
|
"columns": ", ".join(self.quote_name(column) for column in columns),
|
||||||
|
"extra": "",
|
||||||
|
}
|
||||||
|
|
||||||
|
def _create_fk_sql(self, model, field, suffix):
|
||||||
|
from_table = model._meta.db_table
|
||||||
|
from_column = field.column
|
||||||
|
to_table = field.related_field.model._meta.db_table
|
||||||
|
to_column = field.related_field.column
|
||||||
|
suffix = suffix % {
|
||||||
|
"to_table": to_table,
|
||||||
|
"to_column": to_column,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.sql_create_fk % {
|
||||||
|
"table": self.quote_name(from_table),
|
||||||
|
"name": self._create_index_name(model, [from_column], suffix=suffix),
|
||||||
|
"column": self.quote_name(from_column),
|
||||||
|
"to_table": self.quote_name(to_table),
|
||||||
|
"to_column": self.quote_name(to_column),
|
||||||
|
}
|
||||||
|
|
||||||
|
def _create_unique_sql(self, model, columns):
|
||||||
|
return self.sql_create_unique % {
|
||||||
|
"table": self.quote_name(model._meta.db_table),
|
||||||
|
"name": self._create_index_name(model, columns, suffix="_uniq"),
|
||||||
|
"columns": ", ".join(self.quote_name(column) for column in columns),
|
||||||
|
}
|
||||||
|
|
||||||
|
def _delete_constraint_sql(self, template, model, name):
|
||||||
|
return template % {
|
||||||
|
"table": self.quote_name(model._meta.db_table),
|
||||||
|
"name": name,
|
||||||
|
}
|
||||||
|
|
||||||
|
def _constraint_names(self, model, column_names=None, unique=None,
|
||||||
|
primary_key=None, index=None, foreign_key=None,
|
||||||
|
check=None):
|
||||||
"""
|
"""
|
||||||
Returns all constraint names matching the columns and conditions
|
Returns all constraint names matching the columns and conditions
|
||||||
"""
|
"""
|
||||||
|
|
Loading…
Reference in New Issue