Fixed #29547 -- Added support for partial indexes.
Thanks to Ian Foote, Mariusz Felisiak, Simon Charettes, and Markus Holtermann for comments and feedback.
This commit is contained in:
parent
9625d13f7b
commit
a906c98982
|
@ -38,6 +38,7 @@ class PostGISSchemaEditor(DatabaseSchemaEditor):
|
||||||
"using": "USING %s" % self.geom_index_type,
|
"using": "USING %s" % self.geom_index_type,
|
||||||
"columns": field_column,
|
"columns": field_column,
|
||||||
"extra": '',
|
"extra": '',
|
||||||
|
"condition": '',
|
||||||
}
|
}
|
||||||
|
|
||||||
def _alter_column_type_sql(self, table, old_field, new_field, new_type):
|
def _alter_column_type_sql(self, table, old_field, new_field, new_type):
|
||||||
|
|
|
@ -276,6 +276,9 @@ class BaseDatabaseFeatures:
|
||||||
# in UPDATE statements to ensure the expression has the correct type?
|
# in UPDATE statements to ensure the expression has the correct type?
|
||||||
requires_casted_case_in_updates = False
|
requires_casted_case_in_updates = False
|
||||||
|
|
||||||
|
# Does the backend support partial indexes (CREATE INDEX ... WHERE ...)?
|
||||||
|
supports_partial_indexes = True
|
||||||
|
|
||||||
def __init__(self, connection):
|
def __init__(self, connection):
|
||||||
self.connection = connection
|
self.connection = connection
|
||||||
|
|
||||||
|
|
|
@ -75,7 +75,7 @@ class BaseDatabaseSchemaEditor:
|
||||||
sql_create_inline_fk = None
|
sql_create_inline_fk = None
|
||||||
sql_delete_fk = "ALTER TABLE %(table)s DROP CONSTRAINT %(name)s"
|
sql_delete_fk = "ALTER TABLE %(table)s DROP CONSTRAINT %(name)s"
|
||||||
|
|
||||||
sql_create_index = "CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s"
|
sql_create_index = "CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s%(condition)s"
|
||||||
sql_delete_index = "DROP INDEX %(name)s"
|
sql_delete_index = "DROP INDEX %(name)s"
|
||||||
|
|
||||||
sql_create_pk = "ALTER TABLE %(table)s ADD CONSTRAINT %(name)s PRIMARY KEY (%(columns)s)"
|
sql_create_pk = "ALTER TABLE %(table)s ADD CONSTRAINT %(name)s PRIMARY KEY (%(columns)s)"
|
||||||
|
@ -326,7 +326,7 @@ class BaseDatabaseSchemaEditor:
|
||||||
|
|
||||||
def add_index(self, model, index):
|
def add_index(self, model, index):
|
||||||
"""Add an index on a model."""
|
"""Add an index on a model."""
|
||||||
self.execute(index.create_sql(model, self))
|
self.execute(index.create_sql(model, self), params=None)
|
||||||
|
|
||||||
def remove_index(self, model, index):
|
def remove_index(self, model, index):
|
||||||
"""Remove an index from a model."""
|
"""Remove an index from a model."""
|
||||||
|
@ -905,7 +905,8 @@ class BaseDatabaseSchemaEditor:
|
||||||
return ''
|
return ''
|
||||||
|
|
||||||
def _create_index_sql(self, model, fields, *, name=None, suffix='', using='',
|
def _create_index_sql(self, model, fields, *, name=None, suffix='', using='',
|
||||||
db_tablespace=None, col_suffixes=(), sql=None, opclasses=()):
|
db_tablespace=None, col_suffixes=(), sql=None, opclasses=(),
|
||||||
|
condition=''):
|
||||||
"""
|
"""
|
||||||
Return the SQL statement to create the index for one or several fields.
|
Return the SQL statement to create the index for one or several fields.
|
||||||
`sql` can be specified if the syntax differs from the standard (GIS
|
`sql` can be specified if the syntax differs from the standard (GIS
|
||||||
|
@ -929,6 +930,7 @@ class BaseDatabaseSchemaEditor:
|
||||||
using=using,
|
using=using,
|
||||||
columns=self._index_columns(table, columns, col_suffixes, opclasses),
|
columns=self._index_columns(table, columns, col_suffixes, opclasses),
|
||||||
extra=tablespace_sql,
|
extra=tablespace_sql,
|
||||||
|
condition=condition,
|
||||||
)
|
)
|
||||||
|
|
||||||
def _index_columns(self, table, columns, col_suffixes, opclasses):
|
def _index_columns(self, table, columns, col_suffixes, opclasses):
|
||||||
|
|
|
@ -51,6 +51,8 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
||||||
db_functions_convert_bytes_to_str = True
|
db_functions_convert_bytes_to_str = True
|
||||||
# Alias MySQL's TRADITIONAL to TEXT for consistency with other backends.
|
# Alias MySQL's TRADITIONAL to TEXT for consistency with other backends.
|
||||||
supported_explain_formats = {'JSON', 'TEXT', 'TRADITIONAL'}
|
supported_explain_formats = {'JSON', 'TEXT', 'TRADITIONAL'}
|
||||||
|
# Neither MySQL nor MariaDB support partial indexes.
|
||||||
|
supports_partial_indexes = False
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def _mysql_storage_engine(self):
|
def _mysql_storage_engine(self):
|
||||||
|
|
|
@ -24,6 +24,8 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
sql_create_pk = "ALTER TABLE %(table)s ADD CONSTRAINT %(name)s PRIMARY KEY (%(columns)s)"
|
sql_create_pk = "ALTER TABLE %(table)s ADD CONSTRAINT %(name)s PRIMARY KEY (%(columns)s)"
|
||||||
sql_delete_pk = "ALTER TABLE %(table)s DROP PRIMARY KEY"
|
sql_delete_pk = "ALTER TABLE %(table)s DROP PRIMARY KEY"
|
||||||
|
|
||||||
|
sql_create_index = 'CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s'
|
||||||
|
|
||||||
def quote_value(self, value):
|
def quote_value(self, value):
|
||||||
self.connection.ensure_connection()
|
self.connection.ensure_connection()
|
||||||
quoted = self.connection.connection.escape(value, self.connection.connection.encoders)
|
quoted = self.connection.connection.escape(value, self.connection.connection.encoders)
|
||||||
|
|
|
@ -55,6 +55,7 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
||||||
supports_over_clause = True
|
supports_over_clause = True
|
||||||
supports_ignore_conflicts = False
|
supports_ignore_conflicts = False
|
||||||
max_query_params = 2**16 - 1
|
max_query_params = 2**16 - 1
|
||||||
|
supports_partial_indexes = False
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def has_fetch_offset_support(self):
|
def has_fetch_offset_support(self):
|
||||||
|
|
|
@ -16,6 +16,7 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
sql_alter_column_no_default = "MODIFY %(column)s DEFAULT NULL"
|
sql_alter_column_no_default = "MODIFY %(column)s DEFAULT NULL"
|
||||||
sql_delete_column = "ALTER TABLE %(table)s DROP COLUMN %(column)s"
|
sql_delete_column = "ALTER TABLE %(table)s DROP COLUMN %(column)s"
|
||||||
sql_delete_table = "DROP TABLE %(table)s CASCADE CONSTRAINTS"
|
sql_delete_table = "DROP TABLE %(table)s CASCADE CONSTRAINTS"
|
||||||
|
sql_create_index = "CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s"
|
||||||
|
|
||||||
def quote_value(self, value):
|
def quote_value(self, value):
|
||||||
if isinstance(value, (datetime.date, datetime.time, datetime.datetime)):
|
if isinstance(value, (datetime.date, datetime.time, datetime.datetime)):
|
||||||
|
|
|
@ -12,7 +12,7 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
||||||
sql_delete_sequence = "DROP SEQUENCE IF EXISTS %(sequence)s CASCADE"
|
sql_delete_sequence = "DROP SEQUENCE IF EXISTS %(sequence)s CASCADE"
|
||||||
sql_set_sequence_max = "SELECT setval('%(sequence)s', MAX(%(column)s)) FROM %(table)s"
|
sql_set_sequence_max = "SELECT setval('%(sequence)s', MAX(%(column)s)) FROM %(table)s"
|
||||||
|
|
||||||
sql_create_index = "CREATE INDEX %(name)s ON %(table)s%(using)s (%(columns)s)%(extra)s"
|
sql_create_index = "CREATE INDEX %(name)s ON %(table)s%(using)s (%(columns)s)%(extra)s%(condition)s"
|
||||||
sql_delete_index = "DROP INDEX IF EXISTS %(name)s"
|
sql_delete_index = "DROP INDEX IF EXISTS %(name)s"
|
||||||
|
|
||||||
# Setting the constraint to IMMEDIATE runs any deferred checks to allow
|
# Setting the constraint to IMMEDIATE runs any deferred checks to allow
|
||||||
|
|
|
@ -34,6 +34,7 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
||||||
supports_cast_with_precision = False
|
supports_cast_with_precision = False
|
||||||
time_cast_precision = 3
|
time_cast_precision = 3
|
||||||
can_release_savepoints = True
|
can_release_savepoints = True
|
||||||
|
supports_partial_indexes = Database.version_info >= (3, 8, 0)
|
||||||
# Is "ALTER TABLE ... RENAME COLUMN" supported?
|
# Is "ALTER TABLE ... RENAME COLUMN" supported?
|
||||||
can_alter_table_rename_column = Database.sqlite_version_info >= (3, 25, 0)
|
can_alter_table_rename_column = Database.sqlite_version_info >= (3, 25, 0)
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
from django.db.backends.utils import names_digest, split_identifier
|
from django.db.backends.utils import names_digest, split_identifier
|
||||||
|
from django.db.models.query_utils import Q
|
||||||
|
from django.db.models.sql import Query
|
||||||
|
|
||||||
__all__ = ['Index']
|
__all__ = ['Index']
|
||||||
|
|
||||||
|
@ -9,9 +11,13 @@ class Index:
|
||||||
# cross-database compatibility with Oracle)
|
# cross-database compatibility with Oracle)
|
||||||
max_name_length = 30
|
max_name_length = 30
|
||||||
|
|
||||||
def __init__(self, *, fields=(), name=None, db_tablespace=None, opclasses=()):
|
def __init__(self, *, fields=(), name=None, db_tablespace=None, opclasses=(), condition=None):
|
||||||
if opclasses and not name:
|
if opclasses and not name:
|
||||||
raise ValueError('An index must be named to use opclasses.')
|
raise ValueError('An index must be named to use opclasses.')
|
||||||
|
if not isinstance(condition, (type(None), Q)):
|
||||||
|
raise ValueError('Index.condition must be a Q instance.')
|
||||||
|
if condition and not name:
|
||||||
|
raise ValueError('An index must be named to use condition.')
|
||||||
if not isinstance(fields, (list, tuple)):
|
if not isinstance(fields, (list, tuple)):
|
||||||
raise ValueError('Index.fields must be a list or tuple.')
|
raise ValueError('Index.fields must be a list or tuple.')
|
||||||
if not isinstance(opclasses, (list, tuple)):
|
if not isinstance(opclasses, (list, tuple)):
|
||||||
|
@ -35,6 +41,7 @@ class Index:
|
||||||
raise ValueError(errors)
|
raise ValueError(errors)
|
||||||
self.db_tablespace = db_tablespace
|
self.db_tablespace = db_tablespace
|
||||||
self.opclasses = opclasses
|
self.opclasses = opclasses
|
||||||
|
self.condition = condition
|
||||||
|
|
||||||
def check_name(self):
|
def check_name(self):
|
||||||
errors = []
|
errors = []
|
||||||
|
@ -48,12 +55,25 @@ class Index:
|
||||||
self.name = 'D%s' % self.name[1:]
|
self.name = 'D%s' % self.name[1:]
|
||||||
return errors
|
return errors
|
||||||
|
|
||||||
|
def _get_condition_sql(self, model, schema_editor):
|
||||||
|
if self.condition is None:
|
||||||
|
return ''
|
||||||
|
query = Query(model=model)
|
||||||
|
query.add_q(self.condition)
|
||||||
|
compiler = query.get_compiler(connection=schema_editor.connection)
|
||||||
|
# Only the WhereNode is of interest for the partial index.
|
||||||
|
sql, params = query.where.as_sql(compiler=compiler, connection=schema_editor.connection)
|
||||||
|
# BaseDatabaseSchemaEditor does the same map on the params, but since
|
||||||
|
# it's handled outside of that class, the work is done here.
|
||||||
|
return ' WHERE ' + (sql % tuple(map(schema_editor.quote_value, params)))
|
||||||
|
|
||||||
def create_sql(self, model, schema_editor, using=''):
|
def create_sql(self, model, schema_editor, using=''):
|
||||||
fields = [model._meta.get_field(field_name) for field_name, _ in self.fields_orders]
|
fields = [model._meta.get_field(field_name) for field_name, _ in self.fields_orders]
|
||||||
col_suffixes = [order[1] for order in self.fields_orders]
|
col_suffixes = [order[1] for order in self.fields_orders]
|
||||||
|
condition = self._get_condition_sql(model, schema_editor)
|
||||||
return schema_editor._create_index_sql(
|
return schema_editor._create_index_sql(
|
||||||
model, fields, name=self.name, using=using, db_tablespace=self.db_tablespace,
|
model, fields, name=self.name, using=using, db_tablespace=self.db_tablespace,
|
||||||
col_suffixes=col_suffixes, opclasses=self.opclasses,
|
col_suffixes=col_suffixes, opclasses=self.opclasses, condition=condition,
|
||||||
)
|
)
|
||||||
|
|
||||||
def remove_sql(self, model, schema_editor):
|
def remove_sql(self, model, schema_editor):
|
||||||
|
@ -71,6 +91,8 @@ class Index:
|
||||||
kwargs['db_tablespace'] = self.db_tablespace
|
kwargs['db_tablespace'] = self.db_tablespace
|
||||||
if self.opclasses:
|
if self.opclasses:
|
||||||
kwargs['opclasses'] = self.opclasses
|
kwargs['opclasses'] = self.opclasses
|
||||||
|
if self.condition:
|
||||||
|
kwargs['condition'] = self.condition
|
||||||
return (path, (), kwargs)
|
return (path, (), kwargs)
|
||||||
|
|
||||||
def clone(self):
|
def clone(self):
|
||||||
|
@ -107,7 +129,10 @@ class Index:
|
||||||
self.check_name()
|
self.check_name()
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "<%s: fields='%s'>" % (self.__class__.__name__, ', '.join(self.fields))
|
return "<%s: fields='%s'%s>" % (
|
||||||
|
self.__class__.__name__, ', '.join(self.fields),
|
||||||
|
'' if self.condition is None else ', condition=%s' % self.condition,
|
||||||
|
)
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __eq__(self, other):
|
||||||
return (self.__class__ == other.__class__) and (self.deconstruct() == other.deconstruct())
|
return (self.__class__ == other.__class__) and (self.deconstruct() == other.deconstruct())
|
||||||
|
|
|
@ -21,7 +21,7 @@ options`_.
|
||||||
``Index`` options
|
``Index`` options
|
||||||
=================
|
=================
|
||||||
|
|
||||||
.. class:: Index(fields=(), name=None, db_tablespace=None, opclasses=())
|
.. class:: Index(fields=(), name=None, db_tablespace=None, opclasses=(), condition=None)
|
||||||
|
|
||||||
Creates an index (B-Tree) in the database.
|
Creates an index (B-Tree) in the database.
|
||||||
|
|
||||||
|
@ -92,3 +92,44 @@ opclasses=['jsonb_path_ops'])`` creates a gin index on ``jsonfield`` using
|
||||||
``opclasses`` are ignored for databases besides PostgreSQL.
|
``opclasses`` are ignored for databases besides PostgreSQL.
|
||||||
|
|
||||||
:attr:`Index.name` is required when using ``opclasses``.
|
:attr:`Index.name` is required when using ``opclasses``.
|
||||||
|
|
||||||
|
``condition``
|
||||||
|
-------------
|
||||||
|
|
||||||
|
.. attribute:: Index.condition
|
||||||
|
|
||||||
|
.. versionadded:: 2.2
|
||||||
|
|
||||||
|
If the table is very large and your queries mostly target a subset of rows,
|
||||||
|
it may be useful to restrict an index to that subset. Specify a condition as a
|
||||||
|
:class:`~django.db.models.Q`. For example, ``condition=Q(pages__gt=400)``
|
||||||
|
indexes records with more than 400 pages.
|
||||||
|
|
||||||
|
:attr:`Index.name` is required when using ``condition``.
|
||||||
|
|
||||||
|
.. admonition:: Restrictions on PostgreSQL
|
||||||
|
|
||||||
|
PostgreSQL requires functions referenced in the condition to be be marked as
|
||||||
|
IMMUTABLE. Django doesn't validate this but PostgreSQL will error. This
|
||||||
|
means that functions such as :ref:`date-functions` and
|
||||||
|
:class:`~django.db.models.functions.Concat` aren't accepted. If you store
|
||||||
|
dates in :class:`~django.db.models.DateTimeField`, comparison to
|
||||||
|
:class:`~datetime.datetime` objects may require the ``tzinfo`` argument
|
||||||
|
to be provided because otherwise the comparison could result in a mutable
|
||||||
|
function due to the casting Django does for :ref:`lookups <field-lookups>`.
|
||||||
|
|
||||||
|
.. admonition:: Restrictions on SQLite
|
||||||
|
|
||||||
|
SQLite `imposes restrictions <https://www.sqlite.org/partialindex.html>`_
|
||||||
|
on how a partial index can be constructed.
|
||||||
|
|
||||||
|
.. admonition:: Oracle
|
||||||
|
|
||||||
|
Oracle does not support partial indexes. Instead, partial indexes can be
|
||||||
|
emulated using functional indexes. Use a :doc:`migration
|
||||||
|
</topics/migrations>` to add the index using :class:`.RunSQL`.
|
||||||
|
|
||||||
|
.. admonition:: MySQL and MariaDB
|
||||||
|
|
||||||
|
The ``condition`` argument is ignored with MySQL and MariaDB as neither
|
||||||
|
supports conditional indexes.
|
||||||
|
|
|
@ -201,6 +201,8 @@ Models
|
||||||
|
|
||||||
* Added support for PostgreSQL operator classes (:attr:`.Index.opclasses`).
|
* Added support for PostgreSQL operator classes (:attr:`.Index.opclasses`).
|
||||||
|
|
||||||
|
* Added support for partial indexes (:attr:`.Index.condition`).
|
||||||
|
|
||||||
* Added many :ref:`math database functions <math-functions>`.
|
* Added many :ref:`math database functions <math-functions>`.
|
||||||
|
|
||||||
* Setting the new ``ignore_conflicts`` parameter of
|
* Setting the new ``ignore_conflicts`` parameter of
|
||||||
|
@ -288,6 +290,9 @@ Database backend API
|
||||||
|
|
||||||
* ``DatabaseFeatures.uses_savepoints`` now defaults to ``True``.
|
* ``DatabaseFeatures.uses_savepoints`` now defaults to ``True``.
|
||||||
|
|
||||||
|
* Third party database backends must implement support for partial indexes or
|
||||||
|
set ``DatabaseFeatures.supports_partial_indexes`` to ``False``.
|
||||||
|
|
||||||
:mod:`django.contrib.gis`
|
:mod:`django.contrib.gis`
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,7 @@ class ArticleTranslation(models.Model):
|
||||||
class Article(models.Model):
|
class Article(models.Model):
|
||||||
headline = models.CharField(max_length=100)
|
headline = models.CharField(max_length=100)
|
||||||
pub_date = models.DateTimeField()
|
pub_date = models.DateTimeField()
|
||||||
|
published = models.BooleanField(default=False)
|
||||||
|
|
||||||
# Add virtual relation to the ArticleTranslation model.
|
# Add virtual relation to the ArticleTranslation model.
|
||||||
translation = CurrentTranslation(ArticleTranslation, models.CASCADE, ['id'], ['article'])
|
translation = CurrentTranslation(ArticleTranslation, models.CASCADE, ['id'], ['article'])
|
||||||
|
|
|
@ -1,10 +1,15 @@
|
||||||
|
import datetime
|
||||||
from unittest import skipIf, skipUnless
|
from unittest import skipIf, skipUnless
|
||||||
|
|
||||||
from django.db import connection
|
from django.db import connection
|
||||||
from django.db.models import Index
|
from django.db.models import Index
|
||||||
from django.db.models.deletion import CASCADE
|
from django.db.models.deletion import CASCADE
|
||||||
from django.db.models.fields.related import ForeignKey
|
from django.db.models.fields.related import ForeignKey
|
||||||
from django.test import TestCase, TransactionTestCase
|
from django.db.models.query_utils import Q
|
||||||
|
from django.test import (
|
||||||
|
TestCase, TransactionTestCase, skipIfDBFeature, skipUnlessDBFeature,
|
||||||
|
)
|
||||||
|
from django.utils import timezone
|
||||||
|
|
||||||
from .models import (
|
from .models import (
|
||||||
Article, ArticleTranslation, IndexedArticle2, IndexTogetherSingleList,
|
Article, ArticleTranslation, IndexedArticle2, IndexTogetherSingleList,
|
||||||
|
@ -85,6 +90,28 @@ class SchemaIndexesNotPostgreSQLTests(TransactionTestCase):
|
||||||
editor.add_index(IndexedArticle2, index)
|
editor.add_index(IndexedArticle2, index)
|
||||||
|
|
||||||
|
|
||||||
|
# The `condition` parameter is ignored by databases that don't support partial
|
||||||
|
# indexes.
|
||||||
|
@skipIfDBFeature('supports_partial_indexes')
|
||||||
|
class PartialIndexConditionIgnoredTests(TransactionTestCase):
|
||||||
|
available_apps = ['indexes']
|
||||||
|
|
||||||
|
def test_condition_ignored(self):
|
||||||
|
index = Index(
|
||||||
|
name='test_condition_ignored',
|
||||||
|
fields=['published'],
|
||||||
|
condition=Q(published=True),
|
||||||
|
)
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
# This would error if condition weren't ignored.
|
||||||
|
editor.add_index(Article, index)
|
||||||
|
|
||||||
|
self.assertNotIn(
|
||||||
|
'WHERE %s.%s' % (editor.quote_name(Article._meta.db_table), 'published'),
|
||||||
|
str(index.create_sql(Article, editor))
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@skipUnless(connection.vendor == 'postgresql', 'PostgreSQL tests')
|
@skipUnless(connection.vendor == 'postgresql', 'PostgreSQL tests')
|
||||||
class SchemaIndexesPostgreSQLTests(TransactionTestCase):
|
class SchemaIndexesPostgreSQLTests(TransactionTestCase):
|
||||||
available_apps = ['indexes']
|
available_apps = ['indexes']
|
||||||
|
@ -139,6 +166,35 @@ class SchemaIndexesPostgreSQLTests(TransactionTestCase):
|
||||||
)
|
)
|
||||||
self.assertCountEqual(cursor.fetchall(), expected_ops_classes)
|
self.assertCountEqual(cursor.fetchall(), expected_ops_classes)
|
||||||
|
|
||||||
|
def test_ops_class_partial(self):
|
||||||
|
index = Index(
|
||||||
|
name='test_ops_class_partial',
|
||||||
|
fields=['body'],
|
||||||
|
opclasses=['text_pattern_ops'],
|
||||||
|
condition=Q(headline__contains='China'),
|
||||||
|
)
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.add_index(IndexedArticle2, index)
|
||||||
|
with editor.connection.cursor() as cursor:
|
||||||
|
cursor.execute(self.get_opclass_query % 'test_ops_class_partial')
|
||||||
|
self.assertCountEqual(cursor.fetchall(), [('text_pattern_ops', 'test_ops_class_partial')])
|
||||||
|
|
||||||
|
def test_ops_class_partial_tablespace(self):
|
||||||
|
indexname = 'test_ops_class_tblspace'
|
||||||
|
index = Index(
|
||||||
|
name=indexname,
|
||||||
|
fields=['body'],
|
||||||
|
opclasses=['text_pattern_ops'],
|
||||||
|
condition=Q(headline__contains='China'),
|
||||||
|
db_tablespace='pg_default',
|
||||||
|
)
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.add_index(IndexedArticle2, index)
|
||||||
|
self.assertIn('TABLESPACE "pg_default" ', str(index.create_sql(IndexedArticle2, editor)))
|
||||||
|
with editor.connection.cursor() as cursor:
|
||||||
|
cursor.execute(self.get_opclass_query % indexname)
|
||||||
|
self.assertCountEqual(cursor.fetchall(), [('text_pattern_ops', indexname)])
|
||||||
|
|
||||||
|
|
||||||
@skipUnless(connection.vendor == 'mysql', 'MySQL tests')
|
@skipUnless(connection.vendor == 'mysql', 'MySQL tests')
|
||||||
class SchemaIndexesMySQLTests(TransactionTestCase):
|
class SchemaIndexesMySQLTests(TransactionTestCase):
|
||||||
|
@ -178,3 +234,108 @@ class SchemaIndexesMySQLTests(TransactionTestCase):
|
||||||
if field_created:
|
if field_created:
|
||||||
with connection.schema_editor() as editor:
|
with connection.schema_editor() as editor:
|
||||||
editor.remove_field(ArticleTranslation, new_field)
|
editor.remove_field(ArticleTranslation, new_field)
|
||||||
|
|
||||||
|
|
||||||
|
@skipUnlessDBFeature('supports_partial_indexes')
|
||||||
|
class PartialIndexTests(TestCase):
|
||||||
|
# Schema editor is used to create the index to test that it works.
|
||||||
|
|
||||||
|
def test_partial_index(self):
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
index = Index(
|
||||||
|
name='recent_article_idx',
|
||||||
|
fields=['pub_date'],
|
||||||
|
condition=Q(
|
||||||
|
pub_date__gt=datetime.datetime(
|
||||||
|
year=2015, month=1, day=1,
|
||||||
|
# PostgreSQL would otherwise complain about the lookup
|
||||||
|
# being converted to a mutable function (by removing
|
||||||
|
# the timezone in the cast) which is forbidden.
|
||||||
|
tzinfo=timezone.get_current_timezone(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
self.assertIn(
|
||||||
|
'WHERE %s.%s' % (editor.quote_name(Article._meta.db_table), editor.quote_name("pub_date")),
|
||||||
|
str(index.create_sql(Article, schema_editor=editor))
|
||||||
|
)
|
||||||
|
editor.add_index(index=index, model=Article)
|
||||||
|
self.assertIn(index.name, connection.introspection.get_constraints(
|
||||||
|
cursor=connection.cursor(), table_name=Article._meta.db_table,
|
||||||
|
))
|
||||||
|
|
||||||
|
def test_integer_restriction_partial(self):
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
index = Index(
|
||||||
|
name='recent_article_idx',
|
||||||
|
fields=['id'],
|
||||||
|
condition=Q(pk__gt=1),
|
||||||
|
)
|
||||||
|
self.assertIn(
|
||||||
|
'WHERE %s.%s' % (editor.quote_name(Article._meta.db_table), editor.quote_name('id')),
|
||||||
|
str(index.create_sql(Article, schema_editor=editor))
|
||||||
|
)
|
||||||
|
editor.add_index(index=index, model=Article)
|
||||||
|
self.assertIn(index.name, connection.introspection.get_constraints(
|
||||||
|
cursor=connection.cursor(), table_name=Article._meta.db_table,
|
||||||
|
))
|
||||||
|
|
||||||
|
def test_boolean_restriction_partial(self):
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
index = Index(
|
||||||
|
name='published_index',
|
||||||
|
fields=['published'],
|
||||||
|
condition=Q(published=True),
|
||||||
|
)
|
||||||
|
self.assertIn(
|
||||||
|
'WHERE %s.%s' % (editor.quote_name(Article._meta.db_table), editor.quote_name('published')),
|
||||||
|
str(index.create_sql(Article, schema_editor=editor))
|
||||||
|
)
|
||||||
|
editor.add_index(index=index, model=Article)
|
||||||
|
self.assertIn(index.name, connection.introspection.get_constraints(
|
||||||
|
cursor=connection.cursor(), table_name=Article._meta.db_table,
|
||||||
|
))
|
||||||
|
|
||||||
|
def test_multiple_conditions(self):
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
index = Index(
|
||||||
|
name='recent_article_idx',
|
||||||
|
fields=['pub_date', 'headline'],
|
||||||
|
condition=(
|
||||||
|
Q(pub_date__gt=datetime.datetime(
|
||||||
|
year=2015,
|
||||||
|
month=1,
|
||||||
|
day=1,
|
||||||
|
tzinfo=timezone.get_current_timezone(),
|
||||||
|
)) & Q(headline__contains='China')
|
||||||
|
),
|
||||||
|
)
|
||||||
|
sql = str(index.create_sql(Article, schema_editor=editor))
|
||||||
|
where = sql.find('WHERE')
|
||||||
|
self.assertIn(
|
||||||
|
'WHERE (%s.%s' % (editor.quote_name(Article._meta.db_table), editor.quote_name("pub_date")),
|
||||||
|
sql
|
||||||
|
)
|
||||||
|
# Because each backend has different syntax for the operators,
|
||||||
|
# check ONLY the occurrence of headline in the SQL.
|
||||||
|
self.assertGreater(sql.rfind('headline'), where)
|
||||||
|
editor.add_index(index=index, model=Article)
|
||||||
|
self.assertIn(index.name, connection.introspection.get_constraints(
|
||||||
|
cursor=connection.cursor(), table_name=Article._meta.db_table,
|
||||||
|
))
|
||||||
|
|
||||||
|
def test_is_null_condition(self):
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
index = Index(
|
||||||
|
name='recent_article_idx',
|
||||||
|
fields=['pub_date'],
|
||||||
|
condition=Q(pub_date__isnull=False),
|
||||||
|
)
|
||||||
|
self.assertIn(
|
||||||
|
'WHERE %s.%s IS NOT NULL' % (editor.quote_name(Article._meta.db_table), editor.quote_name("pub_date")),
|
||||||
|
str(index.create_sql(Article, schema_editor=editor))
|
||||||
|
)
|
||||||
|
editor.add_index(index=index, model=Article)
|
||||||
|
self.assertIn(index.name, connection.introspection.get_constraints(
|
||||||
|
cursor=connection.cursor(), table_name=Article._meta.db_table,
|
||||||
|
))
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.db import connection, models
|
from django.db import connection, models
|
||||||
|
from django.db.models.query_utils import Q
|
||||||
from django.test import SimpleTestCase, skipUnlessDBFeature
|
from django.test import SimpleTestCase, skipUnlessDBFeature
|
||||||
from django.test.utils import isolate_apps
|
from django.test.utils import isolate_apps
|
||||||
|
|
||||||
|
@ -14,8 +15,10 @@ class IndexesTests(SimpleTestCase):
|
||||||
def test_repr(self):
|
def test_repr(self):
|
||||||
index = models.Index(fields=['title'])
|
index = models.Index(fields=['title'])
|
||||||
multi_col_index = models.Index(fields=['title', 'author'])
|
multi_col_index = models.Index(fields=['title', 'author'])
|
||||||
|
partial_index = models.Index(fields=['title'], name='long_books_idx', condition=Q(pages__gt=400))
|
||||||
self.assertEqual(repr(index), "<Index: fields='title'>")
|
self.assertEqual(repr(index), "<Index: fields='title'>")
|
||||||
self.assertEqual(repr(multi_col_index), "<Index: fields='title, author'>")
|
self.assertEqual(repr(multi_col_index), "<Index: fields='title, author'>")
|
||||||
|
self.assertEqual(repr(partial_index), "<Index: fields='title', condition=(AND: ('pages__gt', 400))>")
|
||||||
|
|
||||||
def test_eq(self):
|
def test_eq(self):
|
||||||
index = models.Index(fields=['title'])
|
index = models.Index(fields=['title'])
|
||||||
|
@ -52,6 +55,14 @@ class IndexesTests(SimpleTestCase):
|
||||||
with self.assertRaisesMessage(ValueError, msg):
|
with self.assertRaisesMessage(ValueError, msg):
|
||||||
models.Index(name='test_opclass', fields=['field', 'other'], opclasses=['jsonb_path_ops'])
|
models.Index(name='test_opclass', fields=['field', 'other'], opclasses=['jsonb_path_ops'])
|
||||||
|
|
||||||
|
def test_condition_requires_index_name(self):
|
||||||
|
with self.assertRaisesMessage(ValueError, 'An index must be named to use condition.'):
|
||||||
|
models.Index(condition=Q(pages__gt=400))
|
||||||
|
|
||||||
|
def test_condition_must_be_q(self):
|
||||||
|
with self.assertRaisesMessage(ValueError, 'Index.condition must be a Q instance.'):
|
||||||
|
models.Index(condition='invalid', name='long_book_idx')
|
||||||
|
|
||||||
def test_max_name_length(self):
|
def test_max_name_length(self):
|
||||||
msg = 'Index names cannot be longer than 30 characters.'
|
msg = 'Index names cannot be longer than 30 characters.'
|
||||||
with self.assertRaisesMessage(ValueError, msg):
|
with self.assertRaisesMessage(ValueError, msg):
|
||||||
|
@ -110,6 +121,25 @@ class IndexesTests(SimpleTestCase):
|
||||||
{'fields': ['title'], 'name': 'model_index_title_196f42_idx', 'db_tablespace': 'idx_tbls'}
|
{'fields': ['title'], 'name': 'model_index_title_196f42_idx', 'db_tablespace': 'idx_tbls'}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_deconstruct_with_condition(self):
|
||||||
|
index = models.Index(
|
||||||
|
name='big_book_index',
|
||||||
|
fields=['title'],
|
||||||
|
condition=Q(pages__gt=400),
|
||||||
|
)
|
||||||
|
index.set_name_with_model(Book)
|
||||||
|
path, args, kwargs = index.deconstruct()
|
||||||
|
self.assertEqual(path, 'django.db.models.Index')
|
||||||
|
self.assertEqual(args, ())
|
||||||
|
self.assertEqual(
|
||||||
|
kwargs,
|
||||||
|
{
|
||||||
|
'fields': ['title'],
|
||||||
|
'name': 'model_index_title_196f42_idx',
|
||||||
|
'condition': Q(pages__gt=400),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
def test_clone(self):
|
def test_clone(self):
|
||||||
index = models.Index(fields=['title'])
|
index = models.Index(fields=['title'])
|
||||||
new_index = index.clone()
|
new_index = index.clone()
|
||||||
|
|
|
@ -4,8 +4,12 @@ from django.contrib.postgres.indexes import (
|
||||||
BrinIndex, BTreeIndex, GinIndex, GistIndex, HashIndex, SpGistIndex,
|
BrinIndex, BTreeIndex, GinIndex, GistIndex, HashIndex, SpGistIndex,
|
||||||
)
|
)
|
||||||
from django.db import connection
|
from django.db import connection
|
||||||
|
from django.db.models import CharField
|
||||||
|
from django.db.models.functions import Length
|
||||||
|
from django.db.models.query_utils import Q
|
||||||
from django.db.utils import NotSupportedError
|
from django.db.utils import NotSupportedError
|
||||||
from django.test import skipUnlessDBFeature
|
from django.test import skipUnlessDBFeature
|
||||||
|
from django.test.utils import register_lookup
|
||||||
|
|
||||||
from . import PostgreSQLTestCase
|
from . import PostgreSQLTestCase
|
||||||
from .models import CharFieldModel, IntegerArrayModel
|
from .models import CharFieldModel, IntegerArrayModel
|
||||||
|
@ -172,6 +176,36 @@ class SchemaTests(PostgreSQLTestCase):
|
||||||
editor.remove_index(IntegerArrayModel, index)
|
editor.remove_index(IntegerArrayModel, index)
|
||||||
self.assertNotIn(index_name, self.get_constraints(IntegerArrayModel._meta.db_table))
|
self.assertNotIn(index_name, self.get_constraints(IntegerArrayModel._meta.db_table))
|
||||||
|
|
||||||
|
def test_partial_gin_index(self):
|
||||||
|
with register_lookup(CharField, Length):
|
||||||
|
index_name = 'char_field_gin_partial_idx'
|
||||||
|
index = GinIndex(fields=['field'], name=index_name, condition=Q(field__length=40))
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.add_index(CharFieldModel, index)
|
||||||
|
constraints = self.get_constraints(CharFieldModel._meta.db_table)
|
||||||
|
self.assertEqual(constraints[index_name]['type'], 'gin')
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.remove_index(CharFieldModel, index)
|
||||||
|
self.assertNotIn(index_name, self.get_constraints(CharFieldModel._meta.db_table))
|
||||||
|
|
||||||
|
def test_partial_gin_index_with_tablespace(self):
|
||||||
|
with register_lookup(CharField, Length):
|
||||||
|
index_name = 'char_field_gin_partial_idx'
|
||||||
|
index = GinIndex(
|
||||||
|
fields=['field'],
|
||||||
|
name=index_name,
|
||||||
|
condition=Q(field__length=40),
|
||||||
|
db_tablespace='pg_default',
|
||||||
|
)
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.add_index(CharFieldModel, index)
|
||||||
|
self.assertIn('TABLESPACE "pg_default" ', str(index.create_sql(CharFieldModel, editor)))
|
||||||
|
constraints = self.get_constraints(CharFieldModel._meta.db_table)
|
||||||
|
self.assertEqual(constraints[index_name]['type'], 'gin')
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.remove_index(CharFieldModel, index)
|
||||||
|
self.assertNotIn(index_name, self.get_constraints(CharFieldModel._meta.db_table))
|
||||||
|
|
||||||
@skipUnlessDBFeature('has_gin_pending_list_limit')
|
@skipUnlessDBFeature('has_gin_pending_list_limit')
|
||||||
def test_gin_parameters(self):
|
def test_gin_parameters(self):
|
||||||
index_name = 'integer_array_gin_params'
|
index_name = 'integer_array_gin_params'
|
||||||
|
|
|
@ -2135,6 +2135,7 @@ class SchemaTests(TransactionTestCase):
|
||||||
"using": "",
|
"using": "",
|
||||||
"columns": editor.quote_name(column),
|
"columns": editor.quote_name(column),
|
||||||
"extra": "",
|
"extra": "",
|
||||||
|
"condition": "",
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
if connection.features.uppercases_column_names:
|
if connection.features.uppercases_column_names:
|
||||||
|
|
Loading…
Reference in New Issue