All tests passing on MySQL
This commit is contained in:
parent
cab044c66c
commit
f7955c703d
|
@ -105,32 +105,42 @@ class DatabaseIntrospection(BaseDatabaseIntrospection):
|
||||||
def get_constraints(self, cursor, table_name):
|
def get_constraints(self, cursor, table_name):
|
||||||
"""
|
"""
|
||||||
Retrieves any constraints (unique, pk, fk, check) across one or more columns.
|
Retrieves any constraints (unique, pk, fk, check) across one or more columns.
|
||||||
Returns {'cnname': {'columns': set(columns), 'primary_key': bool, 'unique': bool}}
|
Returns {'cnname': {'columns': set(columns), 'primary_key': bool, 'unique': bool, 'foreign_key': None|(tbl, col)}}
|
||||||
"""
|
"""
|
||||||
constraints = {}
|
constraints = {}
|
||||||
# Loop over the constraint tables, collecting things as constraints
|
# Get the actual constraint names and columns
|
||||||
ifsc_tables = ["constraint_column_usage", "key_column_usage"]
|
name_query = """
|
||||||
for ifsc_table in ifsc_tables:
|
SELECT kc.`constraint_name`, kc.`column_name`,
|
||||||
cursor.execute("""
|
kc.`referenced_table_name`, kc.`referenced_column_name`
|
||||||
SELECT kc.constraint_name, kc.column_name, c.constraint_type
|
FROM information_schema.key_column_usage AS kc
|
||||||
FROM information_schema.%s AS kc
|
|
||||||
JOIN information_schema.table_constraints AS c ON
|
|
||||||
kc.table_schema = c.table_schema AND
|
|
||||||
kc.table_name = c.table_name AND
|
|
||||||
kc.constraint_name = c.constraint_name
|
|
||||||
WHERE
|
WHERE
|
||||||
kc.table_schema = %%s AND
|
kc.table_schema = %s AND
|
||||||
kc.table_name = %%s
|
kc.table_name = %s
|
||||||
""" % ifsc_table, [self.connection.settings_dict['NAME'], table_name])
|
"""
|
||||||
for constraint, column, kind in cursor.fetchall():
|
cursor.execute(name_query, [self.connection.settings_dict['NAME'], table_name])
|
||||||
# If we're the first column, make the record
|
for constraint, column, ref_table, ref_column in cursor.fetchall():
|
||||||
if constraint not in constraints:
|
if constraint not in constraints:
|
||||||
constraints[constraint] = {
|
constraints[constraint] = {
|
||||||
"columns": set(),
|
'columns': set(),
|
||||||
"primary_key": kind.lower() == "primary key",
|
'primary_key': False,
|
||||||
"foreign_key": kind.lower() == "foreign key",
|
'unique': False,
|
||||||
"unique": kind.lower() in ["primary key", "unique"],
|
'foreign_key': (ref_table, ref_column) if ref_column else None,
|
||||||
}
|
}
|
||||||
# Record the details
|
|
||||||
constraints[constraint]['columns'].add(column)
|
constraints[constraint]['columns'].add(column)
|
||||||
|
# Now get the constraint types
|
||||||
|
type_query = """
|
||||||
|
SELECT c.constraint_name, c.constraint_type
|
||||||
|
FROM information_schema.table_constraints AS c
|
||||||
|
WHERE
|
||||||
|
c.table_schema = %s AND
|
||||||
|
c.table_name = %s
|
||||||
|
"""
|
||||||
|
cursor.execute(type_query, [self.connection.settings_dict['NAME'], table_name])
|
||||||
|
for constraint, kind in cursor.fetchall():
|
||||||
|
if kind.lower() == "primary key":
|
||||||
|
constraints[constraint]['primary_key'] = True
|
||||||
|
constraints[constraint]['unique'] = True
|
||||||
|
elif kind.lower() == "unique":
|
||||||
|
constraints[constraint]['unique'] = True
|
||||||
|
# Return
|
||||||
return constraints
|
return constraints
|
||||||
|
|
|
@ -7,6 +7,8 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
|
|
||||||
sql_alter_column_null = "MODIFY %(column)s %(type)s NULL"
|
sql_alter_column_null = "MODIFY %(column)s %(type)s NULL"
|
||||||
sql_alter_column_not_null = "MODIFY %(column)s %(type)s NULL"
|
sql_alter_column_not_null = "MODIFY %(column)s %(type)s NULL"
|
||||||
|
sql_alter_column_type = "MODIFY %(column)s %(type)s"
|
||||||
|
sql_rename_column = "ALTER TABLE %(table)s CHANGE %(old_column)s %(new_column)s %(type)s"
|
||||||
|
|
||||||
sql_delete_unique = "ALTER TABLE %(table)s DROP INDEX %(name)s"
|
sql_delete_unique = "ALTER TABLE %(table)s DROP INDEX %(name)s"
|
||||||
|
|
||||||
|
@ -17,8 +19,5 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
|
|
||||||
sql_delete_pk = "ALTER TABLE %(table)s DROP PRIMARY KEY"
|
sql_delete_pk = "ALTER TABLE %(table)s DROP PRIMARY KEY"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
alter_string_set_null = 'MODIFY %(column)s %(type)s NULL;'
|
alter_string_set_null = 'MODIFY %(column)s %(type)s NULL;'
|
||||||
alter_string_drop_null = 'MODIFY %(column)s %(type)s NOT NULL;'
|
alter_string_drop_null = 'MODIFY %(column)s %(type)s NOT NULL;'
|
||||||
|
|
|
@ -25,6 +25,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
- Repointing of M2Ms
|
- Repointing of M2Ms
|
||||||
- Check constraints (PosIntField)
|
- Check constraints (PosIntField)
|
||||||
- PK changing
|
- PK changing
|
||||||
|
- db_index on alter field
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Overrideable SQL templates
|
# Overrideable SQL templates
|
||||||
|
@ -358,7 +359,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
"""
|
"""
|
||||||
# Ensure this field is even column-based
|
# Ensure this field is even column-based
|
||||||
old_type = old_field.db_type(connection=self.connection)
|
old_type = old_field.db_type(connection=self.connection)
|
||||||
new_type = new_field.db_type(connection=self.connection)
|
new_type = self._type_for_alter(new_field)
|
||||||
if old_type is None and new_type is None:
|
if old_type is None and new_type is None:
|
||||||
# TODO: Handle M2M fields being repointed
|
# TODO: Handle M2M fields being repointed
|
||||||
return
|
return
|
||||||
|
@ -389,6 +390,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
"table": self.quote_name(model._meta.db_table),
|
"table": self.quote_name(model._meta.db_table),
|
||||||
"old_column": self.quote_name(old_field.column),
|
"old_column": self.quote_name(old_field.column),
|
||||||
"new_column": self.quote_name(new_field.column),
|
"new_column": self.quote_name(new_field.column),
|
||||||
|
"type": new_type,
|
||||||
})
|
})
|
||||||
# Next, start accumulating actions to do
|
# Next, start accumulating actions to do
|
||||||
actions = []
|
actions = []
|
||||||
|
@ -426,6 +428,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
actions.append((
|
actions.append((
|
||||||
self.sql_alter_column_null % {
|
self.sql_alter_column_null % {
|
||||||
"column": self.quote_name(new_field.column),
|
"column": self.quote_name(new_field.column),
|
||||||
|
"type": new_type,
|
||||||
},
|
},
|
||||||
[],
|
[],
|
||||||
))
|
))
|
||||||
|
@ -433,6 +436,7 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
actions.append((
|
actions.append((
|
||||||
self.sql_alter_column_null % {
|
self.sql_alter_column_null % {
|
||||||
"column": self.quote_name(new_field.column),
|
"column": self.quote_name(new_field.column),
|
||||||
|
"type": new_type,
|
||||||
},
|
},
|
||||||
[],
|
[],
|
||||||
))
|
))
|
||||||
|
@ -460,6 +464,14 @@ class BaseDatabaseSchemaEditor(object):
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def _type_for_alter(self, field):
|
||||||
|
"""
|
||||||
|
Returns a field's type suitable for ALTER COLUMN.
|
||||||
|
By default it just returns field.db_type().
|
||||||
|
To be overriden by backend specific subclasses
|
||||||
|
"""
|
||||||
|
return field.db_type(connection=self.connection)
|
||||||
|
|
||||||
def _create_index_name(self, model, column_names, suffix=""):
|
def _create_index_name(self, model, column_names, suffix=""):
|
||||||
"Generates a unique name for an index/unique constraint."
|
"Generates a unique name for an index/unique constraint."
|
||||||
# If there is just one column in the index, use a default algorithm from Django
|
# If there is just one column in the index, use a default algorithm from Django
|
||||||
|
|
|
@ -39,6 +39,7 @@ class SchemaTests(TestCase):
|
||||||
connection.rollback()
|
connection.rollback()
|
||||||
# Delete any tables made for our models
|
# Delete any tables made for our models
|
||||||
cursor = connection.cursor()
|
cursor = connection.cursor()
|
||||||
|
connection.disable_constraint_checking()
|
||||||
for model in self.models:
|
for model in self.models:
|
||||||
# Remove any M2M tables first
|
# Remove any M2M tables first
|
||||||
for field in model._meta.local_many_to_many:
|
for field in model._meta.local_many_to_many:
|
||||||
|
@ -59,6 +60,7 @@ class SchemaTests(TestCase):
|
||||||
connection.rollback()
|
connection.rollback()
|
||||||
else:
|
else:
|
||||||
connection.commit()
|
connection.commit()
|
||||||
|
connection.enable_constraint_checking()
|
||||||
# Unhook our models
|
# Unhook our models
|
||||||
for model in self.models:
|
for model in self.models:
|
||||||
model._meta.managed = False
|
model._meta.managed = False
|
||||||
|
|
Loading…
Reference in New Issue