allow to use sets in field choices (#1482)
This commit is contained in:
parent
ed34c2ca68
commit
b27c7ce11b
@ -4,7 +4,8 @@ Changelog
|
|||||||
|
|
||||||
Development
|
Development
|
||||||
===========
|
===========
|
||||||
- (Fill this out as you fix issues and develop you features).
|
- (Fill this out as you fix issues and develop your features).
|
||||||
|
- Fixed using sets in field choices #1481
|
||||||
- POTENTIAL BREAKING CHANGE: Fixed limit/skip/hint/batch_size chaining #1476
|
- POTENTIAL BREAKING CHANGE: Fixed limit/skip/hint/batch_size chaining #1476
|
||||||
- POTENTIAL BREAKING CHANGE: Changed a public `QuerySet.clone_into` method to a private `QuerySet._clone_into` #1476
|
- POTENTIAL BREAKING CHANGE: Changed a public `QuerySet.clone_into` method to a private `QuerySet._clone_into` #1476
|
||||||
- Fixed connecting to a replica set with PyMongo 2.x #1436
|
- Fixed connecting to a replica set with PyMongo 2.x #1436
|
||||||
|
@ -150,7 +150,7 @@ arguments can be set on all fields:
|
|||||||
.. note:: If set, this field is also accessible through the `pk` field.
|
.. note:: If set, this field is also accessible through the `pk` field.
|
||||||
|
|
||||||
:attr:`choices` (Default: None)
|
:attr:`choices` (Default: None)
|
||||||
An iterable (e.g. a list or tuple) of choices to which the value of this
|
An iterable (e.g. list, tuple or set) of choices to which the value of this
|
||||||
field should be limited.
|
field should be limited.
|
||||||
|
|
||||||
Can be either be a nested tuples of value (stored in mongo) and a
|
Can be either be a nested tuples of value (stored in mongo) and a
|
||||||
|
@ -193,7 +193,8 @@ class BaseField(object):
|
|||||||
EmbeddedDocument = _import_class('EmbeddedDocument')
|
EmbeddedDocument = _import_class('EmbeddedDocument')
|
||||||
|
|
||||||
choice_list = self.choices
|
choice_list = self.choices
|
||||||
if isinstance(choice_list[0], (list, tuple)):
|
if isinstance(next(iter(choice_list)), (list, tuple)):
|
||||||
|
# next(iter) is useful for sets
|
||||||
choice_list = [k for k, _ in choice_list]
|
choice_list = [k for k, _ in choice_list]
|
||||||
|
|
||||||
# Choices which are other types of Documents
|
# Choices which are other types of Documents
|
||||||
|
@ -1,13 +1,12 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
import six
|
|
||||||
from nose.plugins.skip import SkipTest
|
|
||||||
|
|
||||||
import datetime
|
import datetime
|
||||||
import unittest
|
import unittest
|
||||||
import uuid
|
import uuid
|
||||||
import math
|
import math
|
||||||
import itertools
|
import itertools
|
||||||
import re
|
import re
|
||||||
|
|
||||||
|
from nose.plugins.skip import SkipTest
|
||||||
import six
|
import six
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -27,21 +26,13 @@ from mongoengine import *
|
|||||||
from mongoengine.connection import get_db
|
from mongoengine.connection import get_db
|
||||||
from mongoengine.base import (BaseDict, BaseField, EmbeddedDocumentList,
|
from mongoengine.base import (BaseDict, BaseField, EmbeddedDocumentList,
|
||||||
_document_registry)
|
_document_registry)
|
||||||
from mongoengine.errors import NotRegistered, DoesNotExist
|
|
||||||
|
from tests.utils import MongoDBTestCase
|
||||||
|
|
||||||
__all__ = ("FieldTest", "EmbeddedDocumentListFieldTestCase")
|
__all__ = ("FieldTest", "EmbeddedDocumentListFieldTestCase")
|
||||||
|
|
||||||
|
|
||||||
class FieldTest(unittest.TestCase):
|
class FieldTest(MongoDBTestCase):
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
connect(db='mongoenginetest')
|
|
||||||
self.db = get_db()
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.db.drop_collection('fs.files')
|
|
||||||
self.db.drop_collection('fs.chunks')
|
|
||||||
self.db.drop_collection('mongoengine.counters')
|
|
||||||
|
|
||||||
def test_default_values_nothing_set(self):
|
def test_default_values_nothing_set(self):
|
||||||
"""Ensure that default field values are used when creating a document.
|
"""Ensure that default field values are used when creating a document.
|
||||||
@ -3186,26 +3177,42 @@ class FieldTest(unittest.TestCase):
|
|||||||
att.delete()
|
att.delete()
|
||||||
self.assertEqual(0, Attachment.objects.count())
|
self.assertEqual(0, Attachment.objects.count())
|
||||||
|
|
||||||
def test_choices_validation(self):
|
def test_choices_allow_using_sets_as_choices(self):
|
||||||
"""Ensure that value is in a container of allowed values.
|
"""Ensure that sets can be used when setting choices
|
||||||
"""
|
"""
|
||||||
class Shirt(Document):
|
class Shirt(Document):
|
||||||
size = StringField(max_length=3, choices=(
|
size = StringField(choices={'M', 'L'})
|
||||||
('S', 'Small'), ('M', 'Medium'), ('L', 'Large'),
|
|
||||||
('XL', 'Extra Large'), ('XXL', 'Extra Extra Large')))
|
|
||||||
|
|
||||||
Shirt.drop_collection()
|
Shirt(size='M').validate()
|
||||||
|
|
||||||
|
def test_choices_validation_allow_no_value(self):
|
||||||
|
"""Ensure that .validate passes and no value was provided
|
||||||
|
for a field setup with choices
|
||||||
|
"""
|
||||||
|
class Shirt(Document):
|
||||||
|
size = StringField(choices=('S', 'M'))
|
||||||
|
|
||||||
shirt = Shirt()
|
shirt = Shirt()
|
||||||
shirt.validate()
|
shirt.validate()
|
||||||
|
|
||||||
shirt.size = "S"
|
def test_choices_validation_accept_possible_value(self):
|
||||||
|
"""Ensure that value is in a container of allowed values.
|
||||||
|
"""
|
||||||
|
class Shirt(Document):
|
||||||
|
size = StringField(choices=('S', 'M'))
|
||||||
|
|
||||||
|
shirt = Shirt(size='S')
|
||||||
shirt.validate()
|
shirt.validate()
|
||||||
|
|
||||||
shirt.size = "XS"
|
def test_choices_validation_reject_unknown_value(self):
|
||||||
self.assertRaises(ValidationError, shirt.validate)
|
"""Ensure that unallowed value are rejected upon validation
|
||||||
|
"""
|
||||||
|
class Shirt(Document):
|
||||||
|
size = StringField(choices=('S', 'M'))
|
||||||
|
|
||||||
Shirt.drop_collection()
|
shirt = Shirt(size="XS")
|
||||||
|
with self.assertRaises(ValidationError):
|
||||||
|
shirt.validate()
|
||||||
|
|
||||||
def test_choices_validation_documents(self):
|
def test_choices_validation_documents(self):
|
||||||
"""
|
"""
|
||||||
@ -4420,7 +4427,8 @@ class EmbeddedDocumentListFieldTestCase(unittest.TestCase):
|
|||||||
my_list = ListField(EmbeddedDocumentField(EmbeddedWithUnique))
|
my_list = ListField(EmbeddedDocumentField(EmbeddedWithUnique))
|
||||||
|
|
||||||
A(my_list=[]).save()
|
A(my_list=[]).save()
|
||||||
self.assertRaises(NotUniqueError, lambda: A(my_list=[]).save())
|
with self.assertRaises(NotUniqueError):
|
||||||
|
A(my_list=[]).save()
|
||||||
|
|
||||||
class EmbeddedWithSparseUnique(EmbeddedDocument):
|
class EmbeddedWithSparseUnique(EmbeddedDocument):
|
||||||
number = IntField(unique=True, sparse=True)
|
number = IntField(unique=True, sparse=True)
|
||||||
@ -4431,6 +4439,9 @@ class EmbeddedDocumentListFieldTestCase(unittest.TestCase):
|
|||||||
B(my_list=[]).save()
|
B(my_list=[]).save()
|
||||||
B(my_list=[]).save()
|
B(my_list=[]).save()
|
||||||
|
|
||||||
|
A.drop_collection()
|
||||||
|
B.drop_collection()
|
||||||
|
|
||||||
def test_filtered_delete(self):
|
def test_filtered_delete(self):
|
||||||
"""
|
"""
|
||||||
Tests the delete method of a List of Embedded Documents
|
Tests the delete method of a List of Embedded Documents
|
||||||
|
@ -18,15 +18,13 @@ try:
|
|||||||
except ImportError:
|
except ImportError:
|
||||||
HAS_PIL = False
|
HAS_PIL = False
|
||||||
|
|
||||||
|
from tests.utils import MongoDBTestCase
|
||||||
|
|
||||||
TEST_IMAGE_PATH = os.path.join(os.path.dirname(__file__), 'mongoengine.png')
|
TEST_IMAGE_PATH = os.path.join(os.path.dirname(__file__), 'mongoengine.png')
|
||||||
TEST_IMAGE2_PATH = os.path.join(os.path.dirname(__file__), 'mongodb_leaf.png')
|
TEST_IMAGE2_PATH = os.path.join(os.path.dirname(__file__), 'mongodb_leaf.png')
|
||||||
|
|
||||||
|
|
||||||
class FileTest(unittest.TestCase):
|
class FileTest(MongoDBTestCase):
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
connect(db='mongoenginetest')
|
|
||||||
self.db = get_db()
|
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
self.db.drop_collection('fs.files')
|
self.db.drop_collection('fs.files')
|
||||||
|
22
tests/utils.py
Normal file
22
tests/utils.py
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import unittest
|
||||||
|
|
||||||
|
from mongoengine import connect
|
||||||
|
from mongoengine.connection import get_db
|
||||||
|
|
||||||
|
MONGO_TEST_DB = 'mongoenginetest'
|
||||||
|
|
||||||
|
|
||||||
|
class MongoDBTestCase(unittest.TestCase):
|
||||||
|
"""Base class for tests that need a mongodb connection
|
||||||
|
db is being dropped automatically
|
||||||
|
"""
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls):
|
||||||
|
cls._connection = connect(db=MONGO_TEST_DB)
|
||||||
|
cls._connection.drop_database(MONGO_TEST_DB)
|
||||||
|
cls.db = get_db()
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def tearDownClass(cls):
|
||||||
|
cls._connection.drop_database(MONGO_TEST_DB)
|
Loading…
x
Reference in New Issue
Block a user