Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(11)

Unified Diff: tools/deep_memory_profiler/visualizer/app.py

Issue 23781012: Upload file to app engine and generate public url for dmprof visualizer. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Use blobstore to store json file Created 7 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: tools/deep_memory_profiler/visualizer/app.py
diff --git a/tools/deep_memory_profiler/visualizer/app.py b/tools/deep_memory_profiler/visualizer/app.py
new file mode 100644
index 0000000000000000000000000000000000000000..58b4cefec5c7ac7315a9e6bc51e4c47138825274
--- /dev/null
+++ b/tools/deep_memory_profiler/visualizer/app.py
@@ -0,0 +1,107 @@
+# Copyright 2013 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+import jinja2
+import json
+import os
+import urllib
+import webapp2
+
+from google.appengine.ext import ndb
+from google.appengine.ext import blobstore
+from google.appengine.ext.webapp import blobstore_handlers
+
+
+JINJA_ENVIRONMENT = jinja2.Environment(
+ loader=jinja2.FileSystemLoader(os.path.dirname(__file__)),
+ extensions=['jinja2.ext.autoescape'])
+
+
+class Profiler(ndb.Model):
+ """Profiler entity to store json data. Use run_id as its key.
+ Json data will be stored at blobstore, but can be referred by BlobKey which
+ can be constructed by blob_str."""
+ blob_str = ndb.StringProperty()
sullivan 2013/09/17 14:58:47 Why not ndb.BlobKeyProperty?
junjianx 2013/09/20 06:31:09 Done.
+
+
+class Template(ndb.Model):
+ """Template to breakdown profiler with multiple tags.
+ Use content as its key."""
+ content = ndb.PickleProperty()
sullivan 2013/09/17 14:58:47 ndb.JsonProperty? Also, isn't this redundant with
junjianx 2013/09/20 06:31:09 I replaced PickleProperty with JsonProperty. But i
+ tags = ndb.StringProperty(repeated=True)
sullivan 2013/09/17 14:58:47 Where are tags used?
junjianx 2013/09/20 06:31:09 Tag will be used in next issue to help user managi
+
+
+class MainPage(webapp2.RequestHandler):
+ """Show breakdown with received profiler-id and template-id. If nothing was
+ received, show blank page waiting user to upload file."""
+ def get(self):
+ page_template = JINJA_ENVIRONMENT.get_template('index.html')
+ upload_url = blobstore.create_upload_url('/upload')
+
+ # Get profiler id and template id from url query.
+ prof_id = self.request.get('prof_id')
+ tmpl_id = self.request.get('tmpl_id')
+
+ template_values = { 'upload_url': upload_url }
+
+ if not prof_id or not tmpl_id:
+ self.response.write(page_template.render(template_values))
+ else:
+ # Get entity key.
+ prof_key = ndb.Key(urlsafe=prof_id)
sullivan 2013/09/17 14:58:47 It looks like it's also possible to construct a Pr
junjianx 2013/09/20 06:31:09 Done.
+ tmpl_key = ndb.Key(urlsafe=tmpl_id)
+
+ # Get entity object.
+ profiler = Profiler.query(ancestor=prof_key).get()
+ template = Template.query(ancestor=tmpl_key).get()
+
+ # Use blob string to construct BlobKey.
+ blob_key = blobstore.BlobKey(profiler.blob_str)
+ template_values['json'] = blobstore.BlobReader(blob_key).read()
+ template_values['template'] = template.content
+
+ self.response.write(page_template.render(template_values))
+
+
+class UploadHandler(blobstore_handlers.BlobstoreUploadHandler):
+ """Handle file uploading with BlobstoreUploadHandler. BlobstoreUploadHandler
+ can deal with files overweighing size limitation within one HTTP connection so
+ that user can upload large json file."""
+ def post(self):
+ blob_info = self.get_uploads('file')[0]
+ json_file = blob_info.open().read()
+ json_data = json.loads(json_file)
+
+ # Check the uniqueness of data run_id and store new one.
+ run_id = json_data['run_id']
+ prof_key = ndb.Key('Profiler', run_id)
+ if Profiler.query(ancestor=prof_key).count() == 0:
+ profiler = Profiler(parent=prof_key)
+ profiler.blob_str = str(blob_info.key())
+ profiler.put()
sullivan 2013/09/17 14:58:47 Did you mean to have a hierarchy of Profile object
junjianx 2013/09/20 06:31:09 Done.
+
+ # Check the uniqueness of template content and store new one.
+ for tag, content in json_data['templates'].iteritems():
+ content = json.dumps(content)
+ tmpl_key = ndb.Key('Template', content)
+ if tag == json_data['default_template']:
+ default_key = tmpl_key
+ if (Template.query(ancestor=tmpl_key).count() == 0):
+ template = Template(parent=tmpl_key)
+ template.content = content
+ template.tags = [tag]
+ template.put()
sullivan 2013/09/17 14:58:47 It looks like the templates are already stored in
junjianx 2013/09/20 06:31:09 Template represents one category of breakdown. Tem
+
+ # Jump to new graph page using default template.
+ query_params = {
+ 'prof_id': prof_key.urlsafe(),
+ 'tmpl_id': default_key.urlsafe()
+ }
+ self.redirect('/?' + urllib.urlencode(query_params))
+
+
+application = webapp2.WSGIApplication([
+ ('/', MainPage),
+ ('/upload', UploadHandler)
+], debug=True)
« no previous file with comments | « no previous file | tools/deep_memory_profiler/visualizer/index.html » ('j') | tools/deep_memory_profiler/visualizer/index.html » ('J')

Powered by Google App Engine
This is Rietveld 408576698