Handle updating and getting None values
Fixes updating a field to None, so it works in a similar fashion as unsetting it via save() Updated to handle null data from the database Fixes #362
This commit is contained in:
parent
beacfae400
commit
0187a0e113
1
AUTHORS
1
AUTHORS
@ -80,3 +80,4 @@ that much better:
|
|||||||
* Stephen Young
|
* Stephen Young
|
||||||
* tkloc
|
* tkloc
|
||||||
* aid
|
* aid
|
||||||
|
* yamaneko1212
|
||||||
|
@ -5,6 +5,7 @@ Changelog
|
|||||||
Changes in dev
|
Changes in dev
|
||||||
==============
|
==============
|
||||||
|
|
||||||
|
- Fixed Handle None values for non-required fields.
|
||||||
- Removed Document._get_subclasses() - no longer required
|
- Removed Document._get_subclasses() - no longer required
|
||||||
- Fixed bug requiring subclasses when not actually needed
|
- Fixed bug requiring subclasses when not actually needed
|
||||||
- Fixed deletion of dynamic data
|
- Fixed deletion of dynamic data
|
||||||
|
@ -751,6 +751,13 @@ class BaseDocument(object):
|
|||||||
self._data[name] = value
|
self._data[name] = value
|
||||||
if hasattr(self, '_changed_fields'):
|
if hasattr(self, '_changed_fields'):
|
||||||
self._mark_as_changed(name)
|
self._mark_as_changed(name)
|
||||||
|
|
||||||
|
# Handle None values for required fields
|
||||||
|
if value is None and name in getattr(self, '_fields', {}):
|
||||||
|
self._data[name] = value
|
||||||
|
if hasattr(self, '_changed_fields'):
|
||||||
|
self._mark_as_changed(name)
|
||||||
|
return
|
||||||
super(BaseDocument, self).__setattr__(name, value)
|
super(BaseDocument, self).__setattr__(name, value)
|
||||||
|
|
||||||
def __expand_dynamic_values(self, name, value):
|
def __expand_dynamic_values(self, name, value):
|
||||||
|
@ -694,8 +694,7 @@ class QuerySet(object):
|
|||||||
elif op in ('in', 'nin', 'all', 'near'):
|
elif op in ('in', 'nin', 'all', 'near'):
|
||||||
# 'in', 'nin' and 'all' require a list of values
|
# 'in', 'nin' and 'all' require a list of values
|
||||||
value = [field.prepare_query_value(op, v) for v in value]
|
value = [field.prepare_query_value(op, v) for v in value]
|
||||||
|
|
||||||
|
|
||||||
# if op and op not in match_operators:
|
# if op and op not in match_operators:
|
||||||
if op:
|
if op:
|
||||||
if op in geo_operators:
|
if op in geo_operators:
|
||||||
@ -1301,7 +1300,8 @@ class QuerySet(object):
|
|||||||
field = cleaned_fields[-1]
|
field = cleaned_fields[-1]
|
||||||
|
|
||||||
if op in (None, 'set', 'push', 'pull', 'addToSet'):
|
if op in (None, 'set', 'push', 'pull', 'addToSet'):
|
||||||
value = field.prepare_query_value(op, value)
|
if field.required or value is not None:
|
||||||
|
value = field.prepare_query_value(op, value)
|
||||||
elif op in ('pushAll', 'pullAll'):
|
elif op in ('pushAll', 'pullAll'):
|
||||||
value = [field.prepare_query_value(op, v) for v in value]
|
value = [field.prepare_query_value(op, v) for v in value]
|
||||||
|
|
||||||
|
@ -46,6 +46,81 @@ class FieldTest(unittest.TestCase):
|
|||||||
person = Person(age=30)
|
person = Person(age=30)
|
||||||
self.assertRaises(ValidationError, person.validate)
|
self.assertRaises(ValidationError, person.validate)
|
||||||
|
|
||||||
|
def test_not_required_handles_none_in_update(self):
|
||||||
|
"""Ensure that every fields should accept None if required is False.
|
||||||
|
"""
|
||||||
|
|
||||||
|
class HandleNoneFields(Document):
|
||||||
|
str_fld = StringField()
|
||||||
|
int_fld = IntField()
|
||||||
|
flt_fld = FloatField()
|
||||||
|
comp_dt_fld = ComplexDateTimeField()
|
||||||
|
|
||||||
|
HandleNoneFields.drop_collection()
|
||||||
|
|
||||||
|
doc = HandleNoneFields()
|
||||||
|
doc.str_fld = u'spam ham egg'
|
||||||
|
doc.int_fld = 42
|
||||||
|
doc.flt_fld = 4.2
|
||||||
|
doc.com_dt_fld = datetime.datetime.utcnow()
|
||||||
|
doc.save()
|
||||||
|
|
||||||
|
res = HandleNoneFields.objects(id=doc.id).update(
|
||||||
|
set__str_fld=None,
|
||||||
|
set__int_fld=None,
|
||||||
|
set__flt_fld=None,
|
||||||
|
set__comp_dt_fld=None,
|
||||||
|
)
|
||||||
|
self.assertEqual(res, 1)
|
||||||
|
|
||||||
|
# Retrive data from db and verify it.
|
||||||
|
ret = HandleNoneFields.objects.all()[0]
|
||||||
|
|
||||||
|
self.assertEqual(ret.str_fld, None)
|
||||||
|
self.assertEqual(ret.int_fld, None)
|
||||||
|
self.assertEqual(ret.flt_fld, None)
|
||||||
|
|
||||||
|
# Return current time if retrived value is None.
|
||||||
|
self.assertTrue(isinstance(ret.comp_dt_fld, datetime.datetime))
|
||||||
|
|
||||||
|
def test_not_required_handles_none_from_database(self):
|
||||||
|
"""Ensure that every fields can handle null values from the database.
|
||||||
|
"""
|
||||||
|
|
||||||
|
class HandleNoneFields(Document):
|
||||||
|
str_fld = StringField(required=True)
|
||||||
|
int_fld = IntField(required=True)
|
||||||
|
flt_fld = FloatField(required=True)
|
||||||
|
comp_dt_fld = ComplexDateTimeField(required=True)
|
||||||
|
|
||||||
|
HandleNoneFields.drop_collection()
|
||||||
|
|
||||||
|
doc = HandleNoneFields()
|
||||||
|
doc.str_fld = u'spam ham egg'
|
||||||
|
doc.int_fld = 42
|
||||||
|
doc.flt_fld = 4.2
|
||||||
|
doc.com_dt_fld = datetime.datetime.utcnow()
|
||||||
|
doc.save()
|
||||||
|
|
||||||
|
collection = self.db[HandleNoneFields._get_collection_name()]
|
||||||
|
obj = collection.update({"_id": doc.id}, {"$unset": {
|
||||||
|
"str_fld": 1,
|
||||||
|
"int_fld": 1,
|
||||||
|
"flt_fld": 1,
|
||||||
|
"comp_dt_fld": 1}
|
||||||
|
})
|
||||||
|
|
||||||
|
# Retrive data from db and verify it.
|
||||||
|
ret = HandleNoneFields.objects.all()[0]
|
||||||
|
|
||||||
|
self.assertEqual(ret.str_fld, None)
|
||||||
|
self.assertEqual(ret.int_fld, None)
|
||||||
|
self.assertEqual(ret.flt_fld, None)
|
||||||
|
# Return current time if retrived value is None.
|
||||||
|
self.assert_(isinstance(ret.comp_dt_fld, datetime.datetime))
|
||||||
|
|
||||||
|
self.assertRaises(ValidationError, ret.validate)
|
||||||
|
|
||||||
def test_object_id_validation(self):
|
def test_object_id_validation(self):
|
||||||
"""Ensure that invalid values cannot be assigned to string fields.
|
"""Ensure that invalid values cannot be assigned to string fields.
|
||||||
"""
|
"""
|
||||||
|
Loading…
x
Reference in New Issue
Block a user