mirror of
				https://github.com/django/django.git
				synced 2025-10-26 07:06:08 +00:00 
			
		
		
		
	Fixed #30943 -- Added BloomIndex to django.contrib.postgres.
This commit is contained in:
		
				
					committed by
					
						 Mariusz Felisiak
						Mariusz Felisiak
					
				
			
			
				
	
			
			
			
						parent
						
							26554cf5d1
						
					
				
				
					commit
					02983c5242
				
			| @@ -3,8 +3,8 @@ from django.db.utils import NotSupportedError | ||||
| from django.utils.functional import cached_property | ||||
|  | ||||
| __all__ = [ | ||||
|     'BrinIndex', 'BTreeIndex', 'GinIndex', 'GistIndex', 'HashIndex', | ||||
|     'SpGistIndex', | ||||
|     'BloomIndex', 'BrinIndex', 'BTreeIndex', 'GinIndex', 'GistIndex', | ||||
|     'HashIndex', 'SpGistIndex', | ||||
| ] | ||||
|  | ||||
|  | ||||
| @@ -36,6 +36,54 @@ class PostgresIndex(Index): | ||||
|         return [] | ||||
|  | ||||
|  | ||||
| class BloomIndex(PostgresIndex): | ||||
|     suffix = 'bloom' | ||||
|  | ||||
|     def __init__(self, *, length=None, columns=(), **kwargs): | ||||
|         super().__init__(**kwargs) | ||||
|         if len(self.fields) > 32: | ||||
|             raise ValueError('Bloom indexes support a maximum of 32 fields.') | ||||
|         if not isinstance(columns, (list, tuple)): | ||||
|             raise ValueError('BloomIndex.columns must be a list or tuple.') | ||||
|         if len(columns) > len(self.fields): | ||||
|             raise ValueError( | ||||
|                 'BloomIndex.columns cannot have more values than fields.' | ||||
|             ) | ||||
|         if not all(0 < col <= 4095 for col in columns): | ||||
|             raise ValueError( | ||||
|                 'BloomIndex.columns must contain integers from 1 to 4095.', | ||||
|             ) | ||||
|         if length is not None and not 0 < length <= 4096: | ||||
|             raise ValueError( | ||||
|                 'BloomIndex.length must be None or an integer from 1 to 4096.', | ||||
|             ) | ||||
|         self.length = length | ||||
|         self.columns = columns | ||||
|  | ||||
|     def deconstruct(self): | ||||
|         path, args, kwargs = super().deconstruct() | ||||
|         if self.length is not None: | ||||
|             kwargs['length'] = self.length | ||||
|         if self.columns: | ||||
|             kwargs['columns'] = self.columns | ||||
|         return path, args, kwargs | ||||
|  | ||||
|     def check_supported(self, schema_editor): | ||||
|         if not schema_editor.connection.features.has_bloom_index: | ||||
|             raise NotSupportedError('Bloom indexes require PostgreSQL 9.6+.') | ||||
|  | ||||
|     def get_with_params(self): | ||||
|         with_params = [] | ||||
|         if self.length is not None: | ||||
|             with_params.append('length = %d' % self.length) | ||||
|         if self.columns: | ||||
|             with_params.extend( | ||||
|                 'col%d = %d' % (i, v) | ||||
|                 for i, v in enumerate(self.columns, start=1) | ||||
|             ) | ||||
|         return with_params | ||||
|  | ||||
|  | ||||
| class BrinIndex(PostgresIndex): | ||||
|     suffix = 'brin' | ||||
|  | ||||
|   | ||||
| @@ -37,6 +37,12 @@ class CreateExtension(Operation): | ||||
|         return "Creates extension %s" % self.name | ||||
|  | ||||
|  | ||||
| class BloomExtension(CreateExtension): | ||||
|  | ||||
|     def __init__(self): | ||||
|         self.name = 'bloom' | ||||
|  | ||||
|  | ||||
| class BtreeGinExtension(CreateExtension): | ||||
|  | ||||
|     def __init__(self): | ||||
|   | ||||
| @@ -68,6 +68,7 @@ class DatabaseFeatures(BaseDatabaseFeatures): | ||||
|     def is_postgresql_12(self): | ||||
|         return self.connection.pg_version >= 120000 | ||||
|  | ||||
|     has_bloom_index = property(operator.attrgetter('is_postgresql_9_6')) | ||||
|     has_brin_autosummarize = property(operator.attrgetter('is_postgresql_10')) | ||||
|     has_phraseto_tsquery = property(operator.attrgetter('is_postgresql_9_6')) | ||||
|     supports_table_partitions = property(operator.attrgetter('is_postgresql_10')) | ||||
|   | ||||
| @@ -7,6 +7,29 @@ PostgreSQL specific model indexes | ||||
| The following are PostgreSQL specific :doc:`indexes </ref/models/indexes>` | ||||
| available from the ``django.contrib.postgres.indexes`` module. | ||||
|  | ||||
| ``BloomIndex`` | ||||
| ============== | ||||
|  | ||||
| .. class:: BloomIndex(length=None, columns=(), **options) | ||||
|  | ||||
|     .. versionadded:: 3.1 | ||||
|  | ||||
|     Creates a bloom_ index. | ||||
|  | ||||
|     To use this index access you need to activate the bloom_ extension on | ||||
|     PostgreSQL. You can install it using the | ||||
|     :class:`~django.contrib.postgres.operations.BloomExtension` migration | ||||
|     operation. | ||||
|  | ||||
|     Provide an integer number of bits from 1 to 4096 to the ``length`` | ||||
|     parameter to specify the length of each index entry. PostgreSQL's default | ||||
|     is 80. | ||||
|  | ||||
|     The ``columns`` argument takes a tuple or list of up to 32 values that are | ||||
|     integer number of bits from 1 to 4095. | ||||
|  | ||||
|     .. _bloom: https://www.postgresql.org/docs/current/bloom.html | ||||
|  | ||||
| ``BrinIndex`` | ||||
| ============= | ||||
|  | ||||
|   | ||||
| @@ -49,6 +49,15 @@ run the query ``CREATE EXTENSION IF NOT EXISTS hstore;``. | ||||
|  | ||||
|         This is a required argument. The name of the extension to be installed. | ||||
|  | ||||
| ``BloomExtension`` | ||||
| ================== | ||||
|  | ||||
| .. class:: BloomExtension() | ||||
|  | ||||
|     .. versionadded:: 3.1 | ||||
|  | ||||
|     Install the ``bloom`` extension. | ||||
|  | ||||
| ``BtreeGinExtension`` | ||||
| ===================== | ||||
|  | ||||
|   | ||||
| @@ -71,7 +71,10 @@ Minor features | ||||
| :mod:`django.contrib.postgres` | ||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| * ... | ||||
| * The new :class:`~django.contrib.postgres.indexes.BloomIndex` class allows | ||||
|   creating ``bloom`` indexes in the database. The new | ||||
|   :class:`~django.contrib.postgres.operations.BloomExtension` migration | ||||
|   operation installs the ``bloom`` extension to add support for this index. | ||||
|  | ||||
| :mod:`django.contrib.redirects` | ||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
|   | ||||
| @@ -4,11 +4,12 @@ from django.db import migrations | ||||
|  | ||||
| try: | ||||
|     from django.contrib.postgres.operations import ( | ||||
|         BtreeGinExtension, BtreeGistExtension, CITextExtension, | ||||
|         BloomExtension, BtreeGinExtension, BtreeGistExtension, CITextExtension, | ||||
|         CreateExtension, CryptoExtension, HStoreExtension, TrigramExtension, | ||||
|         UnaccentExtension, | ||||
|     ) | ||||
| except ImportError: | ||||
|     BloomExtension = mock.Mock() | ||||
|     BtreeGinExtension = mock.Mock() | ||||
|     BtreeGistExtension = mock.Mock() | ||||
|     CITextExtension = mock.Mock() | ||||
| @@ -22,6 +23,7 @@ except ImportError: | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     operations = [ | ||||
|         BloomExtension(), | ||||
|         BtreeGinExtension(), | ||||
|         BtreeGistExtension(), | ||||
|         CITextExtension(), | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| from unittest import mock | ||||
|  | ||||
| from django.contrib.postgres.indexes import ( | ||||
|     BrinIndex, BTreeIndex, GinIndex, GistIndex, HashIndex, SpGistIndex, | ||||
|     BloomIndex, BrinIndex, BTreeIndex, GinIndex, GistIndex, HashIndex, | ||||
|     SpGistIndex, | ||||
| ) | ||||
| from django.db import connection | ||||
| from django.db.models import CharField | ||||
| @@ -30,6 +31,50 @@ class IndexTestMixin: | ||||
|         self.assertEqual(kwargs, {'fields': ['title'], 'name': 'test_title_%s' % self.index_class.suffix}) | ||||
|  | ||||
|  | ||||
| class BloomIndexTests(IndexTestMixin, PostgreSQLSimpleTestCase): | ||||
|     index_class = BloomIndex | ||||
|  | ||||
|     def test_suffix(self): | ||||
|         self.assertEqual(BloomIndex.suffix, 'bloom') | ||||
|  | ||||
|     def test_deconstruction(self): | ||||
|         index = BloomIndex(fields=['title'], name='test_bloom', length=80, columns=[4]) | ||||
|         path, args, kwargs = index.deconstruct() | ||||
|         self.assertEqual(path, 'django.contrib.postgres.indexes.BloomIndex') | ||||
|         self.assertEqual(args, ()) | ||||
|         self.assertEqual(kwargs, { | ||||
|             'fields': ['title'], | ||||
|             'name': 'test_bloom', | ||||
|             'length': 80, | ||||
|             'columns': [4], | ||||
|         }) | ||||
|  | ||||
|     def test_invalid_fields(self): | ||||
|         msg = 'Bloom indexes support a maximum of 32 fields.' | ||||
|         with self.assertRaisesMessage(ValueError, msg): | ||||
|             BloomIndex(fields=['title'] * 33, name='test_bloom') | ||||
|  | ||||
|     def test_invalid_columns(self): | ||||
|         msg = 'BloomIndex.columns must be a list or tuple.' | ||||
|         with self.assertRaisesMessage(ValueError, msg): | ||||
|             BloomIndex(fields=['title'], name='test_bloom', columns='x') | ||||
|         msg = 'BloomIndex.columns cannot have more values than fields.' | ||||
|         with self.assertRaisesMessage(ValueError, msg): | ||||
|             BloomIndex(fields=['title'], name='test_bloom', columns=[4, 3]) | ||||
|  | ||||
|     def test_invalid_columns_value(self): | ||||
|         msg = 'BloomIndex.columns must contain integers from 1 to 4095.' | ||||
|         for length in (0, 4096): | ||||
|             with self.subTest(length), self.assertRaisesMessage(ValueError, msg): | ||||
|                 BloomIndex(fields=['title'], name='test_bloom', columns=[length]) | ||||
|  | ||||
|     def test_invalid_length(self): | ||||
|         msg = 'BloomIndex.length must be None or an integer from 1 to 4096.' | ||||
|         for length in (0, 4097): | ||||
|             with self.subTest(length), self.assertRaisesMessage(ValueError, msg): | ||||
|                 BloomIndex(fields=['title'], name='test_bloom', length=length) | ||||
|  | ||||
|  | ||||
| class BrinIndexTests(IndexTestMixin, PostgreSQLSimpleTestCase): | ||||
|     index_class = BrinIndex | ||||
|  | ||||
| @@ -217,6 +262,41 @@ class SchemaTests(PostgreSQLTestCase): | ||||
|             editor.remove_index(IntegerArrayModel, index) | ||||
|         self.assertNotIn(index_name, self.get_constraints(IntegerArrayModel._meta.db_table)) | ||||
|  | ||||
|     @skipUnlessDBFeature('has_bloom_index') | ||||
|     def test_bloom_index(self): | ||||
|         index_name = 'char_field_model_field_bloom' | ||||
|         index = BloomIndex(fields=['field'], name=index_name) | ||||
|         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'], BloomIndex.suffix) | ||||
|         with connection.schema_editor() as editor: | ||||
|             editor.remove_index(CharFieldModel, index) | ||||
|         self.assertNotIn(index_name, self.get_constraints(CharFieldModel._meta.db_table)) | ||||
|  | ||||
|     @skipUnlessDBFeature('has_bloom_index') | ||||
|     def test_bloom_parameters(self): | ||||
|         index_name = 'char_field_model_field_bloom_params' | ||||
|         index = BloomIndex(fields=['field'], name=index_name, length=512, columns=[3]) | ||||
|         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'], BloomIndex.suffix) | ||||
|         self.assertEqual(constraints[index_name]['options'], ['length=512', 'col1=3']) | ||||
|         with connection.schema_editor() as editor: | ||||
|             editor.remove_index(CharFieldModel, index) | ||||
|         self.assertNotIn(index_name, self.get_constraints(CharFieldModel._meta.db_table)) | ||||
|  | ||||
|     def test_bloom_index_not_supported(self): | ||||
|         index_name = 'bloom_index_exception' | ||||
|         index = BloomIndex(fields=['field'], name=index_name) | ||||
|         msg = 'Bloom indexes require PostgreSQL 9.6+.' | ||||
|         with self.assertRaisesMessage(NotSupportedError, msg): | ||||
|             with mock.patch('django.db.backends.postgresql.features.DatabaseFeatures.has_bloom_index', False): | ||||
|                 with connection.schema_editor() as editor: | ||||
|                     editor.add_index(CharFieldModel, index) | ||||
|         self.assertNotIn(index_name, self.get_constraints(CharFieldModel._meta.db_table)) | ||||
|  | ||||
|     def test_brin_index(self): | ||||
|         index_name = 'char_field_model_field_brin' | ||||
|         index = BrinIndex(fields=['field'], name=index_name, pages_per_range=4) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user