-
Notifications
You must be signed in to change notification settings - Fork 5
Inheritance
MongoLite support inheritance and polymorphism.
If a document inherites from another document, its structure (skeleton and optional) will be updated to match its parents' structure as well. This work too with default_values:
@connection.register
class User(Document):
__database__ = "tutorial"
__collection__ = "users"
skeleton = {
"name": unicode,
"is_registered": bool,
}
optional = {
"age": int,
}
default_values = {"is_registered": False}
class ProfessionalUser(User):
skeleton = {
"activity": unicode,
}The skeleton is updated:
>>> user = connection.User()
>>> user
{"name": None, "is_registered": False}
>>> pro = connection.ProfessionalUser()
>>> pro
{"name": None, "is_registered": False, "activity": None}When we query a document, we get all documents of the same instance:
>>> users = connection.User.find() # get User instances onlyThe issue here is that if the fetched document is a ProfessionalUser (ie, it contains the activity field), it will be wrapped into a User instance, not a ProfessionalUser instance. This is a huge issue if we need to access to method related to ProfessionalUser.
The solution is to add the _type field into the skeleton:
@connection.register
class User(Document):
__database__ = "tutorial"
__collection__ = "users"
skeleton = {
"_type": unicode,
"name": unicode,
"is_registered": bool,
}
optional = {
"age": int,
}
default_values = {"is_registered": False}
class ProfessionalUser(User):
skeleton = {
"activity": unicode,
}
def professional_stuff(self):
return "stuff"This will add the document class name into the _type field so MongoLite will know how to wrap the fetched document:
>>> user = connection.User()
>>> user
{"_type": "User", "name": None, "is_registered": False}
>>> user["name"] = u"Timy"
>>> user.save()
>>> pro = connection.ProfessionalUser()
>>> pro
{"_type": "ProfessionalUser", "name": None, "is_registered": False, "activity": None}
>>> pro["name"] = u"Bob"
>>> pro["activity"] = u"Alien"
>>> pro.save()Now MongoLite will be able to wrap the correct document:
>>> user = connection.User.find_one({"name":"Bob"})
>>> user.professional_stuff()
"stuff"No problem. Just overwrite the type_field attribute:
@connection.register
class User(Document):
type_field = "_t"
skeleton = {
"_type": unicode,
"_t": unicode,
"name": unicode,
}If you don't planed to use inherited queries, just set type_field to None;
In the following example, we have to object A and B wich herite from Root. And we want to build an object C from A and B. Let's build Root, A and B first :
class Root(Document):
skeleton = {
"root": int,
}
default_values = {"root": 42}
class A(Root):
skeleton = {
"a_field": unicode,
}
default_values = {"a_field": "hello world"}class B(Root):
skeleton = {
"b_field": unicode,
}Polymorphisme just work as expected:
class C(A,B):
skeleton = {
"c_field": float
}>>> c = C()
>>> c == {'b_field': None, 'root': 42, 'c_field': None, 'a_field': "hello world"}
True
>>> C.default_values
{"root": 42, "a_field": "hello world"}