Commit 1799e5f3 authored by Tijmen de Mes's avatar Tijmen de Mes

Added support for sylk history api

parent 94a8f926
...@@ -1644,6 +1644,7 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -1644,6 +1644,7 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
notification_center.add_observer(self, name='BlinkGotDispositionNotification') notification_center.add_observer(self, name='BlinkGotDispositionNotification')
notification_center.add_observer(self, name='BlinkMessageDidSucceed') notification_center.add_observer(self, name='BlinkMessageDidSucceed')
notification_center.add_observer(self, name='BlinkMessageDidFail') notification_center.add_observer(self, name='BlinkMessageDidFail')
notification_center.add_observer(self, name='BlinkMessageWillRemove')
notification_center.add_observer(self, name='BlinkMessageHistoryLoadDidSucceed') notification_center.add_observer(self, name='BlinkMessageHistoryLoadDidSucceed')
notification_center.add_observer(self, name='BlinkMessageHistoryLoadDidFail') notification_center.add_observer(self, name='BlinkMessageHistoryLoadDidFail')
notification_center.add_observer(self, name='BlinkMessageHistoryLastContactsDidSucceed') notification_center.add_observer(self, name='BlinkMessageHistoryLastContactsDidSucceed')
...@@ -2386,6 +2387,11 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2386,6 +2387,11 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
message = notification.data.message message = notification.data.message
direction = notification.data.message.direction direction = notification.data.message.direction
try:
history = notification.data.history
except AttributeError:
history = False
try: try:
received_account = notification.data.account received_account = notification.data.account
except AttributeError: except AttributeError:
...@@ -2423,10 +2429,10 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2423,10 +2429,10 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
self.pending_decryption.remove(message) self.pending_decryption.remove(message)
session.chat_widget.update_message_text(message.id, content) session.chat_widget.update_message_text(message.id, content)
else: else:
session.chat_widget.add_message(ChatMessage(content, sender, direction, id=message.id, timestamp=message.timestamp)) session.chat_widget.add_message(ChatMessage(content, sender, direction, id=message.id, timestamp=message.timestamp, history=history))
session.chat_widget.update_message_encryption(message.id, message.is_secure) session.chat_widget.update_message_encryption(message.id, message.is_secure)
else: else:
self.render_after_load.append(ChatMessage(content, sender, direction, id=message.id, timestamp=message.timestamp)) self.render_after_load.append(ChatMessage(content, sender, direction, id=message.id, timestamp=message.timestamp, history=history))
if direction != 'outgoing' and message.disposition is not None and 'display' in message.disposition and not encrypted: if direction != 'outgoing' and message.disposition is not None and 'display' in message.disposition and not encrypted:
if self.selected_session.blink_session is blink_session and not self.isMinimized() and self.isActiveWindow(): if self.selected_session.blink_session is blink_session and not self.isMinimized() and self.isActiveWindow():
...@@ -2476,6 +2482,15 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2476,6 +2482,15 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
session.chat_widget.add_message(ChatStatus(f'Delivery failed: {notification.data.data.code} - {notification.data.data.reason}')) session.chat_widget.add_message(ChatStatus(f'Delivery failed: {notification.data.data.code} - {notification.data.data.reason}'))
session.chat_widget.update_message_status(id=notification.data.id, status='failed') session.chat_widget.update_message_status(id=notification.data.id, status='failed')
def _NH_BlinkMessageWillRemove(self, notification):
blink_session = notification.sender
session = blink_session.items.chat
if session is None:
return
session.chat_widget.remove_message(notification.data)
def _NH_PGPMessageDidDecrypt(self, notification): def _NH_PGPMessageDidDecrypt(self, notification):
blink_session = notification.sender blink_session = notification.sender
session = blink_session.items.chat session = blink_session.items.chat
...@@ -2573,7 +2588,6 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2573,7 +2588,6 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
session.chat_widget.add_message(ChatMessage(content, sender, message.direction, id=message.message_id, timestamp=timestamp, history=True)) session.chat_widget.add_message(ChatMessage(content, sender, message.direction, id=message.message_id, timestamp=timestamp, history=True))
# session.chat_widget.add_message(ChatMessage(content, sender, message.direction, id=message.id, timestamp=timestamp)) # session.chat_widget.add_message(ChatMessage(content, sender, message.direction, id=message.id, timestamp=timestamp))
if message.direction == "outgoing": if message.direction == "outgoing":
session.chat_widget.update_message_status(id=message.message_id, status=message.state) session.chat_widget.update_message_status(id=message.message_id, status=message.state)
elif message.state != 'displayed' and 'display' in message.disposition and not encrypted: elif message.state != 'displayed' and 'display' in message.disposition and not encrypted:
......
...@@ -61,6 +61,8 @@ class HistoryManager(object, metaclass=Singleton): ...@@ -61,6 +61,8 @@ class HistoryManager(object, metaclass=Singleton):
notification_center.add_observer(self, name='BlinkGotDispositionNotification') notification_center.add_observer(self, name='BlinkGotDispositionNotification')
notification_center.add_observer(self, name='BlinkDidSendDispositionNotification') notification_center.add_observer(self, name='BlinkDidSendDispositionNotification')
notification_center.add_observer(self, name='BlinkGotHistoryMessage') notification_center.add_observer(self, name='BlinkGotHistoryMessage')
notification_center.add_observer(self, name='BlinkGotHistoryMessageRemove')
notification_center.add_observer(self, name='BlinkGotHistoryConversationRemove')
@run_in_thread('file-io') @run_in_thread('file-io')
def save(self): def save(self):
...@@ -157,6 +159,12 @@ class HistoryManager(object, metaclass=Singleton): ...@@ -157,6 +159,12 @@ class HistoryManager(object, metaclass=Singleton):
account = notification.sender account = notification.sender
self.message_history.add_from_history(account, **notification.data.__dict__) self.message_history.add_from_history(account, **notification.data.__dict__)
def _NH_BlinkGotHistoryMessageRemove(self, notification):
self.message_history.remove_message(notification.data)
def _NH_BlinkGotHistoryConversationRemove(self, notification):
self.message_history.remove_contact_messages(notification.sender, notification.data)
def _NH_BlinkMessageDidSucceed(self, notification): def _NH_BlinkMessageDidSucceed(self, notification):
data = notification.data data = notification.data
self.message_history.update(data.id, 'accepted') self.message_history.update(data.id, 'accepted')
......
import bisect import bisect
import json
import os import os
import re import re
import requests
import random import random
import urllib
import uuid import uuid
from collections import deque from collections import deque
...@@ -17,9 +20,12 @@ from application.notification import IObserver, NotificationCenter, Notification ...@@ -17,9 +20,12 @@ from application.notification import IObserver, NotificationCenter, Notification
from application.python import Null from application.python import Null
from application.system import makedirs from application.system import makedirs
from application.python.types import Singleton from application.python.types import Singleton
from datetime import timezone
from dateutil.tz import tzlocal
from urllib.parse import urlsplit, urlunsplit, quote
from zope.interface import implementer from zope.interface import implementer
from sipsimple.account import Account, AccountManager from sipsimple.account import Account, AccountManager, BonjourAccount
from sipsimple.addressbook import AddressbookManager, Group, Contact, ContactURI from sipsimple.addressbook import AddressbookManager, Group, Contact, ContactURI
from sipsimple.configuration import DuplicateIDError from sipsimple.configuration import DuplicateIDError
from sipsimple.configuration.settings import SIPSimpleSettings from sipsimple.configuration.settings import SIPSimpleSettings
...@@ -401,7 +407,7 @@ class OutgoingMessage(object): ...@@ -401,7 +407,7 @@ class OutgoingMessage(object):
# TODO # TODO
def send(self): def send(self):
if self.content_type.lower() == 'text/pgp-private-key': if self.content_type.lower() in ['text/pgp-private-key', 'application/sylk-api-token']:
self._lookup() self._lookup()
return return
...@@ -426,7 +432,7 @@ class OutgoingMessage(object): ...@@ -426,7 +432,7 @@ class OutgoingMessage(object):
notification.center.remove_observer(self, sender=notification.sender) notification.center.remove_observer(self, sender=notification.sender)
if notification.sender is self.lookup: if notification.sender is self.lookup:
routes = notification.data.result routes = notification.data.result
if self.content_type.lower() == 'text/pgp-private-key': if self.content_type.lower() in ['text/pgp-private-key', 'application/sylk-api-token']:
self._send(routes) self._send(routes)
return return
...@@ -441,6 +447,9 @@ class OutgoingMessage(object): ...@@ -441,6 +447,9 @@ class OutgoingMessage(object):
public_key = f.read().decode() public_key = f.read().decode()
public_key_message = OutgoingMessage(self.session.account, self.contact, str(public_key), 'text/pgp-public-key', session=self.session) public_key_message = OutgoingMessage(self.session.account, self.contact, str(public_key), 'text/pgp-public-key', session=self.session)
MessageManager()._send_message(public_key_message) MessageManager()._send_message(public_key_message)
if self.account.sms.enable_pgp and not stream.can_encrypt:
lookup_message = OutgoingMessage(self.account, self.contact, 'Public key request', 'application/sylk-api-pgp-key-lookup', session=self.session)
lookup_message.send()
self.session.routes = routes self.session.routes = routes
self._send() self._send()
...@@ -496,6 +505,7 @@ class MessageManager(object, metaclass=Singleton): ...@@ -496,6 +505,7 @@ class MessageManager(object, metaclass=Singleton):
self.sessions = [] self.sessions = []
self._outgoing_message_queue = deque() self._outgoing_message_queue = deque()
self._incoming_encrypted_message_queue = deque() self._incoming_encrypted_message_queue = deque()
self._sync_queue = deque()
self.pgp_requests = RequestList() self.pgp_requests = RequestList()
notification_center = NotificationCenter() notification_center = NotificationCenter()
...@@ -506,6 +516,8 @@ class MessageManager(object, metaclass=Singleton): ...@@ -506,6 +516,8 @@ class MessageManager(object, metaclass=Singleton):
notification_center.add_observer(self, name='PGPKeysDidGenerate') notification_center.add_observer(self, name='PGPKeysDidGenerate')
notification_center.add_observer(self, name='PGPMessageDidNotDecrypt') notification_center.add_observer(self, name='PGPMessageDidNotDecrypt')
notification_center.add_observer(self, name='PGPMessageDidDecrypt') notification_center.add_observer(self, name='PGPMessageDidDecrypt')
notification_center.add_observer(self, name='SIPAccountRegistrationDidSucceed')
notification_center.add_observer(self, name='BlinkServerHistoryWasFetched')
def _add_contact_to_messages_group(self, account, contact): # Maybe this needs to be placed in Contacts? -- Tijmen def _add_contact_to_messages_group(self, account, contact): # Maybe this needs to be placed in Contacts? -- Tijmen
if not account.sms.add_unknown_contacts: if not account.sms.add_unknown_contacts:
...@@ -586,7 +598,13 @@ class MessageManager(object, metaclass=Singleton): ...@@ -586,7 +598,13 @@ class MessageManager(object, metaclass=Singleton):
self.send_imdn_message(session, message.id, message.timestamp, 'delivered') self.send_imdn_message(session, message.id, message.timestamp, 'delivered')
self._add_contact_to_messages_group(session.account, session.contact) self._add_contact_to_messages_group(session.account, session.contact)
notification_center.post_notification('BlinkGotMessage', sender=session, data=NotificationData(message=message)) notification_center.post_notification('BlinkGotMessage', sender=session, data=NotificationData(message=message, account=account))
def _request_history_synchronization_token(self, account):
from blink.contacts import URIUtils
contact, contact_uri = URIUtils.find_contact(account.uri)
outgoing_message = OutgoingMessage(account, contact, 'Token request', 'application/sylk-api-token')
self._send_message(outgoing_message)
def _send_message(self, outgoing_message): def _send_message(self, outgoing_message):
self._outgoing_message_queue.append(outgoing_message) self._outgoing_message_queue.append(outgoing_message)
...@@ -597,6 +615,151 @@ class MessageManager(object, metaclass=Singleton): ...@@ -597,6 +615,151 @@ class MessageManager(object, metaclass=Singleton):
message = self._outgoing_message_queue.popleft() message = self._outgoing_message_queue.popleft()
message.send() message.send()
@run_in_thread('sync')
def _sync_messages(self, account):
if not account.sms.enable_history_synchronization:
return
if not account.sms.history_synchronization_token:
self._request_history_synchronization_token(account)
return
if not account.sms.history_synchronization_url:
return
if account.sms.history_synchronization_id is not None:
url = urllib.parse.urljoin(f'{account.sms.history_synchronization_url}/', account.sms.history_synchronization_id)
else:
url = account.sms.history_synchronization_url
scheme, netloc, path, query, fragment = urlsplit(url)
path = quote(path)
url = urlunsplit((scheme, netloc, path, query, fragment))
headers = {'Authorization': f'Apikey {account.sms.history_synchronization_token}'}
log.info(f'History synchronization enabled for {account.id}, fetching from: {url}')
try:
r = requests.get(url, headers=headers, timeout=10)
r.raise_for_status()
except (requests.ConnectionError, requests.Timeout) as e:
log.warning(f'SylkServer API connection error: {e}')
except requests.HTTPError as e:
code = e.response.status_code
if code == 401:
self._request_history_synchronization_token(account)
return
log.warning(f'SylkServer API error {e}')
except requests.RequestException as e:
log.warning(f'SylkServer API error {e}')
else:
try:
data = r.json()
except json.JSONDecodeError:
pass
else:
notification_center = NotificationCenter()
notification_center.post_notification('BlinkServerHistoryWasFetched', sender=account, data=data)
@run_in_thread('sync')
def _process_server_history_messages(self, account, messages):
notification_center = NotificationCenter()
last_id = None
log.debug(f'-- Number of messages fetched for {account.id}: {len(messages)}')
while messages:
message = messages.pop(0)
last_id = message['message_id']
content_type = message['content_type'].lower()
if content_type == 'message/imdn':
payload = json.loads(message['content'])
data = NotificationData(id=payload['message_id'], status=message['state'])
kwargs = {'data': data}
from blink.contacts import URIUtils
contact, contact_uri = URIUtils.find_contact(message['contact'])
try:
blink_session = next(session for session in self.sessions if session.contact.settings is contact.settings)
except StopIteration:
pass
else:
kwargs['sender'] = blink_session
notification_center.post_notification('BlinkGotDispositionNotification', **kwargs)
elif content_type == 'application/sylk-conversation-remove':
notification_center.post_notification('BlinkGotHistoryConversationRemove', sender=account, data=message['content'])
elif content_type == 'application/sylk-message-remove':
payload = json.loads(message['content'])
notification_center.post_notification('BlinkGotHistoryMessageRemove', data=payload['message_id'])
from blink.contacts import URIUtils
contact, contact_uri = URIUtils.find_contact(message['contact'])
try:
blink_session = next(session for session in self.sessions if session.contact.settings is contact.settings)
except StopIteration:
pass
else:
notification_center.post_notification('BlinkMessageWillRemove', sender=blink_session, data=payload['message_id'])
elif content_type == 'application/sylk-conversation-read':
pass
elif content_type == 'text/pgp-public-key':
if message.contact != account.id:
self._save_pgp_key(message['content'], message['contact'])
elif content_type.startswith('text/'):
from blink.contacts import URIUtils
contact, contact_uri = URIUtils.find_contact(message['contact'])
sender = account
if message['direction'] == 'incoming':
sender = ChatIdentity(SIPURI.parse(f'sip:{contact.uri.uri}'), contact.name)
timestamp = ISOTimestamp(message['timestamp']).replace(tzinfo=timezone.utc).astimezone(tzlocal())
history_message = BlinkMessage(message['content'],
message['content_type'],
sender,
timestamp=timestamp,
id=message['message_id'],
disposition=message['disposition'],
direction=message['direction'])
encryption = self.check_encryption(history_message.content_type, history_message.content)
notification_center.post_notification('BlinkGotHistoryMessage',
sender=account,
data=NotificationData(
remote_uri=message['contact'],
message=history_message,
encryption=encryption,
state=message['state']))
self._add_contact_to_messages_group(account, contact)
try:
blink_session = next(session for session in self.sessions if session.contact.settings is contact.settings)
except StopIteration:
pass
else:
if ['direction'] == 'incoming' and 'positive-delivery' in history_message.disposition:
log.debug("-- Should send delivered imdn for history message")
self.send_imdn_message(blink_session, history_message.id, history_message.timestamp, 'delivered')
notification_center.post_notification('BlinkGotMessage',
sender=blink_session,
data=NotificationData(
message=history_message,
history=True,
account=account))
if encryption == 'OpenPGP':
if blink_session.fake_streams.get('messages').can_decrypt:
blink_session.fake_streams.get('messages').decrypt(history_message)
else:
self._incoming_encrypted_message_queue.append((history_message, account, contact))
if last_id is not None:
account.sms.history_synchronization_id = last_id
account.save()
@run_in_gui_thread @run_in_gui_thread
def handle_notification(self, notification): def handle_notification(self, notification):
handler = getattr(self, '_NH_%s' % notification.name, Null) handler = getattr(self, '_NH_%s' % notification.name, Null)
...@@ -665,6 +828,13 @@ class MessageManager(object, metaclass=Singleton): ...@@ -665,6 +828,13 @@ class MessageManager(object, metaclass=Singleton):
request.dialog.hide() request.dialog.hide()
self.pgp_requests.remove(request) self.pgp_requests.remove(request)
def _NH_SIPAccountRegistrationDidSucceed(self, notification):
if notification.sender is not BonjourAccount():
self._sync_queue.append(notification.sender)
while self._sync_queue:
sender = self._sync_queue.popleft()
self._sync_messages(sender)
def _NH_SIPEngineGotMessage(self, notification): def _NH_SIPEngineGotMessage(self, notification):
account_manager = AccountManager() account_manager = AccountManager()
account = account_manager.find_account(notification.data.request_uri) account = account_manager.find_account(notification.data.request_uri)
...@@ -721,8 +891,25 @@ class MessageManager(object, metaclass=Singleton): ...@@ -721,8 +891,25 @@ class MessageManager(object, metaclass=Singleton):
elif not account.sms.enable_pgp: elif not account.sms.enable_pgp:
log.info(f"-- Skipping PGP encrypted message, PGP is disabled for {account.id}") log.info(f"-- Skipping PGP encrypted message, PGP is disabled for {account.id}")
return return
if content_type.lower() == 'application/sylk-api-token':
log.info('Message is a Sylk API token')
try:
data = json.loads(body)
except json.decoder.JSONDecodeError:
return
try:
token = data['token']
url = data['url']
except KeyError:
return return
account.sms.history_synchronization_token = token
account.sms.history_synchronization_url = url
account.save()
self._sync_messages(account)
return
if content_type.lower() == 'text/pgp-private-key': if content_type.lower() == 'text/pgp-private-key':
log.info('Message is a private key') log.info('Message is a private key')
...@@ -851,6 +1038,11 @@ class MessageManager(object, metaclass=Singleton): ...@@ -851,6 +1038,11 @@ class MessageManager(object, metaclass=Singleton):
self._handle_incoming_message(message, blink_session, account) self._handle_incoming_message(message, blink_session, account)
def _NH_BlinkServerHistoryWasFetched(self, notification):
account = notification.sender
messages = notification.data['messages']
self._process_server_history_messages(account, messages)
def _NH_BlinkSessionWasCreated(self, notification): def _NH_BlinkSessionWasCreated(self, notification):
session = notification.sender session = notification.sender
self.sessions.append(session) self.sessions.append(session)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment