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()
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()
Back Links
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_linksparameter can not be used in the chaning withdeleteandupdatemethods.