128 lines
4.0 KiB
Python
128 lines
4.0 KiB
Python
from enum import Enum
|
|
|
|
import pytest
|
|
from bson import InvalidDocument
|
|
|
|
from mongoengine import Document, EnumField, ValidationError
|
|
from tests.utils import MongoDBTestCase, get_as_pymongo
|
|
|
|
|
|
class Status(Enum):
|
|
NEW = "new"
|
|
DONE = "done"
|
|
|
|
|
|
class ModelWithEnum(Document):
|
|
status = EnumField(Status)
|
|
|
|
|
|
class TestStringEnumField(MongoDBTestCase):
|
|
def test_storage(self):
|
|
model = ModelWithEnum(status=Status.NEW).save()
|
|
assert get_as_pymongo(model) == {"_id": model.id, "status": "new"}
|
|
|
|
def test_set_enum(self):
|
|
ModelWithEnum.drop_collection()
|
|
ModelWithEnum(status=Status.NEW).save()
|
|
assert ModelWithEnum.objects(status=Status.NEW).count() == 1
|
|
assert ModelWithEnum.objects.first().status == Status.NEW
|
|
|
|
def test_set_by_value(self):
|
|
ModelWithEnum.drop_collection()
|
|
ModelWithEnum(status="new").save()
|
|
assert ModelWithEnum.objects.first().status == Status.NEW
|
|
|
|
def test_filter(self):
|
|
ModelWithEnum.drop_collection()
|
|
ModelWithEnum(status="new").save()
|
|
assert ModelWithEnum.objects(status="new").count() == 1
|
|
assert ModelWithEnum.objects(status=Status.NEW).count() == 1
|
|
assert ModelWithEnum.objects(status=Status.DONE).count() == 0
|
|
|
|
def test_change_value(self):
|
|
m = ModelWithEnum(status="new")
|
|
m.status = Status.DONE
|
|
m.save()
|
|
assert m.status == Status.DONE
|
|
|
|
m.status = "wrong"
|
|
assert m.status == "wrong"
|
|
with pytest.raises(ValidationError):
|
|
m.validate()
|
|
|
|
def test_set_default(self):
|
|
class ModelWithDefault(Document):
|
|
status = EnumField(Status, default=Status.DONE)
|
|
|
|
m = ModelWithDefault().save()
|
|
assert m.status == Status.DONE
|
|
|
|
def test_enum_field_can_be_empty(self):
|
|
ModelWithEnum.drop_collection()
|
|
m = ModelWithEnum().save()
|
|
assert m.status is None
|
|
assert ModelWithEnum.objects()[0].status is None
|
|
assert ModelWithEnum.objects(status=None).count() == 1
|
|
|
|
def test_set_none_explicitly(self):
|
|
ModelWithEnum.drop_collection()
|
|
ModelWithEnum(status=None).save()
|
|
assert ModelWithEnum.objects.first().status is None
|
|
|
|
def test_cannot_create_model_with_wrong_enum_value(self):
|
|
m = ModelWithEnum(status="wrong_one")
|
|
with pytest.raises(ValidationError):
|
|
m.validate()
|
|
|
|
def test_user_is_informed_when_tries_to_set_choices(self):
|
|
with pytest.raises(ValueError, match="'choices' can't be set on EnumField"):
|
|
EnumField(Status, choices=["my", "custom", "options"])
|
|
|
|
|
|
class Color(Enum):
|
|
RED = 1
|
|
BLUE = 2
|
|
|
|
|
|
class ModelWithColor(Document):
|
|
color = EnumField(Color, default=Color.RED)
|
|
|
|
|
|
class TestIntEnumField(MongoDBTestCase):
|
|
def test_enum_with_int(self):
|
|
ModelWithColor.drop_collection()
|
|
m = ModelWithColor().save()
|
|
assert m.color == Color.RED
|
|
assert ModelWithColor.objects(color=Color.RED).count() == 1
|
|
assert ModelWithColor.objects(color=1).count() == 1
|
|
assert ModelWithColor.objects(color=2).count() == 0
|
|
|
|
def test_create_int_enum_by_value(self):
|
|
model = ModelWithColor(color=2).save()
|
|
assert model.color == Color.BLUE
|
|
|
|
def test_storage_enum_with_int(self):
|
|
model = ModelWithColor(color=Color.BLUE).save()
|
|
assert get_as_pymongo(model) == {"_id": model.id, "color": 2}
|
|
|
|
def test_validate_model(self):
|
|
with pytest.raises(ValidationError, match="Value must be one of"):
|
|
ModelWithColor(color=3).validate()
|
|
|
|
with pytest.raises(ValidationError, match="Value must be one of"):
|
|
ModelWithColor(color="wrong_type").validate()
|
|
|
|
|
|
class TestFunkyEnumField(MongoDBTestCase):
|
|
def test_enum_incompatible_bson_type_fails_during_save(self):
|
|
class FunkyColor(Enum):
|
|
YELLOW = object()
|
|
|
|
class ModelWithFunkyColor(Document):
|
|
color = EnumField(FunkyColor)
|
|
|
|
m = ModelWithFunkyColor(color=FunkyColor.YELLOW)
|
|
|
|
with pytest.raises(InvalidDocument, match="[cC]annot encode object"):
|
|
m.save()
|