Add message translation layer than can handle "me" as an ACL.
This commit is contained in:
20
api.py
20
api.py
@@ -63,12 +63,13 @@ def Pin(google_user, client, client_address, instance_id, args):
|
|||||||
sender_message_id = args['sender_message_id']
|
sender_message_id = args['sender_message_id']
|
||||||
|
|
||||||
try:
|
try:
|
||||||
pin = models.Subject.FindOrCreate(subject).Pin(
|
pin = models.Subject.FindOrCreate(subject, client).Pin(
|
||||||
message,
|
message,
|
||||||
models.Client.profile.get_value_for_datastore(client),
|
models.Client.profile.get_value_for_datastore(client),
|
||||||
sender_message_id,
|
sender_message_id,
|
||||||
client_address,
|
client_address,
|
||||||
instance)
|
instance,
|
||||||
|
subject)
|
||||||
except models.DuplicateMessage as e:
|
except models.DuplicateMessage as e:
|
||||||
logging.warning('Duplicate pin: %s', sender_message_id)
|
logging.warning('Duplicate pin: %s', sender_message_id)
|
||||||
return {
|
return {
|
||||||
@@ -93,11 +94,12 @@ def SendMessage(google_user, client, client_address, instance_id, args):
|
|||||||
sender_message_id = args['sender_message_id']
|
sender_message_id = args['sender_message_id']
|
||||||
|
|
||||||
try:
|
try:
|
||||||
msg = models.Subject.FindOrCreate(subject).SendMessage(
|
msg = models.Subject.FindOrCreate(subject, client).SendMessage(
|
||||||
message,
|
message,
|
||||||
models.Client.profile.get_value_for_datastore(client),
|
models.Client.profile.get_value_for_datastore(client),
|
||||||
sender_message_id,
|
sender_message_id,
|
||||||
client_address)
|
client_address,
|
||||||
|
subject)
|
||||||
except models.DuplicateMessage as e:
|
except models.DuplicateMessage as e:
|
||||||
logging.warning('Duplicate message: %s', sender_message_id)
|
logging.warning('Duplicate message: %s', sender_message_id)
|
||||||
return {
|
return {
|
||||||
@@ -118,7 +120,7 @@ def SendMessage(google_user, client, client_address, instance_id, args):
|
|||||||
|
|
||||||
def Subscribe(google_user, client, client_address, instance_id, args):
|
def Subscribe(google_user, client, client_address, instance_id, args):
|
||||||
instance = models.Instance.FromID(instance_id)
|
instance = models.Instance.FromID(instance_id)
|
||||||
subject = models.Subject.FindOrCreate(args['subject'])
|
subject = models.Subject.FindOrCreate(args['subject'], client)
|
||||||
messages = args.get('messages', 0)
|
messages = args.get('messages', 0)
|
||||||
last_id = args.get('last_id', None)
|
last_id = args.get('last_id', None)
|
||||||
|
|
||||||
@@ -141,7 +143,7 @@ def Subscribe(google_user, client, client_address, instance_id, args):
|
|||||||
return {
|
return {
|
||||||
'result': 'ok',
|
'result': 'ok',
|
||||||
'events': models.Subscription.FindOrCreate(
|
'events': models.Subscription.FindOrCreate(
|
||||||
subject, client, instance, messages, last_id),
|
subject, client, instance, args['subject'], messages, last_id),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -151,7 +153,7 @@ def Unpin(google_user, client, client_address, instance_id, args):
|
|||||||
sender_message_id = args['sender_message_id']
|
sender_message_id = args['sender_message_id']
|
||||||
|
|
||||||
try:
|
try:
|
||||||
models.Subject.FindOrCreate(subject).Unpin(
|
models.Subject.FindOrCreate(subject, client).Unpin(
|
||||||
models.Client.profile.get_value_for_datastore(client),
|
models.Client.profile.get_value_for_datastore(client),
|
||||||
sender_message_id,
|
sender_message_id,
|
||||||
instance.key())
|
instance.key())
|
||||||
@@ -168,8 +170,8 @@ def Unpin(google_user, client, client_address, instance_id, args):
|
|||||||
|
|
||||||
def Unsubscribe(google_user, client, client_address, instance_id, args):
|
def Unsubscribe(google_user, client, client_address, instance_id, args):
|
||||||
instance = models.Instance.FromID(instance_id)
|
instance = models.Instance.FromID(instance_id)
|
||||||
subject = models.Subject.FindOrCreate(args['subject'])
|
subject = models.Subject.FindOrCreate(args['subject'], client)
|
||||||
models.Subscription.Remove(subject, instance)
|
models.Subscription.Remove(subject, instance, args['subject'])
|
||||||
|
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
|
|||||||
@@ -35,7 +35,8 @@ class OnChannelConnect(webapp2.RequestHandler):
|
|||||||
message = {
|
message = {
|
||||||
'event_type': 'close',
|
'event_type': 'close',
|
||||||
}
|
}
|
||||||
channel.send_message(instance_id, json.dumps(msg, default=utils.EncodeJSON))
|
channel.send_message(
|
||||||
|
instance_id, json.dumps(message, default=utils.EncodeJSON))
|
||||||
return
|
return
|
||||||
instance.active = True
|
instance.active = True
|
||||||
instance.put()
|
instance.put()
|
||||||
|
|||||||
@@ -130,10 +130,12 @@ class Subject(db.Model):
|
|||||||
return hashobj.hexdigest()
|
return hashobj.hexdigest()
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def FindOrCreate(cls, subject):
|
def FindOrCreate(cls, subject, client):
|
||||||
if 'readable_only_by' in subject:
|
if 'readable_only_by' in subject:
|
||||||
if subject['readable_only_by'] == 'admin':
|
if subject['readable_only_by'] == 'admin':
|
||||||
readable_only_by = Profile.ADMIN_KEY
|
readable_only_by = Profile.ADMIN_KEY
|
||||||
|
elif subject['readable_only_by'] == 'me':
|
||||||
|
readable_only_by = Client.profile.get_value_for_datastore(client)
|
||||||
else:
|
else:
|
||||||
readable_only_by = db.Key(subject['readable_only_by'])
|
readable_only_by = db.Key(subject['readable_only_by'])
|
||||||
else:
|
else:
|
||||||
@@ -142,6 +144,8 @@ class Subject(db.Model):
|
|||||||
if 'writable_only_by' in subject:
|
if 'writable_only_by' in subject:
|
||||||
if subject['writable_only_by'] == 'admin':
|
if subject['writable_only_by'] == 'admin':
|
||||||
writable_only_by = Profile.ADMIN_KEY
|
writable_only_by = Profile.ADMIN_KEY
|
||||||
|
elif subject['writable_only_by'] == 'me':
|
||||||
|
writable_only_by = Client.profile.get_value_for_datastore(client)
|
||||||
else:
|
else:
|
||||||
writable_only_by = db.Key(subject['writable_only_by'])
|
writable_only_by = db.Key(subject['writable_only_by'])
|
||||||
else:
|
else:
|
||||||
@@ -238,14 +242,22 @@ class Subject(db.Model):
|
|||||||
readable_only_by != reader):
|
readable_only_by != reader):
|
||||||
raise AccessDenied
|
raise AccessDenied
|
||||||
|
|
||||||
def SendMessage(self, message, sender, sender_message_id, sender_address):
|
def SendMessage(
|
||||||
|
self, message, sender, sender_message_id, sender_address, request):
|
||||||
self.VerifyWritable(sender)
|
self.VerifyWritable(sender)
|
||||||
|
readable_only_by_me = (request.get('readable_only_by') == 'me')
|
||||||
|
writable_only_by_me = (request.get('writable_only_by') == 'me')
|
||||||
|
try:
|
||||||
obj, subscriptions = self.PutMessage(
|
obj, subscriptions = self.PutMessage(
|
||||||
message, sender, sender_message_id, sender_address)
|
message, sender, sender_message_id, sender_address)
|
||||||
|
except DuplicateMessage as e:
|
||||||
|
e.original = self.TranslateEvent(
|
||||||
|
e.original, readable_only_by_me, writable_only_by_me)
|
||||||
|
raise e
|
||||||
event = obj.ToEvent()
|
event = obj.ToEvent()
|
||||||
for subscription in subscriptions:
|
for subscription in subscriptions:
|
||||||
subscription.SendMessage(event)
|
subscription.SendMessage(event)
|
||||||
return event
|
return self.TranslateEvent(event, readable_only_by_me, writable_only_by_me)
|
||||||
|
|
||||||
@db.transactional()
|
@db.transactional()
|
||||||
def PutPin(self, message, sender, sender_message_id,
|
def PutPin(self, message, sender, sender_message_id,
|
||||||
@@ -273,14 +285,22 @@ class Subject(db.Model):
|
|||||||
|
|
||||||
return (obj, list(Subscription.all().ancestor(self)))
|
return (obj, list(Subscription.all().ancestor(self)))
|
||||||
|
|
||||||
def Pin(self, message, sender, sender_message_id, sender_address, instance):
|
def Pin(self, message, sender, sender_message_id, sender_address, instance,
|
||||||
|
request):
|
||||||
self.VerifyWritable(sender)
|
self.VerifyWritable(sender)
|
||||||
|
readable_only_by_me = (request.get('readable_only_by') == 'me')
|
||||||
|
writable_only_by_me = (request.get('writable_only_by') == 'me')
|
||||||
|
try:
|
||||||
obj, subscriptions = self.PutPin(
|
obj, subscriptions = self.PutPin(
|
||||||
message, sender, sender_message_id, instance, sender_address)
|
message, sender, sender_message_id, instance, sender_address)
|
||||||
|
except DuplicateMessage as e:
|
||||||
|
e.original = self.TranslateEvent(
|
||||||
|
e.original, readable_only_by_me, writable_only_by_me)
|
||||||
|
raise e
|
||||||
event = obj.ToEvent()
|
event = obj.ToEvent()
|
||||||
for subscription in subscriptions:
|
for subscription in subscriptions:
|
||||||
subscription.SendMessage(event)
|
subscription.SendMessage(event)
|
||||||
return event
|
return self.TranslateEvent(event, readable_only_by_me, writable_only_by_me)
|
||||||
|
|
||||||
@db.transactional()
|
@db.transactional()
|
||||||
def RemovePin(self, sender, sender_message_id, instance_key):
|
def RemovePin(self, sender, sender_message_id, instance_key):
|
||||||
@@ -323,6 +343,19 @@ class Subject(db.Model):
|
|||||||
ret['writable_only_by'] = str(writable_only_by)
|
ret['writable_only_by'] = str(writable_only_by)
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def TranslateEvent(cls, event, readable_only_by_me, writable_only_by_me):
|
||||||
|
if readable_only_by_me:
|
||||||
|
event['subject']['readable_only_by'] = 'me'
|
||||||
|
if writable_only_by_me:
|
||||||
|
event['subject']['writable_only_by'] = 'me'
|
||||||
|
return event
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def TranslateEvents(cls, events, readable_only_by_me, writable_only_by_me):
|
||||||
|
return [cls.TranslateEvent(event, readable_only_by_me, writable_only_by_me)
|
||||||
|
for event in events]
|
||||||
|
|
||||||
@db.transactional()
|
@db.transactional()
|
||||||
def GetEvents(self, messages, last_id):
|
def GetEvents(self, messages, last_id):
|
||||||
events = [m.ToEvent() for m in self.GetPins()]
|
events = [m.ToEvent() for m in self.GetPins()]
|
||||||
@@ -337,26 +370,39 @@ class Subscription(db.Model):
|
|||||||
# parent=Subject
|
# parent=Subject
|
||||||
|
|
||||||
instance = db.ReferenceProperty(reference_class=Instance, required=True)
|
instance = db.ReferenceProperty(reference_class=Instance, required=True)
|
||||||
|
readable_only_by_me = db.BooleanProperty(required=True, default=False)
|
||||||
|
writable_only_by_me = db.BooleanProperty(required=True, default=False)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@db.transactional()
|
@db.transactional()
|
||||||
def FindOrCreate(cls, subject, client, instance, messages=0, last_id=None):
|
def FindOrCreate(cls, subject, client, instance, request,
|
||||||
|
messages=0, last_id=None):
|
||||||
|
readable_only_by_me = (request.get('readable_only_by') == 'me')
|
||||||
|
writable_only_by_me = (request.get('writable_only_by') == 'me')
|
||||||
subscriptions = (
|
subscriptions = (
|
||||||
cls.all(keys_only=True)
|
cls.all(keys_only=True)
|
||||||
.ancestor(subject)
|
.ancestor(subject)
|
||||||
.filter('instance =', instance)
|
.filter('instance =', instance)
|
||||||
|
.filter('readable_only_by_me =', readable_only_by_me)
|
||||||
|
.filter('writable_only_by_me =', writable_only_by_me)
|
||||||
.fetch(1))
|
.fetch(1))
|
||||||
if not subscriptions:
|
if not subscriptions:
|
||||||
cls(parent=subject, instance=instance).put()
|
cls(parent=subject, instance=instance).put()
|
||||||
return subject.GetEvents(messages, last_id)
|
return subject.TranslateEvents(
|
||||||
|
subject.GetEvents(messages, last_id),
|
||||||
|
readable_only_by_me, writable_only_by_me)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@db.transactional()
|
@db.transactional()
|
||||||
def Remove(cls, subject, instance):
|
def Remove(cls, subject, instance, request):
|
||||||
|
readable_only_by_me = (request.get('readable_only_by') == 'me')
|
||||||
|
writable_only_by_me = (request.get('writable_only_by') == 'me')
|
||||||
subscriptions = (
|
subscriptions = (
|
||||||
cls.all()
|
cls.all()
|
||||||
.ancestor(subject)
|
.ancestor(subject)
|
||||||
.filter('instance =', instance))
|
.filter('instance =', instance)
|
||||||
|
.filter('readable_only_by_me =', readable_only_by_me)
|
||||||
|
.filter('writable_only_by_me =', writable_only_by_me))
|
||||||
for subscription in subscriptions:
|
for subscription in subscriptions:
|
||||||
subscription.delete()
|
subscription.delete()
|
||||||
|
|
||||||
|
|||||||
@@ -470,6 +470,36 @@ asyncTest('sendMessage ACL', function() {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
asyncTest('"me" ACL', function() {
|
||||||
|
expect(7);
|
||||||
|
|
||||||
|
var subject = {
|
||||||
|
'name': randstring(),
|
||||||
|
'readable_only_by': 'me',
|
||||||
|
'writable_only_by': 'me'
|
||||||
|
};
|
||||||
|
var message = randstring();
|
||||||
|
|
||||||
|
var callbacks = {
|
||||||
|
'onMessage': function(e) {
|
||||||
|
equal(e['subject']['name'], subject['name'], 'subject matches');
|
||||||
|
equal(e['subject']['readable_only_by'], 'me', 'readable_only_by matches');
|
||||||
|
equal(e['subject']['writable_only_by'], 'me', 'writable_only_by matches');
|
||||||
|
equal(e['message'], message, 'message matches');
|
||||||
|
cosmo.shutdown();
|
||||||
|
start();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var cosmo = new Cosmopolite(callbacks, null, randstring());
|
||||||
|
cosmo.sendMessage(subject, message).then(function(msg) {
|
||||||
|
equal(msg['subject']['name'], subject['name'], 'subject matches');
|
||||||
|
equal(msg['subject']['readable_only_by'], 'me', 'readable_only_by matches');
|
||||||
|
equal(msg['subject']['writable_only_by'], 'me', 'writable_only_by matches');
|
||||||
|
});
|
||||||
|
cosmo.subscribe(subject, -1);
|
||||||
|
});
|
||||||
|
|
||||||
asyncTest('pin/unpin', function() {
|
asyncTest('pin/unpin', function() {
|
||||||
expect(5);
|
expect(5);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user