Commit 160162cb authored by Tijmen de Mes's avatar Tijmen de Mes

Store/Process encrypted messages when private key is not set yet

parent aae1a0a9
...@@ -14,7 +14,7 @@ from PyQt5.QtWebKitWidgets import QWebPage, QWebView ...@@ -14,7 +14,7 @@ from PyQt5.QtWebKitWidgets import QWebPage, QWebView
from PyQt5.QtWidgets import QApplication, QAction, QLabel, QListView, QMenu, QStyle, QStyleOption, QStylePainter, QTextEdit, QToolButton from PyQt5.QtWidgets import QApplication, QAction, QLabel, QListView, QMenu, QStyle, QStyleOption, QStylePainter, QTextEdit, QToolButton
from abc import ABCMeta, abstractmethod from abc import ABCMeta, abstractmethod
from application.notification import IObserver, NotificationCenter, ObserverWeakrefProxy from application.notification import IObserver, NotificationCenter, ObserverWeakrefProxy, NotificationData
from application.python import Null, limit from application.python import Null, limit
from application.python.descriptor import WriteOnceAttribute from application.python.descriptor import WriteOnceAttribute
from application.python.types import MarkerType from application.python.types import MarkerType
...@@ -1582,7 +1582,7 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -1582,7 +1582,7 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
self.pending_displayed_notifications = {} self.pending_displayed_notifications = {}
self.render_after_load = [] self.render_after_load = []
self.pending_decryption = []
notification_center = NotificationCenter() notification_center = NotificationCenter()
notification_center.add_observer(self, name='SIPApplicationDidStart') notification_center.add_observer(self, name='SIPApplicationDidStart')
notification_center.add_observer(self, name='BlinkSessionNewIncoming') notification_center.add_observer(self, name='BlinkSessionNewIncoming')
...@@ -1610,6 +1610,8 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -1610,6 +1610,8 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
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')
notification_center.add_observer(self, name='MessageStreamPGPKeysDidLoad')
notification_center.add_observer(self, name='PGPMessageDidDecrypt')
notification_center.add_observer(self, name='PGPMessageDidNotDecrypt') notification_center.add_observer(self, name='PGPMessageDidNotDecrypt')
# self.splitter.splitterMoved.connect(self._SH_SplitterMoved) # check this and decide on what size to have in the window (see Notes) -Dan # self.splitter.splitterMoved.connect(self._SH_SplitterMoved) # check this and decide on what size to have in the window (see Notes) -Dan
...@@ -2347,11 +2349,21 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2347,11 +2349,21 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
if session is None: if session is None:
return return
encrypted = False
if message.content_type.startswith('image/'): if message.content_type.startswith('image/'):
content = '''<img src="data:{};base64,{}" class="scaled-to-fit" />'''.format(message.content_type, message.content.decode('base64').rstrip()) content = '''<img src="data:{};base64,{}" class="scaled-to-fit" />'''.format(message.content_type, message.content.decode('base64').rstrip())
elif message.content_type.startswith('text/'): elif message.content_type.startswith('text/'):
content = message.content if MessageManager().check_encryption(message.content_type, message.content) == 'OpenPGP':
content = HtmlProcessor.autolink(content if message.content_type == 'text/html' else QTextDocument(content).toHtml()) content = f'<img src={session.chat_widget.encrypted_icon.filename} class="inline-message-icon">Encrypted Message'
content = HtmlProcessor.autolink(content)
encrypted = True
self.pending_decryption.append((message))
stream = blink_session.fake_streams.get('messages')
if stream and stream.can_decrypt:
stream.decrypt(message)
else:
content = message.content
content = HtmlProcessor.autolink(content if message.content_type == 'text/html' else QTextDocument(content).toHtml())
else: else:
return return
...@@ -2367,12 +2379,16 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2367,12 +2379,16 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
else: else:
sender = ChatSender(message.sender.display_name or session.name, uri, session.icon.filename) sender = ChatSender(message.sender.display_name or session.name, uri, session.icon.filename)
if session.chat_widget.history_loaded: if session.chat_widget.history_loaded:
session.chat_widget.add_message(ChatMessage(content, sender, 'incoming', id=message.id)) if message in self.pending_decryption:
self.pending_decryption.remove(message)
session.chat_widget.update_message_text(message.id, content)
else:
session.chat_widget.add_message(ChatMessage(content, sender, 'incoming', id=message.id))
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, 'incoming', id=message.id)) self.render_after_load.append(ChatMessage(content, sender, 'incoming', id=message.id))
if message.disposition is not None and 'display' in message.disposition: if 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():
MessageManager().send_imdn_message(blink_session, message.id, message.timestamp, 'displayed') MessageManager().send_imdn_message(blink_session, message.id, message.timestamp, 'displayed')
else: else:
...@@ -2416,6 +2432,33 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2416,6 +2432,33 @@ 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_PGPMessageDidDecrypt(self, notification):
blink_session = notification.sender
session = blink_session.items.chat
message = notification.data.message
if session is None:
return
if isinstance(message, BlinkMessage):
return
if message in self.pending_decryption:
self.pending_decryption.remove(message)
content = message.content
content = HtmlProcessor.autolink(content if message.content_type == 'text/html' else QTextDocument(content).toHtml())
session.chat_widget.update_message_text(message.message_id, content)
notification_center = NotificationCenter()
blink_message = BlinkMessage(content, message.content_type, id=message.message_id, is_secure=True)
notification_center.post_notification('BlinkMessageDidDecrypt', sender=blink_session, data=NotificationData(message=blink_message))
if message.state != 'displayed' and 'display' in message.disposition:
if message.state != 'delivered' and 'positive-delivery' in message.disposition:
MessageManager().send_imdn_message(blink_session, message.message_id, message.timestamp, 'delivered')
if self.selected_session.blink_session is blink_session and not self.isMinimized() and self.isActiveWindow():
MessageManager().send_imdn_message(blink_session, message.message_id, message.timestamp, 'displayed')
session.chat_widget.update_message_encryption(message.message_id, True)
def _NH_PGPMessageDidNotDecrypt(self, notification): def _NH_PGPMessageDidNotDecrypt(self, notification):
blink_session = notification.sender blink_session = notification.sender
session = blink_session.items.chat session = blink_session.items.chat
...@@ -2423,6 +2466,17 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2423,6 +2466,17 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
return return
session.chat_widget.add_message(ChatStatus(f'Decryption failed: {notification.data.data.error}')) session.chat_widget.add_message(ChatStatus(f'Decryption failed: {notification.data.data.error}'))
def _NH_MessageStreamPGPKeysDidLoad(self, notification):
stream = notification.sender
blink_session = stream.blink_session
stream = blink_session.fake_streams.get('messages')
for message in self.pending_decryption:
if isinstance(message, BlinkMessage):
continue
if stream and stream.can_decrypt:
stream.decrypt(message)
def _NH_BlinkMessageHistoryLoadDidSucceed(self, notification): def _NH_BlinkMessageHistoryLoadDidSucceed(self, notification):
blink_session = notification.sender blink_session = notification.sender
session = blink_session.items.chat session = blink_session.items.chat
...@@ -2433,11 +2487,21 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2433,11 +2487,21 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
return return
for message in messages: for message in messages:
encrypted = False
if message.content_type.startswith('image/'): if message.content_type.startswith('image/'):
content = '''<img src="data:{};base64,{}" class="scaled-to-fit" />'''.format(message.content_type, message.content.decode('base64').rstrip()) content = '''<img src="data:{};base64,{}" class="scaled-to-fit" />'''.format(message.content_type, message.content.decode('base64').rstrip())
elif message.content_type.startswith('text/'): elif message.content_type.startswith('text/'):
content = message.content if MessageManager().check_encryption(message.content_type, message.content) == 'OpenPGP':
content = HtmlProcessor.autolink(content if message.content_type == 'text/html' else QTextDocument(content).toHtml()) content = f'<img src={session.chat_widget.encrypted_icon.filename} class="inline-message-icon">Encrypted Message'
content = HtmlProcessor.autolink(content)
encrypted = True
self.pending_decryption.append((message))
stream = blink_session.fake_streams.get('messages')
if stream and stream.can_decrypt:
stream.decrypt(message)
else:
content = message.content
content = HtmlProcessor.autolink(content if message.content_type == 'text/html' else QTextDocument(content).toHtml())
else: else:
return return
# message.sender = SIPURI.parse(f'sip:{message.remote_uri}') # message.sender = SIPURI.parse(f'sip:{message.remote_uri}')
...@@ -2464,7 +2528,7 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin): ...@@ -2464,7 +2528,7 @@ class ChatWindow(base_class, ui_class, ColorHelperMixin):
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: elif message.state != 'displayed' and 'display' in message.disposition and not encrypted:
if message.state != 'delivered' and 'positive-delivery' in message.disposition: if message.state != 'delivered' and 'positive-delivery' in message.disposition:
MessageManager().send_imdn_message(blink_session, message.message_id, message.timestamp, 'delivered') MessageManager().send_imdn_message(blink_session, message.message_id, message.timestamp, 'delivered')
......
...@@ -490,6 +490,7 @@ class MessageManager(object, metaclass=Singleton): ...@@ -490,6 +490,7 @@ class MessageManager(object, metaclass=Singleton):
def __init__(self): def __init__(self):
self.sessions = [] self.sessions = []
self._outgoing_message_queue = deque() self._outgoing_message_queue = deque()
self._incoming_encrypted_message_queue = deque()
self.pgp_requests = RequestList() self.pgp_requests = RequestList()
notification_center = NotificationCenter() notification_center = NotificationCenter()
...@@ -630,6 +631,19 @@ class MessageManager(object, metaclass=Singleton): ...@@ -630,6 +631,19 @@ class MessageManager(object, metaclass=Singleton):
request.account.sms.public_key = f'{filename}.pubkey' request.account.sms.public_key = f'{filename}.pubkey'
request.account.save() request.account.save()
while self._incoming_encrypted_message_queue:
message, account, contact = self._incoming_encrypted_message_queue.popleft()
try:
blink_session = next(session for session in self.sessions if session.contact.settings is contact.settings)
except StopIteration:
pass
else:
stream = blink_session.fake_streams.get('messages')
if not stream.can_encrypt:
stream.enable_pgp()
stream.decrypt(message)
def _SH_ExportPGPKeys(self, request, message): def _SH_ExportPGPKeys(self, request, message):
account = request.account account = request.account
from blink.contacts import URIUtils from blink.contacts import URIUtils
...@@ -776,6 +790,11 @@ class MessageManager(object, metaclass=Singleton): ...@@ -776,6 +790,11 @@ class MessageManager(object, metaclass=Singleton):
if encryption == 'OpenPGP': if encryption == 'OpenPGP':
if blink_session.fake_streams.get('messages').can_decrypt: if blink_session.fake_streams.get('messages').can_decrypt:
blink_session.fake_streams.get('messages').decrypt(message) blink_session.fake_streams.get('messages').decrypt(message)
else:
self._incoming_encrypted_message_queue.append((message, account, contact))
notification_center.post_notification('BlinkMessageIsParsed', sender=blink_session, data=message)
self._add_contact_to_messages_group(blink_session)
notification_center.post_notification('BlinkGotMessage', sender=blink_session, data=message)
return return
self._handle_incoming_message(message, blink_session) self._handle_incoming_message(message, blink_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