mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Refs #18247 -- Fixed SQLite QuerySet filtering on decimal result of Least and Greatest.
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							d5977e492e
						
					
				
				
					commit
					068d75688f
				
			| @@ -575,8 +575,8 @@ class Func(Expression): | |||||||
|         data['expressions'] = data['field'] = arg_joiner.join(sql_parts) |         data['expressions'] = data['field'] = arg_joiner.join(sql_parts) | ||||||
|         return template % data, params |         return template % data, params | ||||||
|  |  | ||||||
|     def as_sqlite(self, compiler, connection): |     def as_sqlite(self, compiler, connection, **extra_context): | ||||||
|         sql, params = self.as_sql(compiler, connection) |         sql, params = self.as_sql(compiler, connection, **extra_context) | ||||||
|         try: |         try: | ||||||
|             if self.output_field.get_internal_type() == 'DecimalField': |             if self.output_field.get_internal_type() == 'DecimalField': | ||||||
|                 sql = 'CAST(%s AS NUMERIC)' % sql |                 sql = 'CAST(%s AS NUMERIC)' % sql | ||||||
|   | |||||||
| @@ -132,7 +132,7 @@ class Greatest(Func): | |||||||
|  |  | ||||||
|     def as_sqlite(self, compiler, connection): |     def as_sqlite(self, compiler, connection): | ||||||
|         """Use the MAX function on SQLite.""" |         """Use the MAX function on SQLite.""" | ||||||
|         return super().as_sql(compiler, connection, function='MAX') |         return super().as_sqlite(compiler, connection, function='MAX') | ||||||
|  |  | ||||||
|  |  | ||||||
| class Least(Func): | class Least(Func): | ||||||
| @@ -152,7 +152,7 @@ class Least(Func): | |||||||
|  |  | ||||||
|     def as_sqlite(self, compiler, connection): |     def as_sqlite(self, compiler, connection): | ||||||
|         """Use the MIN function on SQLite.""" |         """Use the MIN function on SQLite.""" | ||||||
|         return super().as_sql(compiler, connection, function='MIN') |         return super().as_sqlite(compiler, connection, function='MIN') | ||||||
|  |  | ||||||
|  |  | ||||||
| class Length(Transform): | class Length(Transform): | ||||||
|   | |||||||
| @@ -49,3 +49,8 @@ class DTModel(models.Model): | |||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return 'DTModel({0})'.format(self.name) |         return 'DTModel({0})'.format(self.name) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DecimalModel(models.Model): | ||||||
|  |     n1 = models.DecimalField(decimal_places=2, max_digits=6) | ||||||
|  |     n2 = models.DecimalField(decimal_places=2, max_digits=6) | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| from datetime import datetime, timedelta | from datetime import datetime, timedelta | ||||||
|  | from decimal import Decimal | ||||||
| from unittest import skipIf, skipUnless | from unittest import skipIf, skipUnless | ||||||
|  |  | ||||||
| from django.db import connection | from django.db import connection | ||||||
| @@ -11,7 +12,7 @@ from django.db.models.functions import ( | |||||||
| from django.test import TestCase, skipIfDBFeature, skipUnlessDBFeature | from django.test import TestCase, skipIfDBFeature, skipUnlessDBFeature | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
|  |  | ||||||
| from .models import Article, Author, Fan | from .models import Article, Author, DecimalModel, Fan | ||||||
|  |  | ||||||
|  |  | ||||||
| lorem_ipsum = """ | lorem_ipsum = """ | ||||||
| @@ -202,6 +203,15 @@ class FunctionTests(TestCase): | |||||||
|         author.refresh_from_db() |         author.refresh_from_db() | ||||||
|         self.assertEqual(author.alias, 'Jim') |         self.assertEqual(author.alias, 'Jim') | ||||||
|  |  | ||||||
|  |     def test_greatest_decimal_filter(self): | ||||||
|  |         obj = DecimalModel.objects.create(n1=Decimal('1.1'), n2=Decimal('1.2')) | ||||||
|  |         self.assertCountEqual( | ||||||
|  |             DecimalModel.objects.annotate( | ||||||
|  |                 greatest=Greatest('n1', 'n2'), | ||||||
|  |             ).filter(greatest=Decimal('1.2')), | ||||||
|  |             [obj], | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def test_least(self): |     def test_least(self): | ||||||
|         now = timezone.now() |         now = timezone.now() | ||||||
|         before = now - timedelta(hours=1) |         before = now - timedelta(hours=1) | ||||||
| @@ -297,6 +307,15 @@ class FunctionTests(TestCase): | |||||||
|         author.refresh_from_db() |         author.refresh_from_db() | ||||||
|         self.assertEqual(author.alias, 'James Smith') |         self.assertEqual(author.alias, 'James Smith') | ||||||
|  |  | ||||||
|  |     def test_least_decimal_filter(self): | ||||||
|  |         obj = DecimalModel.objects.create(n1=Decimal('1.1'), n2=Decimal('1.2')) | ||||||
|  |         self.assertCountEqual( | ||||||
|  |             DecimalModel.objects.annotate( | ||||||
|  |                 least=Least('n1', 'n2'), | ||||||
|  |             ).filter(least=Decimal('1.1')), | ||||||
|  |             [obj], | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def test_concat(self): |     def test_concat(self): | ||||||
|         Author.objects.create(name='Jayden') |         Author.objects.create(name='Jayden') | ||||||
|         Author.objects.create(name='John Smith', alias='smithj', goes_by='John') |         Author.objects.create(name='John Smith', alias='smithj', goes_by='John') | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user