Skip to content

Relations

The document can contain links to other documents in their fields.

Only top-level fields are fully supported for now.

The following field types are supported:

  • Link[...]
  • Optional[Link[...]]
  • List[Link[...]]
  • Optional[List[Link[...]]]

Also, backward links are supported:

  • BackLink[...]
  • Optional[BackLink[...]]
  • List[BackLink[...]]
  • Optional[List[BackLink[...]]]

Direct link to the document:

from bunnet import Document, Link


class Door(Document):
    height: int = 2
    width: int = 1


class House(Document):
    name: str
    door: Link[Door]

Optional direct link to the document:

from typing import Optional

from bunnet import Document, Link


class Door(Document):
    height: int = 2
    width: int = 1


class House(Document):
    name: str
    door: Optional[Link[Door]]

List of the links:

from typing import List

from bunnet import Document, Link


class Window(Document):
    x: int = 10
    y: int = 10


class House(Document):
    name: str
    door: Link[Door]
    windows: List[Link[Window]]

Optional list of the links:

from typing import List, Optional

from bunnet import Document, Link

class Window(Document):
    x: int = 10
    y: int = 10

class Yard(Document):
    v: int = 10
    y: int = 10

class House(Document):
    name: str
    door: Link[Door]
    windows: List[Link[Window]]
    yards: Optional[List[Link[Yard]]]

Other link patterns are not supported at this moment. If you need something more specific for your use-case, please open an issue on the GitHub page - https://github.com/roman-right/bunnet

Write

The following write methods support relations:

  • insert(...)
  • replace(...)
  • save(...)

To apply a write method to the linked documents, you should pass the respective link_rule argument

house.windows = [Window(x=100, y=100)]
house.name = "NEW NAME"

# The next call will insert a new window object and replace the house instance with updated data
house.save(link_rule=WriteRules.WRITE)

# `insert` and `replace` methods will work the same way

Otherwise, Bunnet can ignore internal links with the link_rule parameter WriteRules.DO_NOTHING

house.door.height = 3
house.name = "NEW NAME"

# The next call will just replace the house instance with new data, but the linked door object will not be synced
house.replace(link_rule=WriteRules.DO_NOTHING)

# `insert` and `save` methods will work the same way

Fetch

Prefetch

You can fetch linked documents on the find query step using the fetch_links parameter

houses = House.find(
    House.name == "test", 
    fetch_links=True
).to_list()
Supported find methods: - find - find_one - get

Bunnet uses the single aggregation query under the hood to fetch all the linked documents. This operation is very effective.

If a direct link is referred to a non-existent document, after fetching it will remain the object of the Link class.

Fetching will ignore non-existent documents for the list of links fields.

Search by linked documents fields

If the fetch_links parameter is set to True, search by linked documents fields is available.

By field of the direct link:

houses = House.find(
    House.door.height == 2,
    fetch_links=True
).to_list()

By list of links:

houses = House.find(
    House.windows.x > 10,
    fetch_links=True
).to_list()

Search by id of the linked documents works using the following syntax:

houses = House.find(
    House.door.id == "DOOR_ID_HERE"
).to_list()

It works the same way with fetch_links equal to True and False and for find_many and find_one methods.

On-demand fetch

If you don't use prefetching, linked documents will be presented as objects of the Link class. You can fetch them manually afterwards. To fetch all the linked documents, you can use the fetch_all_links method

house.fetch_all_links()

It will fetch all the linked documents and replace Link objects with them.

Otherwise, you can fetch a single field:

house.fetch_link(House.door)

This will fetch the Door object and put it into the door field of the house object.

Delete

Delete method works the same way as write operations, but it uses other rules.

To delete all the links on the document deletion, you should use the DeleteRules.DELETE_LINKS value for the link_rule parameter:

house.delete(link_rule=DeleteRules.DELETE_LINKS).run()

To keep linked documents, you can use the DO_NOTHING rule:

house.delete(link_rule=DeleteRules.DO_NOTHING).run()

To init the back link you should have a document with the direct or list of links to the current document.

from typing import List

from bunnet import Document, BackLink, Link
from pydantic import Field


class House(Document):
    name: str
    door: Link["Door"]
    owners: List[Link["Person"]]


class Door(Document):
    height: int = 2
    width: int = 1
    house: BackLink[House] = Field(original_field="door")


class Person(Document):
    name: str
    house: List[BackLink[House]] = Field(original_field="owners")

The original_field parameter is required for the back link field.

Back links support all the operations that normal links support.

Limitations

  • Find operations with the fetch_links parameter can not be used in the chaning with delete and update methods.