Skip to content

Commit

Permalink
Fixed #12429 -- Ensure that raw queries call resolve_columns if the b…
Browse files Browse the repository at this point in the history
…ackend defines it. This ensures (as much as possible) that the model values returned by a raw query match that in normal queries. Thanks to Ian Kelly for the report.

git-svn-id: http://code.djangoproject.com/svn/django/trunk@12904 bcc190cf-cafb-0310-a4f2-bffc1f526a37
  • Loading branch information
freakboy3742 committed Apr 1, 2010
1 parent c39ec6d commit f7cf58a
Show file tree
Hide file tree
Showing 6 changed files with 39 additions and 14 deletions.
11 changes: 9 additions & 2 deletions django/db/models/query.py
Expand Up @@ -1407,20 +1407,27 @@ def model_fields(self):
self._model_fields = {}
for field in self.model._meta.fields:
name, column = field.get_attname_column()
self._model_fields[converter(column)] = name
self._model_fields[converter(column)] = field
return self._model_fields

def transform_results(self, values):
model_init_kwargs = {}
annotations = ()

# Perform database backend type resolution
connection = connections[self.db]
compiler = connection.ops.compiler('SQLCompiler')(self.query, connection, self.db)
if hasattr(compiler, 'resolve_columns'):
fields = [self.model_fields.get(c,None) for c in self.columns]
values = compiler.resolve_columns(values, fields)

# Associate fields to values
for pos, value in enumerate(values):
column = self.columns[pos]

# Separate properties from annotations
if column in self.model_fields.keys():
model_init_kwargs[self.model_fields[column]] = value
model_init_kwargs[self.model_fields[column].attname] = value
else:
annotations += (column, value),

Expand Down
4 changes: 0 additions & 4 deletions django/db/models/sql/compiler.py
Expand Up @@ -14,10 +14,6 @@ def __init__(self, query, connection, using):
self.using = using
self.quote_cache = {}

# Check that the compiler will be able to execute the query
for alias, aggregate in self.query.aggregate_select.items():
self.connection.ops.check_aggregate_support(aggregate)

def pre_sql_setup(self):
"""
Does any necessary class setup immediately prior to producing SQL. This
Expand Down
5 changes: 5 additions & 0 deletions django/db/models/sql/query.py
Expand Up @@ -189,6 +189,11 @@ def get_compiler(self, using=None, connection=None):
raise ValueError("Need either using or connection")
if using:
connection = connections[using]

# Check that the compiler will be able to execute the query
for alias, aggregate in self.aggregate_select.items():
connection.ops.check_aggregate_support(aggregate)

return connection.ops.compiler(self.compiler)(self, connection, using)

def get_meta(self):
Expand Down
Expand Up @@ -40,31 +40,39 @@
"model": "raw_query.book",
"fields": {
"author": 1,
"title": "The awesome book"
"title": "The awesome book",
"paperback": false,
"opening_line": "It was a bright cold day in April and the clocks were striking thirteen."
}
},
{
"pk": 2,
"model": "raw_query.book",
"fields": {
"author": 1,
"title": "The horrible book"
"title": "The horrible book",
"paperback": true,
"opening_line": "On an evening in the latter part of May a middle-aged man was walking homeward from Shaston to the village of Marlott, in the adjoining Vale of Blakemore, or Blackmoor."
}
},
{
"pk": 3,
"model": "raw_query.book",
"fields": {
"author": 1,
"title": "Another awesome book"
"title": "Another awesome book",
"paperback": false,
"opening_line": "A squat grey building of only thirty-four stories."
}
},
{
"pk": 4,
"model": "raw_query.book",
"fields": {
"author": 3,
"title": "Some other book"
"title": "Some other book",
"paperback": true,
"opening_line": "It was the day my grandmother exploded."
}
},
{
Expand Down
2 changes: 2 additions & 0 deletions tests/modeltests/raw_query/models.py
Expand Up @@ -17,6 +17,8 @@ def __init__(self, *args, **kwargs):
class Book(models.Model):
title = models.CharField(max_length=255)
author = models.ForeignKey(Author)
paperback = models.BooleanField()
opening_line = models.TextField()

class Coffee(models.Model):
brand = models.CharField(max_length=255, db_column="name")
Expand Down
15 changes: 11 additions & 4 deletions tests/modeltests/raw_query/tests.py
Expand Up @@ -7,17 +7,18 @@


class RawQueryTests(TestCase):
fixtures = ['raw_query_books.json']

def assertSuccessfulRawQuery(self, model, query, expected_results,
expected_annotations=(), params=[], translations=None):
"""
Execute the passed query against the passed model and check the output
"""
results = list(model.objects.raw(query, params=params, translations=translations))
self.assertProcessed(results, expected_results, expected_annotations)
self.assertProcessed(model, results, expected_results, expected_annotations)
self.assertAnnotations(results, expected_annotations)

def assertProcessed(self, results, orig, expected_annotations=()):
def assertProcessed(self, model, results, orig, expected_annotations=()):
"""
Compare the results of a raw query against expected results
"""
Expand All @@ -27,7 +28,13 @@ def assertProcessed(self, results, orig, expected_annotations=()):
for annotation in expected_annotations:
setattr(orig_item, *annotation)

self.assertEqual(item.id, orig_item.id)
for field in model._meta.fields:
# Check that all values on the model are equal
self.assertEquals(getattr(item,field.attname),
getattr(orig_item,field.attname))
# This includes checking that they are the same type
self.assertEquals(type(getattr(item,field.attname)),
type(getattr(orig_item,field.attname)))

def assertNoAnnotations(self, results):
"""
Expand Down Expand Up @@ -115,7 +122,7 @@ def testParams(self):
author = Author.objects.all()[2]
params = [author.first_name]
results = list(Author.objects.raw(query, params=params))
self.assertProcessed(results, [author])
self.assertProcessed(Author, results, [author])
self.assertNoAnnotations(results)
self.assertEqual(len(results), 1)

Expand Down

0 comments on commit f7cf58a

Please sign in to comment.