Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

ndb to_dict method does not include object's key

I am leveraging ndb's to_dict method to convert an object's properties into a python dict. From everything I can tell, this method does not include the object's key or parent within the dict as per the documentation:

https://developers.google.com/appengine/docs/python/ndb/modelclass#Model_to_dict

However for my situation I need the key to be in the dict. My preference would be to leverage the builtin method and subclass it or something similar rather than create my own to_dict method.

What is the best way to accomplish this or am I missing something obvious? Thanks in advance.

FYI: I am not leveraging django for this project but instead straight python deployed up to gae.

like image 396
Alan Ruth Avatar asked May 31 '13 05:05

Alan Ruth


2 Answers

You're not missing anything ;-)

Just add the key to the dictionary after you call to_dict, and yes override the method.

If you have multiple models that don't share the same base class with your custom to_dict, I would implement it as a mixin.

to define to_dict as a method of a Mixin class. you would

class ModelUtils(object):
    def to_dict(self):
        result = super(ModelUtils,self).to_dict()
        result['key'] = self.key.id() #get the key as a string
        return result

Then to use it.

class MyModel(ModelUtils,ndb.Model):
    # some properties etc...
like image 163
Tim Hoffman Avatar answered Nov 02 '22 07:11

Tim Hoffman


Another easy way to achieve that (without having to override to_dict) is to add a ComputedProperty that returns the id, like so:

class MyModel(ndb.Model):

  # this property always returns the value of self.key.id()
  uid = ndb.ComputedProperty(lambda self: self.key.id(), indexed=False)

  # other properties ...

A ComputedProperty will be added to the result of to_dict just like any other property.

There are just two constraints:

  1. Apparently the name of the property can not be key (since that would conflict with the actual key) and id doesn't work either.
  2. This won't work if you don't specify a key or id when you create the object.

Also, the computed value will be written to the data store when you call put(), so it consumes some of your storage space.

An advantage is that this supports the include and exclude arguments of to_dict() out of the box.

like image 3
Marten Avatar answered Nov 02 '22 06:11

Marten