BeanieODM / beanie

Asynchronous Python ODM for MongoDB
http://beanie-odm.dev/
Apache License 2.0
2.02k stars 213 forks source link

[BUG] before_event, after_event can not trigger #882

Closed hd10180 closed 7 months ago

hd10180 commented 7 months ago

Describe the bug before_event, after_event can not trigger beanie version: 1.24.0 & 1.25.0 both have the same behavior

To Reproduce a completely code

import asyncio
from datetime import datetime

from beanie import (
    Document,
    Insert,
    Replace,
    SaveChanges,
    after_event,
    before_event,
    init_beanie,
)
from motor.motor_asyncio import AsyncIOMotorClient
from pydantic import BaseModel, Field

class Users(Document):
    name: str
    enabled: bool = True
    desc: str | None = None
    created: datetime = Field(default_factory=datetime.utcnow)

    # !附加检查
    @before_event([Insert, Replace, SaveChanges])
    def checker_before(self):
        print("before..............")
        if not self.desc:
            del self.desc

    @after_event([Insert, Replace, SaveChanges])
    def checker(self):
        print("after...............")
        if not self.desc:
            del self.desc

    class Settings:
        name = "users"
        use_state_management = True

class UpdateUsers(BaseModel):
    enabled: bool | None = None
    desc: str | None = None

async def main():
    import beanie

    print(f"beanie version: {beanie.__version__}")
    cli = AsyncIOMotorClient("mongodb://localhost:27017")
    db = cli["test_wrong_collection"]
    await db.drop_collection(Users.Settings.name)
    await db[Users.Settings.name].insert_one(
        {"name": "lucy", "enabled": True, "created": datetime.utcnow()}
    )
    print(await db[Users.Settings.name].find_one({"name": "lucy"}))
    await init_beanie(db, document_models=[Users])
    test_user = Users(name="hello")
    await test_user.save()
    print(await db[Users.Settings.name].find_one({"name": "hello"}))

asyncio.run(main())

output:

beanie version: 1.24.0
{'_id': ObjectId('65d802dc64da453e0dd830be'), 'name': 'lucy', 'enabled': True, 'created': datetime.datetime(2024, 2, 23, 2, 28, 44, 893000)}
{'_id': ObjectId('65d802dc64da453e0dd830bf'), 'created': datetime.datetime(2024, 2, 23, 2, 28, 44, 947000), 'desc': None, 'enabled': True, 'name': 'hello'}

Expected behavior the record which name="hello" should not have desc=None attr, and it should call checker_before and checker function, but they didn't call

Additional context none

roman-right commented 7 months ago

Hi @hd10180 , Thank you for the issue. Please use event Save for this case. Save and SaveChanges are different and use different events.

import asyncio
from datetime import datetime

from beanie import (
    Document,
    Insert,
    Replace,
    SaveChanges,
    Save,
    after_event,
    before_event,
    init_beanie,
)
from motor.motor_asyncio import AsyncIOMotorClient
from pydantic import BaseModel, Field

class Users(Document):
    name: str
    enabled: bool = True
    desc: str | None = None
    created: datetime = Field(default_factory=datetime.utcnow)

    # !附加检查
    @before_event([Insert, Replace, SaveChanges, Save])
    def checker_before(self):
        print("before..............")
        if not self.desc:
            del self.desc

    @after_event([Insert, Replace, SaveChanges, Save])
    def checker(self):
        print("after...............")
        if not self.desc:
            del self.desc

    class Settings:
        name = "users"
        use_state_management = True

class UpdateUsers(BaseModel):
    enabled: bool | None = None
    desc: str | None = None

async def main():
    import beanie

    print(f"beanie version: {beanie.__version__}")
    cli = AsyncIOMotorClient("mongodb://beanie:beanie@localhost:27017")
    db = cli["test_wrong_collection"]
    await db.drop_collection(Users.Settings.name)
    await db[Users.Settings.name].insert_one(
        {"name": "lucy", "enabled": True, "created": datetime.utcnow()}
    )
    print(await db[Users.Settings.name].find_one({"name": "lucy"}))
    await init_beanie(db, document_models=[Users])
    test_user = Users(name="hello")
    await test_user.save()
    print(await db[Users.Settings.name].find_one({"name": "hello"}))

asyncio.run(main())
hd10180 commented 7 months ago

Hi @hd10180 , Thank you for the issue. Please use event Save for this case. Save and SaveChanges are different and use different events.

import asyncio
from datetime import datetime

from beanie import (
    Document,
    Insert,
    Replace,
    SaveChanges,
    Save,
    after_event,
    before_event,
    init_beanie,
)
from motor.motor_asyncio import AsyncIOMotorClient
from pydantic import BaseModel, Field

class Users(Document):
    name: str
    enabled: bool = True
    desc: str | None = None
    created: datetime = Field(default_factory=datetime.utcnow)

    # !附加检查
    @before_event([Insert, Replace, SaveChanges, Save])
    def checker_before(self):
        print("before..............")
        if not self.desc:
            del self.desc

    @after_event([Insert, Replace, SaveChanges, Save])
    def checker(self):
        print("after...............")
        if not self.desc:
            del self.desc

    class Settings:
        name = "users"
        use_state_management = True

class UpdateUsers(BaseModel):
    enabled: bool | None = None
    desc: str | None = None

async def main():
    import beanie

    print(f"beanie version: {beanie.__version__}")
    cli = AsyncIOMotorClient("mongodb://beanie:beanie@localhost:27017")
    db = cli["test_wrong_collection"]
    await db.drop_collection(Users.Settings.name)
    await db[Users.Settings.name].insert_one(
        {"name": "lucy", "enabled": True, "created": datetime.utcnow()}
    )
    print(await db[Users.Settings.name].find_one({"name": "lucy"}))
    await init_beanie(db, document_models=[Users])
    test_user = Users(name="hello")
    await test_user.save()
    print(await db[Users.Settings.name].find_one({"name": "hello"}))

asyncio.run(main())

thanks. Sorry for taking up your time, it works fine