Skip to content

Commit d83203c

Browse files
committed
[soc2010/query-refactor] Implemented F() expressions for MongoDB.
git-svn-id: http://code.djangoproject.com/svn/django/branches/soc2010/query-refactor@13430 bcc190cf-cafb-0310-a4f2-bffc1f526a37
1 parent 317010e commit d83203c

File tree

4 files changed

+47
-5
lines changed

4 files changed

+47
-5
lines changed

django/contrib/mongodb/compiler.py

Lines changed: 19 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
from pymongo import ASCENDING, DESCENDING
44

5+
from django.db.models import F
56
from django.db.models.sql.datastructures import FullResultSet, EmptyResultSet
67

78

@@ -153,8 +154,25 @@ def update(self, result_type):
153154
filters = self.get_filters(self.query.where)
154155
# TODO: Don't use set for everything, use INC and such where
155156
# appropriate.
157+
vals = {}
158+
for field, o, value in self.query.values:
159+
if hasattr(value, "evaluate"):
160+
assert value.connector in (value.ADD, value.SUB)
161+
assert not value.negated
162+
assert not value.subtree_parents
163+
lhs, rhs = value.children
164+
if isinstance(lhs, F):
165+
assert not isinstance(rhs, F)
166+
if value.connector == value.SUB:
167+
rhs = -rhs
168+
else:
169+
assert value.connector == value.ADD
170+
rhs, lhs = lhs, rhs
171+
vals.setdefault("$inc", {})[lhs.name] = rhs
172+
else:
173+
vals.setdefault("$set", {})[field.column] = value
156174
return self.connection.db[self.query.model._meta.db_table].update(
157175
filters,
158-
{"$set": dict((f.column, val) for f, o, val in self.query.values)},
176+
vals,
159177
multi=True
160178
)

django/db/models/query.py

Lines changed: 3 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,8 @@
88
from django.db import connections, router, transaction, IntegrityError
99
from django.db.models.aggregates import Aggregate
1010
from django.db.models.fields import DateField
11-
from django.db.models.query_utils import Q, select_related_descend, CollectedObjects, CyclicDependency, deferred_class_factory, InvalidQuery
11+
from django.db.models.query_utils import (Q, select_related_descend,
12+
CollectedObjects, CyclicDependency, deferred_class_factory, InvalidQuery)
1213
from django.db.models import signals, sql
1314
from django.utils.copycompat import deepcopy
1415

@@ -464,7 +465,7 @@ def update(self, **kwargs):
464465
else:
465466
forced_managed = False
466467
try:
467-
rows = query.get_compiler(self.db).execute_sql(None)
468+
rows = query.get_compiler(self.db).update(None)
468469
if forced_managed:
469470
transaction.commit(using=self.db)
470471
else:

tests/regressiontests/mongodb/models.py

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ class Artist(models.Model):
55
id = models.NativeAutoField(primary_key=True)
66
name = models.CharField(max_length=255)
77
good = models.BooleanField()
8+
age = models.IntegerField(null=True)
89

910
current_group = models.ForeignKey("Group", null=True)
1011

tests/regressiontests/mongodb/tests.py

Lines changed: 24 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
from django.db.models import Count
1+
from django.db.models import Count, F
22
from django.test import TestCase
33

44
from models import Artist, Group
@@ -27,6 +27,28 @@ def test_update(self):
2727
l = Artist.objects.get(pk=pk)
2828
self.assertTrue(not l.good)
2929

30+
def test_bulk_update(self):
31+
# Doesn't actually do an op on more than 1 item, but it's the bulk
32+
# update syntax nonetheless
33+
v = Artist.objects.create(name="Van Morrison", good=False)
34+
# How do you make a mistake like this, I don't know...
35+
Artist.objects.filter(pk=v.pk).update(good=True)
36+
self.assertTrue(Artist.objects.get(pk=v.pk).good)
37+
38+
def test_f_expressions(self):
39+
k = Artist.objects.create(name="Keb' Mo'", age=57, good=True)
40+
# Birthday!
41+
Artist.objects.filter(pk=k.pk).update(age=F("age") + 1)
42+
self.assertEqual(Artist.objects.get(pk=k.pk).age, 58)
43+
44+
# Backwards birthday
45+
Artist.objects.filter(pk=k.pk).update(age=F("age") - 1)
46+
self.assertEqual(Artist.objects.get(pk=k.pk).age, 57)
47+
48+
# Birthday again!
49+
Artist.objects.filter(pk=k.pk).update(age=1 + F("age"))
50+
self.assertEqual(Artist.objects.get(pk=k.pk).age, 58)
51+
3052
def test_count(self):
3153
Artist.objects.create(name="Billy Joel", good=True)
3254
Artist.objects.create(name="John Mellencamp", good=True)
@@ -121,7 +143,7 @@ def test_values(self):
121143

122144
self.assertQuerysetEqual(
123145
Artist.objects.values(), [
124-
{"name": "Steve Perry", "good": True, "current_group_id": None, "id": a.pk},
146+
{"name": "Steve Perry", "good": True, "current_group_id": None, "id": a.pk, "age": None},
125147
],
126148
lambda a: a,
127149
)

0 commit comments

Comments
 (0)