Fixed #28275 -- Added more hooks to SchemaEditor._alter_field().
This commit is contained in:
parent
5e9f7f1e1c
commit
823d73be3e
|
@ -407,14 +407,12 @@ class BaseDatabaseSchemaEditor:
|
||||||
# Drop the default if we need to
|
# Drop the default if we need to
|
||||||
# (Django usually does not use in-database defaults)
|
# (Django usually does not use in-database defaults)
|
||||||
if not self.skip_default(field) and self.effective_default(field) is not None:
|
if not self.skip_default(field) and self.effective_default(field) is not None:
|
||||||
|
changes_sql, params = self._alter_column_default_sql(model, None, field, drop=True)
|
||||||
sql = self.sql_alter_column % {
|
sql = self.sql_alter_column % {
|
||||||
"table": self.quote_name(model._meta.db_table),
|
"table": self.quote_name(model._meta.db_table),
|
||||||
"changes": self.sql_alter_column_no_default % {
|
"changes": changes_sql,
|
||||||
"column": self.quote_name(field.column),
|
|
||||||
"type": db_params['type'],
|
|
||||||
}
|
}
|
||||||
}
|
self.execute(sql, params)
|
||||||
self.execute(sql)
|
|
||||||
# Add an index, if required
|
# Add an index, if required
|
||||||
self.deferred_sql.extend(self._field_indexes_sql(model, field))
|
self.deferred_sql.extend(self._field_indexes_sql(model, field))
|
||||||
# Add any FK constraints later
|
# Add any FK constraints later
|
||||||
|
@ -573,9 +571,7 @@ class BaseDatabaseSchemaEditor:
|
||||||
post_actions = []
|
post_actions = []
|
||||||
# Type change?
|
# Type change?
|
||||||
if old_type != new_type:
|
if old_type != new_type:
|
||||||
fragment, other_actions = self._alter_column_type_sql(
|
fragment, other_actions = self._alter_column_type_sql(model, old_field, new_field, new_type)
|
||||||
model._meta.db_table, old_field, new_field, new_type
|
|
||||||
)
|
|
||||||
actions.append(fragment)
|
actions.append(fragment)
|
||||||
post_actions.extend(other_actions)
|
post_actions.extend(other_actions)
|
||||||
# When changing a column NULL constraint to NOT NULL with a given
|
# When changing a column NULL constraint to NOT NULL with a given
|
||||||
|
@ -595,49 +591,12 @@ class BaseDatabaseSchemaEditor:
|
||||||
not self.skip_default(new_field)
|
not self.skip_default(new_field)
|
||||||
)
|
)
|
||||||
if needs_database_default:
|
if needs_database_default:
|
||||||
if self.connection.features.requires_literal_defaults:
|
actions.append(self._alter_column_default_sql(model, old_field, new_field))
|
||||||
# Some databases can't take defaults as a parameter (oracle)
|
|
||||||
# If this is the case, the individual schema backend should
|
|
||||||
# implement prepare_default
|
|
||||||
actions.append((
|
|
||||||
self.sql_alter_column_default % {
|
|
||||||
"column": self.quote_name(new_field.column),
|
|
||||||
"type": new_type,
|
|
||||||
"default": self.prepare_default(new_default),
|
|
||||||
},
|
|
||||||
[],
|
|
||||||
))
|
|
||||||
else:
|
|
||||||
actions.append((
|
|
||||||
self.sql_alter_column_default % {
|
|
||||||
"column": self.quote_name(new_field.column),
|
|
||||||
"type": new_type,
|
|
||||||
"default": "%s",
|
|
||||||
},
|
|
||||||
[new_default],
|
|
||||||
))
|
|
||||||
# Nullability change?
|
# Nullability change?
|
||||||
if old_field.null != new_field.null:
|
if old_field.null != new_field.null:
|
||||||
if (self.connection.features.interprets_empty_strings_as_nulls and
|
fragment = self._alter_column_null_sql(model, old_field, new_field)
|
||||||
new_field.get_internal_type() in ("CharField", "TextField")):
|
if fragment:
|
||||||
# The field is nullable in the database anyway, leave it alone
|
null_actions.append(fragment)
|
||||||
pass
|
|
||||||
elif new_field.null:
|
|
||||||
null_actions.append((
|
|
||||||
self.sql_alter_column_null % {
|
|
||||||
"column": self.quote_name(new_field.column),
|
|
||||||
"type": new_type,
|
|
||||||
},
|
|
||||||
[],
|
|
||||||
))
|
|
||||||
else:
|
|
||||||
null_actions.append((
|
|
||||||
self.sql_alter_column_not_null % {
|
|
||||||
"column": self.quote_name(new_field.column),
|
|
||||||
"type": new_type,
|
|
||||||
},
|
|
||||||
[],
|
|
||||||
))
|
|
||||||
# Only if we have a default and there is a change from NULL to NOT NULL
|
# Only if we have a default and there is a change from NULL to NOT NULL
|
||||||
four_way_default_alteration = (
|
four_way_default_alteration = (
|
||||||
new_field.has_default() and
|
new_field.has_default() and
|
||||||
|
@ -726,7 +685,7 @@ class BaseDatabaseSchemaEditor:
|
||||||
rel_db_params = new_rel.field.db_parameters(connection=self.connection)
|
rel_db_params = new_rel.field.db_parameters(connection=self.connection)
|
||||||
rel_type = rel_db_params['type']
|
rel_type = rel_db_params['type']
|
||||||
fragment, other_actions = self._alter_column_type_sql(
|
fragment, other_actions = self._alter_column_type_sql(
|
||||||
new_rel.related_model._meta.db_table, old_rel.field, new_rel.field, rel_type
|
new_rel.related_model, old_rel.field, new_rel.field, rel_type
|
||||||
)
|
)
|
||||||
self.execute(
|
self.execute(
|
||||||
self.sql_alter_column % {
|
self.sql_alter_column % {
|
||||||
|
@ -760,19 +719,70 @@ class BaseDatabaseSchemaEditor:
|
||||||
# Drop the default if we need to
|
# Drop the default if we need to
|
||||||
# (Django usually does not use in-database defaults)
|
# (Django usually does not use in-database defaults)
|
||||||
if needs_database_default:
|
if needs_database_default:
|
||||||
|
changes_sql, params = self._alter_column_default_sql(model, old_field, new_field, drop=True)
|
||||||
sql = self.sql_alter_column % {
|
sql = self.sql_alter_column % {
|
||||||
"table": self.quote_name(model._meta.db_table),
|
"table": self.quote_name(model._meta.db_table),
|
||||||
"changes": self.sql_alter_column_no_default % {
|
"changes": changes_sql,
|
||||||
"column": self.quote_name(new_field.column),
|
|
||||||
"type": new_type,
|
|
||||||
}
|
}
|
||||||
}
|
self.execute(sql, params)
|
||||||
self.execute(sql)
|
|
||||||
# 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()
|
||||||
|
|
||||||
def _alter_column_type_sql(self, table, old_field, new_field, new_type):
|
def _alter_column_null_sql(self, model, old_field, new_field):
|
||||||
|
"""
|
||||||
|
Hook to specialize column null alteration.
|
||||||
|
|
||||||
|
Return a (sql, params) fragment to set a column to null or non-null
|
||||||
|
as required by new_field, or None if no changes are required.
|
||||||
|
"""
|
||||||
|
if (self.connection.features.interprets_empty_strings_as_nulls and
|
||||||
|
new_field.get_internal_type() in ("CharField", "TextField")):
|
||||||
|
# The field is nullable in the database anyway, leave it alone.
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
new_db_params = new_field.db_parameters(connection=self.connection)
|
||||||
|
sql = self.sql_alter_column_null if new_field.null else self.sql_alter_column_not_null
|
||||||
|
return (
|
||||||
|
sql % {
|
||||||
|
'column': self.quote_name(new_field.column),
|
||||||
|
'type': new_db_params['type'],
|
||||||
|
},
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
|
||||||
|
def _alter_column_default_sql(self, model, old_field, new_field, drop=False):
|
||||||
|
"""
|
||||||
|
Hook to specialize column default alteration.
|
||||||
|
|
||||||
|
Return a (sql, params) fragment to add or drop (depending on the drop
|
||||||
|
argument) a default to new_field's column.
|
||||||
|
"""
|
||||||
|
new_default = self.effective_default(new_field)
|
||||||
|
default = '%s'
|
||||||
|
params = [new_default]
|
||||||
|
|
||||||
|
if drop:
|
||||||
|
params = []
|
||||||
|
elif self.connection.features.requires_literal_defaults:
|
||||||
|
# Some databases (Oracle) can't take defaults as a parameter
|
||||||
|
# If this is the case, the SchemaEditor for that database should
|
||||||
|
# implement prepare_default().
|
||||||
|
default = self.prepare_default(new_default)
|
||||||
|
params = []
|
||||||
|
|
||||||
|
new_db_params = new_field.db_parameters(connection=self.connection)
|
||||||
|
sql = self.sql_alter_column_no_default if drop else self.sql_alter_column_default
|
||||||
|
return (
|
||||||
|
sql % {
|
||||||
|
'column': self.quote_name(new_field.column),
|
||||||
|
'type': new_db_params['type'],
|
||||||
|
'default': default,
|
||||||
|
},
|
||||||
|
params,
|
||||||
|
)
|
||||||
|
|
||||||
|
def _alter_column_type_sql(self, model, old_field, new_field, new_type):
|
||||||
"""
|
"""
|
||||||
Hook to specialize column type alteration for different backends,
|
Hook to specialize column type alteration for different backends,
|
||||||
for cases when a creation type is different to an alteration type
|
for cases when a creation type is different to an alteration type
|
||||||
|
|
|
@ -92,9 +92,9 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
new_type += " NOT NULL"
|
new_type += " NOT NULL"
|
||||||
return new_type
|
return new_type
|
||||||
|
|
||||||
def _alter_column_type_sql(self, table, old_field, new_field, new_type):
|
def _alter_column_type_sql(self, model, old_field, new_field, new_type):
|
||||||
new_type = self._set_field_new_type_null_status(old_field, new_type)
|
new_type = self._set_field_new_type_null_status(old_field, new_type)
|
||||||
return super()._alter_column_type_sql(table, old_field, new_field, new_type)
|
return super()._alter_column_type_sql(model, old_field, new_field, new_type)
|
||||||
|
|
||||||
def _rename_field_sql(self, table, old_field, new_field, new_type):
|
def _rename_field_sql(self, table, old_field, new_field, new_type):
|
||||||
new_type = self._set_field_new_type_null_status(old_field, new_type)
|
new_type = self._set_field_new_type_null_status(old_field, new_type)
|
||||||
|
|
|
@ -52,8 +52,9 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
return self._create_index_sql(model, [field], suffix='_like', sql=self.sql_create_text_index)
|
return self._create_index_sql(model, [field], suffix='_like', sql=self.sql_create_text_index)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def _alter_column_type_sql(self, table, old_field, new_field, new_type):
|
def _alter_column_type_sql(self, model, old_field, new_field, new_type):
|
||||||
"""Make ALTER TYPE with SERIAL make sense."""
|
"""Make ALTER TYPE with SERIAL make sense."""
|
||||||
|
table = model._meta.db_table
|
||||||
if new_type.lower() in ("serial", "bigserial"):
|
if new_type.lower() in ("serial", "bigserial"):
|
||||||
column = new_field.column
|
column = new_field.column
|
||||||
sequence_name = "%s_%s_seq" % (table, column)
|
sequence_name = "%s_%s_seq" % (table, column)
|
||||||
|
@ -100,7 +101,7 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return super()._alter_column_type_sql(table, old_field, new_field, new_type)
|
return super()._alter_column_type_sql(model, old_field, new_field, new_type)
|
||||||
|
|
||||||
def _alter_field(self, model, old_field, new_field, old_type, new_type,
|
def _alter_field(self, model, old_field, new_field, old_type, new_type,
|
||||||
old_db_params, new_db_params, strict=False):
|
old_db_params, new_db_params, strict=False):
|
||||||
|
|
|
@ -289,6 +289,9 @@ Database backend API
|
||||||
attribute with the name of the database that your backend works with. Django
|
attribute with the name of the database that your backend works with. Django
|
||||||
may use it in various messages, such as in system checks.
|
may use it in various messages, such as in system checks.
|
||||||
|
|
||||||
|
* The first argument of ``SchemaEditor._alter_column_type_sql()`` is now
|
||||||
|
``model`` rather than ``table``.
|
||||||
|
|
||||||
* To improve performance when streaming large result sets from the database,
|
* To improve performance when streaming large result sets from the database,
|
||||||
:meth:`.QuerySet.iterator` now fetches 2000 rows at a time instead of 100.
|
:meth:`.QuerySet.iterator` now fetches 2000 rows at a time instead of 100.
|
||||||
The old behavior can be restored using the ``chunk_size`` parameter. For
|
The old behavior can be restored using the ``chunk_size`` parameter. For
|
||||||
|
|
Loading…
Reference in New Issue