123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540 |
- import datetime
- from unittest import skipIf, skipUnless
- from django.db import connection
- from django.db.models import CASCADE, ForeignKey, Index, Q
- from django.test import (
- TestCase, TransactionTestCase, skipIfDBFeature, skipUnlessDBFeature,
- )
- from django.test.utils import override_settings
- from django.utils import timezone
- from .models import (
- Article, ArticleTranslation, IndexedArticle2, IndexTogetherSingleList,
- )
- class SchemaIndexesTests(TestCase):
- """
- Test index handling by the db.backends.schema infrastructure.
- """
- def test_index_name_hash(self):
- """
- Index names should be deterministic.
- """
- editor = connection.schema_editor()
- index_name = editor._create_index_name(
- table_name=Article._meta.db_table,
- column_names=("c1",),
- suffix="123",
- )
- self.assertEqual(index_name, "indexes_article_c1_a52bd80b123")
- def test_index_name(self):
- """
- Index names on the built-in database backends::
- * Are truncated as needed.
- * Include all the column names.
- * Include a deterministic hash.
- """
- long_name = 'l%sng' % ('o' * 100)
- editor = connection.schema_editor()
- index_name = editor._create_index_name(
- table_name=Article._meta.db_table,
- column_names=('c1', 'c2', long_name),
- suffix='ix',
- )
- expected = {
- 'mysql': 'indexes_article_c1_c2_looooooooooooooooooo_255179b2ix',
- 'oracle': 'indexes_a_c1_c2_loo_255179b2ix',
- 'postgresql': 'indexes_article_c1_c2_loooooooooooooooooo_255179b2ix',
- 'sqlite': 'indexes_article_c1_c2_l%sng_255179b2ix' % ('o' * 100),
- }
- if connection.vendor not in expected:
- self.skipTest('This test is only supported on the built-in database backends.')
- self.assertEqual(index_name, expected[connection.vendor])
- def test_index_together(self):
- editor = connection.schema_editor()
- index_sql = [str(statement) for statement in editor._model_indexes_sql(Article)]
- self.assertEqual(len(index_sql), 1)
- # Ensure the index name is properly quoted
- self.assertIn(
- connection.ops.quote_name(
- editor._create_index_name(Article._meta.db_table, ['headline', 'pub_date'], suffix='_idx')
- ),
- index_sql[0]
- )
- def test_index_together_single_list(self):
- # Test for using index_together with a single list (#22172)
- index_sql = connection.schema_editor()._model_indexes_sql(IndexTogetherSingleList)
- self.assertEqual(len(index_sql), 1)
- def test_columns_list_sql(self):
- index = Index(fields=['headline'], name='whitespace_idx')
- editor = connection.schema_editor()
- self.assertIn(
- '(%s)' % editor.quote_name('headline'),
- str(index.create_sql(Article, editor)),
- )
- def test_descending_columns_list_sql(self):
- index = Index(fields=['-headline'], name='whitespace_idx')
- editor = connection.schema_editor()
- self.assertIn(
- '(%s DESC)' % editor.quote_name('headline'),
- str(index.create_sql(Article, editor)),
- )
- @skipIf(connection.vendor == 'postgresql', 'opclasses are PostgreSQL only')
- class SchemaIndexesNotPostgreSQLTests(TransactionTestCase):
- available_apps = ['indexes']
- def test_create_index_ignores_opclasses(self):
- index = Index(
- name='test_ops_class',
- fields=['headline'],
- opclasses=['varchar_pattern_ops'],
- )
- with connection.schema_editor() as editor:
- # This would error if opclasses weren't ignored.
- 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' % editor.quote_name('published'),
- str(index.create_sql(Article, editor))
- )
- @skipUnless(connection.vendor == 'postgresql', 'PostgreSQL tests')
- class SchemaIndexesPostgreSQLTests(TransactionTestCase):
- available_apps = ['indexes']
- get_opclass_query = '''
- SELECT opcname, c.relname FROM pg_opclass AS oc
- JOIN pg_index as i on oc.oid = ANY(i.indclass)
- JOIN pg_class as c on c.oid = i.indexrelid
- WHERE c.relname = '%s'
- '''
- def test_text_indexes(self):
- """Test creation of PostgreSQL-specific text indexes (#12234)"""
- from .models import IndexedArticle
- index_sql = [str(statement) for statement in connection.schema_editor()._model_indexes_sql(IndexedArticle)]
- self.assertEqual(len(index_sql), 5)
- self.assertIn('("headline" varchar_pattern_ops)', index_sql[1])
- self.assertIn('("body" text_pattern_ops)', index_sql[3])
- # unique=True and db_index=True should only create the varchar-specific
- # index (#19441).
- self.assertIn('("slug" varchar_pattern_ops)', index_sql[4])
- def test_virtual_relation_indexes(self):
- """Test indexes are not created for related objects"""
- index_sql = connection.schema_editor()._model_indexes_sql(Article)
- self.assertEqual(len(index_sql), 1)
- def test_ops_class(self):
- index = Index(
- name='test_ops_class',
- fields=['headline'],
- opclasses=['varchar_pattern_ops'],
- )
- 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')
- self.assertEqual(cursor.fetchall(), [('varchar_pattern_ops', 'test_ops_class')])
- def test_ops_class_multiple_columns(self):
- index = Index(
- name='test_ops_class_multiple',
- fields=['headline', 'body'],
- opclasses=['varchar_pattern_ops', 'text_pattern_ops'],
- )
- 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_multiple')
- expected_ops_classes = (
- ('varchar_pattern_ops', 'test_ops_class_multiple'),
- ('text_pattern_ops', 'test_ops_class_multiple'),
- )
- 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)])
- def test_ops_class_descending(self):
- indexname = 'test_ops_class_ordered'
- index = Index(
- name=indexname,
- fields=['-body'],
- opclasses=['text_pattern_ops'],
- )
- with connection.schema_editor() as editor:
- editor.add_index(IndexedArticle2, index)
- with editor.connection.cursor() as cursor:
- cursor.execute(self.get_opclass_query % indexname)
- self.assertCountEqual(cursor.fetchall(), [('text_pattern_ops', indexname)])
- def test_ops_class_descending_partial(self):
- indexname = 'test_ops_class_ordered_partial'
- index = Index(
- name=indexname,
- 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 % indexname)
- self.assertCountEqual(cursor.fetchall(), [('text_pattern_ops', indexname)])
- @skipUnlessDBFeature('supports_covering_indexes')
- def test_ops_class_include(self):
- index_name = 'test_ops_class_include'
- index = Index(
- name=index_name,
- fields=['body'],
- opclasses=['text_pattern_ops'],
- include=['headline'],
- )
- with connection.schema_editor() as editor:
- editor.add_index(IndexedArticle2, index)
- with editor.connection.cursor() as cursor:
- cursor.execute(self.get_opclass_query % index_name)
- self.assertCountEqual(cursor.fetchall(), [('text_pattern_ops', index_name)])
- @skipUnlessDBFeature('supports_covering_indexes')
- def test_ops_class_include_tablespace(self):
- index_name = 'test_ops_class_include_tblspace'
- index = Index(
- name=index_name,
- fields=['body'],
- opclasses=['text_pattern_ops'],
- include=['headline'],
- 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 % index_name)
- self.assertCountEqual(cursor.fetchall(), [('text_pattern_ops', index_name)])
- def test_ops_class_columns_lists_sql(self):
- index = Index(
- fields=['headline'],
- name='whitespace_idx',
- opclasses=['text_pattern_ops'],
- )
- with connection.schema_editor() as editor:
- self.assertIn(
- '(%s text_pattern_ops)' % editor.quote_name('headline'),
- str(index.create_sql(Article, editor)),
- )
- def test_ops_class_descending_columns_list_sql(self):
- index = Index(
- fields=['-headline'],
- name='whitespace_idx',
- opclasses=['text_pattern_ops'],
- )
- with connection.schema_editor() as editor:
- self.assertIn(
- '(%s text_pattern_ops DESC)' % editor.quote_name('headline'),
- str(index.create_sql(Article, editor)),
- )
- @skipUnless(connection.vendor == 'mysql', 'MySQL tests')
- class SchemaIndexesMySQLTests(TransactionTestCase):
- available_apps = ['indexes']
- def test_no_index_for_foreignkey(self):
- """
- MySQL on InnoDB already creates indexes automatically for foreign keys.
- (#14180). An index should be created if db_constraint=False (#26171).
- """
- with connection.cursor() as cursor:
- storage = connection.introspection.get_storage_engine(
- cursor, ArticleTranslation._meta.db_table,
- )
- if storage != "InnoDB":
- self.skip("This test only applies to the InnoDB storage engine")
- index_sql = [str(statement) for statement in connection.schema_editor()._model_indexes_sql(ArticleTranslation)]
- self.assertEqual(index_sql, [
- 'CREATE INDEX `indexes_articletranslation_article_no_constraint_id_d6c0806b` '
- 'ON `indexes_articletranslation` (`article_no_constraint_id`)'
- ])
- # The index also shouldn't be created if the ForeignKey is added after
- # the model was created.
- field_created = False
- try:
- with connection.schema_editor() as editor:
- new_field = ForeignKey(Article, CASCADE)
- new_field.set_attributes_from_name('new_foreign_key')
- editor.add_field(ArticleTranslation, new_field)
- field_created = True
- # No deferred SQL. The FK constraint is included in the
- # statement to add the field.
- self.assertFalse(editor.deferred_sql)
- finally:
- if field_created:
- with connection.schema_editor() as editor:
- editor.remove_field(ArticleTranslation, new_field)
- @skipUnlessDBFeature('supports_partial_indexes')
- # SQLite doesn't support timezone-aware datetimes when USE_TZ is False.
- @override_settings(USE_TZ=True)
- class PartialIndexTests(TransactionTestCase):
- # Schema editor is used to create the index to test that it works.
- available_apps = ['indexes']
- 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' % editor.quote_name('pub_date'),
- str(index.create_sql(Article, schema_editor=editor))
- )
- editor.add_index(index=index, model=Article)
- with connection.cursor() as cursor:
- self.assertIn(index.name, connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- ))
- editor.remove_index(index=index, model=Article)
- 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' % editor.quote_name('id'),
- str(index.create_sql(Article, schema_editor=editor))
- )
- editor.add_index(index=index, model=Article)
- with connection.cursor() as cursor:
- self.assertIn(index.name, connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- ))
- editor.remove_index(index=index, model=Article)
- 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' % editor.quote_name('published'),
- str(index.create_sql(Article, schema_editor=editor))
- )
- editor.add_index(index=index, model=Article)
- with connection.cursor() as cursor:
- self.assertIn(index.name, connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- ))
- editor.remove_index(index=index, model=Article)
- @skipUnlessDBFeature('supports_functions_in_partial_indexes')
- 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' % 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)
- with connection.cursor() as cursor:
- self.assertIn(index.name, connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- ))
- editor.remove_index(index=index, model=Article)
- 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 IS NOT NULL' % editor.quote_name('pub_date'),
- str(index.create_sql(Article, schema_editor=editor))
- )
- editor.add_index(index=index, model=Article)
- with connection.cursor() as cursor:
- self.assertIn(index.name, connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- ))
- editor.remove_index(index=index, model=Article)
- @skipUnlessDBFeature('supports_covering_indexes')
- class CoveringIndexTests(TransactionTestCase):
- available_apps = ['indexes']
- def test_covering_index(self):
- index = Index(
- name='covering_headline_idx',
- fields=['headline'],
- include=['pub_date', 'published'],
- )
- with connection.schema_editor() as editor:
- self.assertIn(
- '(%s) INCLUDE (%s, %s)' % (
- editor.quote_name('headline'),
- editor.quote_name('pub_date'),
- editor.quote_name('published'),
- ),
- str(index.create_sql(Article, editor)),
- )
- editor.add_index(Article, index)
- with connection.cursor() as cursor:
- constraints = connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- )
- self.assertIn(index.name, constraints)
- self.assertEqual(
- constraints[index.name]['columns'],
- ['headline', 'pub_date', 'published'],
- )
- editor.remove_index(Article, index)
- with connection.cursor() as cursor:
- self.assertNotIn(index.name, connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- ))
- def test_covering_partial_index(self):
- index = Index(
- name='covering_partial_headline_idx',
- fields=['headline'],
- include=['pub_date'],
- condition=Q(pub_date__isnull=False),
- )
- with connection.schema_editor() as editor:
- self.assertIn(
- '(%s) INCLUDE (%s) WHERE %s ' % (
- editor.quote_name('headline'),
- editor.quote_name('pub_date'),
- editor.quote_name('pub_date'),
- ),
- str(index.create_sql(Article, editor)),
- )
- editor.add_index(Article, index)
- with connection.cursor() as cursor:
- constraints = connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- )
- self.assertIn(index.name, constraints)
- self.assertEqual(
- constraints[index.name]['columns'],
- ['headline', 'pub_date'],
- )
- editor.remove_index(Article, index)
- with connection.cursor() as cursor:
- self.assertNotIn(index.name, connection.introspection.get_constraints(
- cursor=cursor, table_name=Article._meta.db_table,
- ))
- @skipIfDBFeature('supports_covering_indexes')
- class CoveringIndexIgnoredTests(TransactionTestCase):
- available_apps = ['indexes']
- def test_covering_ignored(self):
- index = Index(
- name='test_covering_ignored',
- fields=['headline'],
- include=['pub_date'],
- )
- with connection.schema_editor() as editor:
- editor.add_index(Article, index)
- self.assertNotIn(
- 'INCLUDE (%s)' % editor.quote_name('headline'),
- str(index.create_sql(Article, editor)),
- )
|