schema.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. from django.db.backends.mysql import schema as myschema
  2. class DatabaseSchemaEditor(myschema.DatabaseSchemaEditor):
  3. sql_delete_table = "DROP TABLE %(table)s"
  4. sql_delete_column = "ALTER TABLE %(table)s DROP COLUMN %(column)s"
  5. sql_create_table = "CREATE TABLE %(table)s (%(definition)s) AUTO_INCREMENT=10000"
  6. def remove_field(self, model, field):
  7. # Drop any Index, TiDB requires explicite deletion
  8. if field.db_index:
  9. idx_names = self._constraint_names(model, [field.column], index=True)
  10. for idx_name in idx_names:
  11. self.execute(self._delete_constraint_sql(self.sql_delete_index, model, idx_name))
  12. super(DatabaseSchemaEditor, self).remove_field(model, field)
  13. def column_sql(self, model, field, include_default=False):
  14. """
  15. Take a field and return its column definition.
  16. The field must already have had set_attributes_from_name() called.
  17. """
  18. # Get the column's type and use that as the basis of the SQL
  19. db_params = field.db_parameters(connection=self.connection)
  20. sql = db_params['type']
  21. params = []
  22. # Check for fields that aren't actually columns (e.g. M2M)
  23. if sql is None:
  24. return None, None
  25. # Work out nullability
  26. null = field.null
  27. # If we were told to include a default value, do so
  28. include_default = include_default and not self.skip_default(field)
  29. if include_default:
  30. default_value = self.effective_default(field)
  31. if default_value is not None:
  32. if self.connection.features.requires_literal_defaults:
  33. # Some databases can't take defaults as a parameter (oracle)
  34. # If this is the case, the individual schema backend should
  35. # implement prepare_default
  36. sql += " DEFAULT %s" % self.prepare_default(default_value)
  37. else:
  38. sql += " DEFAULT %s"
  39. params += [default_value]
  40. # Oracle treats the empty string ('') as null, so coerce the null
  41. # option whenever '' is a possible value.
  42. if (field.empty_strings_allowed and not field.primary_key and
  43. self.connection.features.interprets_empty_strings_as_nulls):
  44. null = True
  45. if null and not self.connection.features.implied_column_null:
  46. sql += " NULL"
  47. elif not null:
  48. sql += " NOT NULL"
  49. # Primary key/unique outputs
  50. if field.primary_key:
  51. sql += " PRIMARY KEY"
  52. # elif field.unique:
  53. # sql += " UNIQUE"
  54. # Optionally add the tablespace if it's an implicitly indexed column
  55. tablespace = field.db_tablespace or model._meta.db_tablespace
  56. if tablespace and self.connection.features.supports_tablespaces and field.unique:
  57. sql += " %s" % self.connection.ops.tablespace_sql(tablespace, inline=True)
  58. # Return the sql
  59. return sql, params
  60. def create_model(self, model):
  61. """
  62. Create a table and any accompanying indexes or unique constraints for
  63. the given `model`.
  64. """
  65. # Create column SQL, add FK deferreds if needed
  66. column_sqls = []
  67. params = []
  68. for field in model._meta.local_fields:
  69. # SQL
  70. definition, extra_params = self.column_sql(model, field)
  71. if definition is None:
  72. continue
  73. # Check constraints can go on the column SQL here
  74. db_params = field.db_parameters(connection=self.connection)
  75. if db_params['check']:
  76. definition += " CHECK (%s)" % db_params['check']
  77. # Autoincrement SQL (for backends with inline variant)
  78. col_type_suffix = field.db_type_suffix(connection=self.connection)
  79. if col_type_suffix:
  80. definition += " %s" % col_type_suffix
  81. params.extend(extra_params)
  82. # FK
  83. if field.remote_field and field.db_constraint:
  84. to_table = field.remote_field.model._meta.db_table
  85. to_column = field.remote_field.model._meta.get_field(field.remote_field.field_name).column
  86. if self.sql_create_inline_fk:
  87. definition += " " + self.sql_create_inline_fk % {
  88. "to_table": self.quote_name(to_table),
  89. "to_column": self.quote_name(to_column),
  90. }
  91. elif self.connection.features.supports_foreign_keys:
  92. self.deferred_sql.append(self._create_fk_sql(model, field, "_fk_%(to_table)s_%(to_column)s"))
  93. # Add the SQL to our big list
  94. column_sqls.append("%s %s" % (
  95. self.quote_name(field.column),
  96. definition,
  97. ))
  98. # Autoincrement SQL (for backends with post table definition variant)
  99. if field.get_internal_type() in ("AutoField", "BigAutoField"):
  100. autoinc_sql = self.connection.ops.autoinc_sql(model._meta.db_table, field.column)
  101. if autoinc_sql:
  102. self.deferred_sql.extend(autoinc_sql)
  103. # Add any unique_togethers (always deferred, as some fields might be
  104. # created afterwards, like geometry fields with some backends)
  105. for fields in model._meta.unique_together:
  106. columns = [model._meta.get_field(field).column for field in fields]
  107. self.deferred_sql.append(self._create_unique_sql(model, columns))
  108. # Make the table
  109. sql = self.sql_create_table % {
  110. "table": self.quote_name(model._meta.db_table),
  111. "definition": ", ".join(column_sqls)
  112. }
  113. if model._meta.db_tablespace:
  114. tablespace_sql = self.connection.ops.tablespace_sql(model._meta.db_tablespace)
  115. if tablespace_sql:
  116. sql += ' ' + tablespace_sql
  117. # Prevent using [] as params, in the case a literal '%' is used in the definition
  118. self.execute(sql, params or None)
  119. # Add any field index and index_together's (deferred as SQLite3 _remake_table needs it)
  120. self.deferred_sql.extend(self._model_indexes_sql(model))
  121. # Make M2M tables
  122. for field in model._meta.local_many_to_many:
  123. if field.remote_field.through._meta.auto_created:
  124. self.create_model(field.remote_field.through)