Using endpoints-proto-datastore, how do you pass a

2019-05-15 08:59发布

问题:

I am trying to pass attributes into an API call that are not contained in my EndpointsModel. For example, say I have the following model:

class MyModel(EndpointsModel):
  attr1 = ndb.StringProperty()

Then assume I want to pass in attr2 as a parameter, but I don't want attr2 to be used as a filter nor do I want it to be stored in the model. I simply want to pass in some string, retrieve it inside of the method and use it to perform some business logic.

The documentation describes the query_fields parameter for specifying the fields you want to pass into the method, but these appear to be coupled to the attributes contained inside the model, therefore you cannot pass in attributes that are not specified in the model.

Likewise, the documentation states that you can pass in attributes via the path variable:

@MyModel.method(request_fields=('id',),
                path='mymodel/{id}', name='mymodel.get'
                http_method='GET')
def MyModelGet(self, my_model):
  # do something with id

But this requires you to change the URL, plus this appears to have the same constraint as query_fields (the attribute must exist in the model).

回答1:

For just this use case, EndpointsAliasProperty was created. It acts much like @property does in Python in that you can specify a getter, setter and doc, but no deleter is specified in this context.

Since these properties will be sent over the wire and used with Google's API infrastructure, a type must be specified, so we can't just use @property. In addition, we need the typical property/field metadata such as repeated, required, etc.

Its use has been documented in one of the samples, but for your specific use case,

from google.appengine.ext import ndb
from endpoints_proto_datastore.ndb import EndpointsAliasProperty
from endpoints_proto_datastore.ndb import EndpointsModel

class MyModel(EndpointsModel):
  attr1 = ndb.StringProperty()

  def attr2_set(self, value):
    # Do some checks on the value, potentially raise
    # endpoints.BadRequestException if not a string
    self._attr2 = value

  @EndpointsAliasProperty(setter=attr2_set)
  def attr2(self):
    # Use getattr in case the value was never set
    return getattr(self, '_attr2', None)

Since no value for property_type was passed to EndpointsAliasProperty, the default of protorpc.messages.StringField is used. If instead you wanted an integer, you could've instead used:

@EndpointsAliasProperty(setter=attr2_set, property_type=messages.IntegerField)