Argument store added to updateEntityProperties.
This argument determines if an entity should be stored in the data model after its properties are updated.
It may be useful, for example, along with tasks (Task Queue API). One may want to make some modifications to an entity during execution of a task, but the developer is sure that at least one new task, which also wants to modify the entity, will be queued, so he or she can just update the entity without saving the changes to the data model, set the entity in memcache and the following task (which is to be executed very shortly) is to retrive the current entity from the memcache (without any expensive calls to the actual data model).
#!/usr/bin/env python
#
# Copyright 2007 Google Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
"""CGI for displaying info about the currently running app in dev_appserver.
This serves pages under /_ah/info/ that display information about the app
currently running in the dev_appserver. It currently serves on these URLs:
/_ah/info/queries:
A list of datastore queries run so far, grouped by kind. Used to suggest
composite indices that should be built.
/_ah/info/index.yaml:
Produces an index.yaml file that can be uploaded to the real app
server by appcfg.py. This information is derived from the query
history above, by removing queries that don't need any indexes to
be built and by combining queries that can use the same index.
"""
import cgi
import wsgiref.handlers
from google.appengine.api import apiproxy_stub_map
from google.appengine.datastore import datastore_pb
from google.appengine.ext import webapp
from google.appengine.tools import dev_appserver_index
class QueriesHandler(webapp.RequestHandler):
"""A handler that displays a list of the datastore queries run so far.
"""
HEADER = """<html>
<head><title>Query History</title></head>
<body>
<h3>Query History</h3>
<p>This is a list of datastore queries your app has run. You have to
make composite indices for these queries before deploying your app.
This is normally done automatically by running dev_appserver, which
will write the file index.yaml into your app's root directory, and
then deploying your app with appcfg, which will upload that
index.yaml.</p>
<p>You can also view a 'clean' <a href="index.yaml">index.yaml</a>
file and save that to your app's root directory.</p>
<table>
<tr><th>Times run</th><th>Query</th></tr>
"""
ROW = """<tr><td>%(count)s</td><td>%(query)s</td></tr>"""
FOOTER = """
</table>
</body>
</html>"""
def Render(self):
"""Renders and returns the query history page HTML.
Returns:
A string, formatted as an HTML page.
"""
history = apiproxy_stub_map.apiproxy.GetStub('datastore_v3').QueryHistory()
history_items = [(count, query) for query, count in history.items()]
history_items.sort(reverse=True)
rows = [self.ROW % {'query': _FormatQuery(query),
'count': count}
for count, query in history_items]
return self.HEADER + '\n'.join(rows) + self.FOOTER
def get(self):
"""Handle a GET. Just calls Render()."""
self.response.out.write(self.Render())
class IndexYamlHandler(webapp.RequestHandler):
"""A handler that renders an index.yaml file suitable for upload."""
def Render(self):
"""Renders and returns the index.yaml file.
Returns:
A string, formatted as an index.yaml file.
"""
datastore_stub = apiproxy_stub_map.apiproxy.GetStub('datastore_v3')
query_history = datastore_stub.QueryHistory()
body = dev_appserver_index.GenerateIndexFromHistory(query_history)
return 'indexes:\n' + body
def get(self):
"""Handle a GET. Just calls Render()."""
self.response.headers['Content-Type'] = 'text/plain'
self.response.out.write(self.Render())
def _FormatQuery(query):
"""Format a Query protobuf as (very simple) HTML.
Args:
query: A datastore_pb.Query instance.
Returns:
A string containing formatted HTML. This is mostly the output of
str(query) with '<' etc. escaped, and '<br>' inserted in front of
Order and Filter parts.
"""
res = cgi.escape(str(query))
res = res.replace('Order', '<br>Order')
res = res.replace('Filter', '<br>Filter')
return res
def _DirectionToString(direction):
"""Turn a direction enum into a string.
Args:
direction: ASCENDING or DESCENDING
Returns:
Either 'asc' or 'descending'.
"""
if direction == datastore_pb.Query_Order.DESCENDING:
return 'descending'
else:
return 'asc'
URL_MAP = {
'/_ah/info/queries': QueriesHandler,
'/_ah/info/index.yaml': IndexYamlHandler,
}
def main():
application = webapp.WSGIApplication(URL_MAP.items())
wsgiref.handlers.CGIHandler().run(application)
if __name__ == '__main__':
main()