* add enum field * add decorator for asyncio * fix enum typing, additional tests, add docs * add more tests Co-authored-by: collerek <collerek@gmail.com>
145 lines
3.9 KiB
Python
145 lines
3.9 KiB
Python
import datetime
|
|
from enum import Enum
|
|
|
|
import databases
|
|
import pydantic
|
|
import pytest
|
|
import sqlalchemy
|
|
|
|
import ormar
|
|
from ormar import ModelDefinitionError
|
|
from tests.settings import DATABASE_URL
|
|
|
|
database = databases.Database(DATABASE_URL, force_rollback=True)
|
|
metadata = sqlalchemy.MetaData()
|
|
|
|
|
|
def time():
|
|
return datetime.datetime.now().time()
|
|
|
|
|
|
class MyEnum(Enum):
|
|
SMALL = 1
|
|
BIG = 2
|
|
|
|
|
|
class Example(ormar.Model):
|
|
class Meta:
|
|
tablename = "example"
|
|
metadata = metadata
|
|
database = database
|
|
|
|
id: int = ormar.Integer(primary_key=True)
|
|
name: str = ormar.String(max_length=200, default="aaa")
|
|
created: datetime.datetime = ormar.DateTime(default=datetime.datetime.now)
|
|
created_day: datetime.date = ormar.Date(default=datetime.date.today)
|
|
created_time: datetime.time = ormar.Time(default=time)
|
|
description: str = ormar.Text(nullable=True)
|
|
value: float = ormar.Float(nullable=True)
|
|
data: pydantic.Json = ormar.JSON(default={})
|
|
size: MyEnum = ormar.Enum(enum_class=MyEnum, default=MyEnum.SMALL)
|
|
|
|
|
|
class EnumExample(ormar.Model):
|
|
class Meta:
|
|
tablename = "enum_example"
|
|
metadata = metadata
|
|
database = database
|
|
|
|
id: int = ormar.Integer(primary_key=True)
|
|
size: MyEnum = ormar.Enum(enum_class=MyEnum, default=MyEnum.SMALL)
|
|
|
|
|
|
@pytest.fixture(autouse=True, scope="module")
|
|
def create_test_database():
|
|
engine = sqlalchemy.create_engine(DATABASE_URL)
|
|
metadata.create_all(engine)
|
|
yield
|
|
metadata.drop_all(engine)
|
|
|
|
|
|
def test_proper_enum_column_type():
|
|
assert Example.__fields__["size"].type_ == MyEnum
|
|
|
|
|
|
def test_accepts_only_proper_enums():
|
|
class WrongEnum(Enum):
|
|
A = 1
|
|
B = 2
|
|
|
|
with pytest.raises(pydantic.ValidationError):
|
|
Example(size=WrongEnum.A)
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_enum_bulk_operations():
|
|
async with database:
|
|
examples = [EnumExample(), EnumExample()]
|
|
await EnumExample.objects.bulk_create(examples)
|
|
|
|
check = await EnumExample.objects.all()
|
|
assert all(x.size == MyEnum.SMALL for x in check)
|
|
|
|
for x in check:
|
|
x.size = MyEnum.BIG
|
|
|
|
await EnumExample.objects.bulk_update(check)
|
|
check2 = await EnumExample.objects.all()
|
|
assert all(x.size == MyEnum.BIG for x in check2)
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_enum_filter():
|
|
async with database:
|
|
examples = [EnumExample(), EnumExample(size=MyEnum.BIG)]
|
|
await EnumExample.objects.bulk_create(examples)
|
|
|
|
check = await EnumExample.objects.all(size=MyEnum.SMALL)
|
|
assert len(check) == 1
|
|
|
|
check = await EnumExample.objects.all(size=MyEnum.BIG)
|
|
assert len(check) == 1
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_model_crud():
|
|
async with database:
|
|
example = Example()
|
|
await example.save()
|
|
|
|
await example.load()
|
|
assert example.created.year == datetime.datetime.now().year
|
|
assert example.created_day == datetime.date.today()
|
|
assert example.description is None
|
|
assert example.value is None
|
|
assert example.data == {}
|
|
assert example.size == MyEnum.SMALL
|
|
|
|
await example.update(data={"foo": 123}, value=123.456, size=MyEnum.BIG)
|
|
await example.load()
|
|
assert example.value == 123.456
|
|
assert example.data == {"foo": 123}
|
|
assert example.size == MyEnum.BIG
|
|
|
|
await example.update(data={"foo": 123}, value=123.456)
|
|
await example.load()
|
|
assert example.value == 123.456
|
|
assert example.data == {"foo": 123}
|
|
|
|
await example.delete()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_invalid_enum_field():
|
|
async with database:
|
|
with pytest.raises(ModelDefinitionError):
|
|
|
|
class Example2(ormar.Model):
|
|
class Meta:
|
|
tablename = "example"
|
|
metadata = metadata
|
|
database = database
|
|
|
|
id: int = ormar.Integer(primary_key=True)
|
|
size: MyEnum = ormar.Enum(enum_class=[])
|