diff --git a/app.yaml b/app.yaml new file mode 100644 index 0000000..419c816 --- /dev/null +++ b/app.yaml @@ -0,0 +1,23 @@ +application: k-sketch-test +version: 1 +runtime: python27 +api_version: 1 +threadsafe: false + +handlers: +- url: /app + static_dir: app + expiration: 1m + +- url: /test + static_dir: test + expiration: 1m + +- url: /.* + script: backend.application + +libraries: +- name: webapp2 + version: "2.5.1" + + diff --git a/backend.py b/backend.py new file mode 100644 index 0000000..8fa25d5 --- /dev/null +++ b/backend.py @@ -0,0 +1,349 @@ +"""Backend Module + +Created on Dec 6, 2012 +@author: Chris Boesch +""" +""" +Note to self: json.loads = json string to objects. json.dumps is object to json string. +""" +import datetime +import logging + +import webapp2 as webapp + +from google.appengine.api import memcache +from google.appengine.ext import db +from google.appengine.ext.webapp.util import run_wsgi_app + +import json + +class Backend(db.Model): + apikey = db.StringProperty(required=True,default='Default-APIKey') + model = db.StringProperty(required=True,default='Default-Model') + #Use backend record id as the model id for simplicity + jsonString = db.TextProperty(required=True,default='{}') + created = db.DateTimeProperty(auto_now_add=True) #The time that the model was created + modified = db.DateTimeProperty(auto_now=True) + + def to_dict(self): + d = dict([(p, unicode(getattr(self, p))) for p in self.properties()]) + d["id"] = self.key().id() + return d + + + @staticmethod + def add(apikey, model, data): + #update ModelCount when adding + jsonString = data + entity = Backend(apikey=apikey, + model=model, + jsonString=jsonString) + + entity.put() + modelCount = ModelCount.all().filter('apikey',apikey).filter('model', model).get() + if modelCount: + modelCount.count += 1 + modelCount.put() + else: + modelCount = ModelCount(apikey=apikey, model=model, count=1) + modelCount.put() + + result = {'model':model, + 'apikey': apikey, + 'id': entity.key().id(), + 'data': json.loads(jsonString)} #this would also check if the json submitted was valid + + return result + + @staticmethod + def get_entities(apikey, model=None, offset=0, limit=50): + #update ModelCount when adding + theQuery = Backend.all().filter('apikey',apikey) + if model: + theQuery = theQuery.filter('model', model) + + objects = theQuery.fetch(limit=limit, offset=offset) + + entities = [] + for object in objects: + entity = {'model':object.model, + 'apikey': apikey, + 'id': object.key().id(), + 'created': object.created, + 'modified': object.modified, + 'data': json.loads(object.jsonString)} + entities.append(entity) + + count = 0 + modelCount = ModelCount.all().filter('apikey',apikey).filter('model', model).get() + if modelCount: + count = modelCount.count + result = {'method':'get_entities', + 'apikey': apikey, + 'model': model, + 'count': count, + 'offset': offset, + 'limit':limit, + 'entities': entities} + return result + + @staticmethod + def get_entity(apikey,model,model_id): + theobject = Backend.get_by_id(int(model_id)) + + result = {'method':'get_model', + 'apikey': apikey, + 'model': model, + 'id': model_id, + 'data': json.loads(theobject.jsonString) + } + return result + + @staticmethod + def clear(apikey, model): + #update model count when clearing model on api + count = 0 + for object in Backend.all().filter('apikey',apikey).filter('model', model): + count += 1 + object.delete() + + modelCount = ModelCount.all().filter('apikey',apikey).filter('model', model).get() + if modelCount: + modelCount.delete() + result = {'items_deleted': count} + return result + + @staticmethod + def clearapikey(apikey): + #update model count when clearing model on api + count = 0 + for object in Backend.all().filter('apikey',apikey): + count += 1 + object.delete() + + modelCount = ModelCount.all().filter('apikey',apikey).get() + if modelCount: + modelCount.delete() + result = {'items_deleted': count} + return result + + #You can't name it delete since db.Model already has a delete method + @staticmethod + def remove(apikey, model, model_id): + #update model count when deleting + entity = Backend.get_by_id(int(model_id)) + + if entity and entity.apikey == apikey and entity.model == model: + entity.delete() + + result = {'method':'delete_model_success', + 'apikey': apikey, + 'model': model, + 'id': model_id + } + else: + result = {'method':'delete_model_not_found'} + + modelCount = ModelCount.all().filter('apikey',apikey).filter('model', model).get() + if modelCount: + modelCount.count -= 1 + modelCount.put() + + return result + + #data is a dictionary that must be merged with current json data and stored. + @staticmethod + def edit_entity(apikey, model, model_id, data): + jsonString = data + entity = Backend.get_by_id(int(model_id)) + entity.jsonString = jsonString + entity.put() + if entity.jsonString: + data = json.loads(entity.jsonString) + else: + data = {} + result = {'model':model, + 'apikey': apikey, + 'id': entity.key().id(), + 'data': data #this would also check if the json submitted was valid + } + return result + +#Quick retrieval for supported models metadata and count stats +class ModelCount(db.Model): + apikey = db.StringProperty(required=True,default='Default-APIKey') + model = db.StringProperty(required=True,default='Default-Model') + count = db.IntegerProperty(required=True, default=0) + created = db.DateTimeProperty(auto_now_add=True) #The time that the model was created + modified = db.DateTimeProperty(auto_now=True) + + +class ActionHandler(webapp.RequestHandler): + """Class which handles bootstrap procedure and seeds the necessary + entities in the datastore. + """ + + def respond(self,result): + """Returns a JSON response to the client. + """ + callback = self.request.get('callback') + self.response.headers['Content-Type'] = 'application/json' + #self.response.headers['Content-Type'] = '%s; charset=%s' % (config.CONTENT_TYPE, config.CHARSET) + self.response.headers['Access-Control-Allow-Origin'] = '*' + self.response.headers['Access-Control-Allow-Methods'] = 'GET, POST, PUT, DELETE, OPTIONS, PATCH, HEAD' + self.response.headers['Access-Control-Allow-Headers'] = 'Origin, Content-Type, X-Requested-With' + self.response.headers['Access-Control-Allow-Credentials'] = 'True' + + #Add a handler to automatically convert datetimes to ISO 8601 strings. + dthandler = lambda obj: obj.isoformat() if isinstance(obj, datetime.datetime) else None + if callback: + content = str(callback) + '(' + json.dumps(result,default=dthandler) + ')' + return self.response.out.write(content) + + return self.response.out.write(json.dumps(result,default=dthandler)) + + def metadata(self,apikey): + #Fetch all ModelCount records for apikey to produce metadata on currently supported models. + models = [] + for mc in ModelCount.all().filter('apikey',apikey): + models.append({'model':mc.model, 'count': mc.count}) + + result = {'method':'metadata', + 'apikey': apikey, + 'model': "metadata", + 'count': len(models), + 'entities': models + } + + return self.respond(result) + + #Dump apikey table + def backup(self,apikey): + #Fetch all ModelCount records for apikey to produce metadata on currently supported models. + dthandler = lambda obj: obj.isoformat() if isinstance(obj, datetime.datetime) else None + + offset = 0 + new_offset = self.request.get("offset") + if new_offset: + offset = int(new_offset) + + result = Backend.get_entities(apikey,offset=offset) + + filename = "Backup_"+apikey+"_offset_"+str(offset)+".json" + self.response.headers['Content-Type'] = 'application/streaming-json' + self.response.content_disposition = 'attachment; filename="'+filename+'"' + + for obj in result['entities']: + self.response.out.write(json.dumps(obj,default=dthandler)+"\n") + return + + #Delete this experimental backup method + def backup_test(self,apikey): + #Fetch all ModelCount records for apikey to produce metadata on currently supported models. + dthandler = lambda obj: obj.isoformat() if isinstance(obj, datetime.datetime) else None + + offset = 0 + new_offset = self.request.get("offset") + if new_offset: + offset = int(new_offset) + + filename = "Backup_test_"+apikey+"_offset_"+str(offset)+".json" + self.response.headers['Content-Type'] = 'application/streaming-json' + self.response.content_disposition = 'attachment; filename="'+filename+'"' + + for entity in Backend.all(): + self.response.out.write(json.dumps(entity.to_dict(),default=dthandler)+"\n") + return + + #return self.respond(result) + + + def clear_apikey(self,apikey): + """Clears the datastore for a an apikey. + """ + result = Backend.clearapikey(apikey) + return self.respond({'method':'clear_apikey'}) + + def clear_model(self,apikey, model): + """Clears the datastore for a model and apikey. + """ + result = Backend.clear(apikey, model) + return self.respond(result) + + def add_or_list_model(self,apikey,model): + #Check for GET paramenter == model to see if this is an add or list. + #Call Backend.add(apikey, model, data) or + #Fetch all models for apikey and return a list. + + #Todo - Check for method. + logging.info(self.request.method) + if self.request.method=="POST": + logging.info("in POST") + logging.info(self.request.body) + result = Backend.add(apikey, model, self.request.body) + #logging.info(result) + return self.respond(result) + + else: + data = self.request.get("obj") + if data: + logging.info("Adding new data: "+data) + result = Backend.add(apikey, model, data) + else: + offset = 0 + new_offset = self.request.get("offset") + if new_offset: + offset = int(new_offset) + + result = Backend.get_entities(apikey, model,offset=offset) + + return self.respond(result) + + def delete_model(self,apikey,model, model_id): + result = Backend.remove(apikey,model, model_id) + + return self.respond(result) + + def get_or_edit_model(self,apikey,model, model_id): + #Check for GET parameter == model to see if this is a get or an edit + #technically the apikey and model are not required. + #To create an error message if the id is not from this apikey? + logging.info("**********************") + logging.info(self.request.method) + logging.info("**********************") + + if self.request.method=="DELETE": + logging.info("It was options") + result = Backend.remove(apikey,model, model_id) + logging.info(result) + return self.respond(result)#(result) + + elif self.request.method=="PUT": + logging.info("It was PUT") + logging.info(self.request.body) + result = Backend.edit_entity(apikey,model,model_id,self.request.body) + #result = Backend.remove(apikey,model, model_id) + #result = json.loads(self.request.body) + #logging.info(result) + return self.respond(result)#(result) + else: + data = self.request.get("obj") + if data: + result = Backend.edit_entity(apikey,model,model_id,data) + else: + result = Backend.get_entity(apikey,model,model_id) + return self.respond(result) + +application = webapp.WSGIApplication([ + webapp.Route('//metadata', handler=ActionHandler, handler_method='metadata'), + webapp.Route('//backup_test', handler=ActionHandler, handler_method='backup_test'), + webapp.Route('//backup', handler=ActionHandler, handler_method='backup'), + webapp.Route('//clear', handler=ActionHandler, handler_method='clear_apikey'), + webapp.Route('///clear', handler=ActionHandler, handler_method='clear_model'), + webapp.Route('////delete', handler=ActionHandler, handler_method='delete_model'), + webapp.Route('///', handler=ActionHandler, handler_method='get_or_edit_model'), + webapp.Route('//', handler=ActionHandler, handler_method='add_or_list_model'), + ], + debug=True) + +