1
0
mirror of https://github.com/profanity-im/profanity.git synced 2024-12-04 14:46:46 -05:00

Add MUC message decryption

This commit is contained in:
Paul Fariello 2019-03-14 22:05:47 +02:20
parent 33b0b46420
commit a3897abba1
5 changed files with 60 additions and 22 deletions

View File

@ -20,6 +20,7 @@
#include "ui/ui.h" #include "ui/ui.h"
#include "ui/window_list.h" #include "ui/window_list.h"
#include "xmpp/connection.h" #include "xmpp/connection.h"
#include "xmpp/muc.h"
#include "xmpp/omemo.h" #include "xmpp/omemo.h"
#include "xmpp/xmpp.h" #include "xmpp/xmpp.h"
@ -387,7 +388,7 @@ omemo_set_device_list(const char *const from, GList * device_list)
} }
} }
free(jid); jid_destroy(jid);
} }
GKeyFile * GKeyFile *
@ -515,18 +516,20 @@ omemo_start_device_session(const char *const jid, uint32_t device_id,
out: out:
g_list_free_full(prekeys, (GDestroyNotify)free_omemo_key); g_list_free_full(prekeys, (GDestroyNotify)free_omemo_key);
free(ownjid); jid_destroy(ownjid);
} }
gboolean gboolean
omemo_on_message_send(ProfChatWin *chatwin, const char *const message, gboolean request_receipt) omemo_on_message_send(ProfChatWin *chatwin, const char *const message, gboolean request_receipt)
{ {
int res; int res;
gboolean ret = FALSE;
Jid *jid = jid_create(connection_get_fulljid()); Jid *jid = jid_create(connection_get_fulljid());
GList *keys = NULL;
GList *recipient_device_id = g_hash_table_lookup(omemo_ctx.device_list, chatwin->barejid); GList *recipient_device_id = g_hash_table_lookup(omemo_ctx.device_list, chatwin->barejid);
if (!recipient_device_id) { if (!recipient_device_id) {
return FALSE; goto out;
} }
GList *sender_device_id = g_hash_table_lookup(omemo_ctx.device_list, jid->barejid); GList *sender_device_id = g_hash_table_lookup(omemo_ctx.device_list, jid->barejid);
@ -552,13 +555,12 @@ omemo_on_message_send(ProfChatWin *chatwin, const char *const message, gboolean
res = aes128gcm_encrypt(ciphertext, &ciphertext_len, tag, &tag_len, (const unsigned char * const)message, strlen(message), iv, key); res = aes128gcm_encrypt(ciphertext, &ciphertext_len, tag, &tag_len, (const unsigned char * const)message, strlen(message), iv, key);
if (res != 0) { if (res != 0) {
log_error("OMEMO: cannot encrypt message"); log_error("OMEMO: cannot encrypt message");
return FALSE; goto out;
} }
memcpy(key_tag, key, AES128_GCM_KEY_LENGTH); memcpy(key_tag, key, AES128_GCM_KEY_LENGTH);
memcpy(key_tag + AES128_GCM_KEY_LENGTH, tag, AES128_GCM_TAG_LENGTH); memcpy(key_tag + AES128_GCM_KEY_LENGTH, tag, AES128_GCM_TAG_LENGTH);
GList *keys = NULL;
GList *device_ids_iter; GList *device_ids_iter;
for (device_ids_iter = recipient_device_id; device_ids_iter != NULL; device_ids_iter = device_ids_iter->next) { for (device_ids_iter = recipient_device_id; device_ids_iter != NULL; device_ids_iter = device_ids_iter->next) {
int res; int res;
@ -623,7 +625,10 @@ omemo_on_message_send(ProfChatWin *chatwin, const char *const message, gboolean
char *id = message_send_chat_omemo(chatwin->barejid, omemo_ctx.device_id, keys, iv, AES128_GCM_IV_LENGTH, ciphertext, ciphertext_len, request_receipt); char *id = message_send_chat_omemo(chatwin->barejid, omemo_ctx.device_id, keys, iv, AES128_GCM_IV_LENGTH, ciphertext, ciphertext_len, request_receipt);
chat_log_omemo_msg_out(chatwin->barejid, message); chat_log_omemo_msg_out(chatwin->barejid, message);
chatwin_outgoing_msg(chatwin, message, id, PROF_MSG_OMEMO, request_receipt); chatwin_outgoing_msg(chatwin, message, id, PROF_MSG_OMEMO, request_receipt);
ret = TRUE;
out:
jid_destroy(jid);
free(id); free(id);
g_list_free_full(keys, free); g_list_free_full(keys, free);
free(ciphertext); free(ciphertext);
@ -632,14 +637,22 @@ omemo_on_message_send(ProfChatWin *chatwin, const char *const message, gboolean
gcry_free(tag); gcry_free(tag);
gcry_free(key_tag); gcry_free(key_tag);
return TRUE; return ret;
} }
char * char *
omemo_on_message_recv(const char *const from, uint32_t sid, omemo_on_message_recv(const char *const from_jid, uint32_t sid,
const unsigned char *const iv, size_t iv_len, GList *keys, const unsigned char *const iv, size_t iv_len, GList *keys,
const unsigned char *const payload, size_t payload_len) const unsigned char *const payload, size_t payload_len, gboolean muc)
{ {
unsigned char *plaintext = NULL;
Jid *sender;
Jid *from = jid_create(from_jid);
if (!from) {
log_error("Invalid jid %s", from_jid);
goto out;
}
int res; int res;
GList *key_iter; GList *key_iter;
omemo_key_t *key = NULL; omemo_key_t *key = NULL;
@ -652,21 +665,35 @@ omemo_on_message_recv(const char *const from, uint32_t sid,
if (!key) { if (!key) {
log_warning("OMEMO: Received a message with no corresponding key"); log_warning("OMEMO: Received a message with no corresponding key");
return NULL; goto out;
}
if (muc) {
GList *roster = muc_roster(from->barejid);
GList *iter;
for (iter = roster; iter != NULL; iter = iter->next) {
Occupant *occupant = (Occupant *)iter->data;
if (g_strcmp0(occupant->nick, from->resourcepart) == 0) {
sender = jid_create(occupant->jid);
break;
}
}
} else {
sender = jid_create(from->barejid);
} }
session_cipher *cipher; session_cipher *cipher;
signal_buffer *plaintext_key; signal_buffer *plaintext_key;
signal_protocol_address address = { signal_protocol_address address = {
.name = from, .name = sender->barejid,
.name_len = strlen(from), .name_len = strlen(sender->barejid),
.device_id = sid .device_id = sid
}; };
res = session_cipher_create(&cipher, omemo_ctx.store, &address, omemo_ctx.signal); res = session_cipher_create(&cipher, omemo_ctx.store, &address, omemo_ctx.signal);
if (res != 0) { if (res != 0) {
log_error("OMEMO: cannot create session cipher"); log_error("OMEMO: cannot create session cipher");
return NULL; goto out;
} }
if (key->prekey) { if (key->prekey) {
@ -687,7 +714,7 @@ omemo_on_message_recv(const char *const from, uint32_t sid,
if (res == 0) { if (res == 0) {
/* Start a new session */ /* Start a new session */
omemo_bundle_request(from, sid, omemo_start_device_session_handle_bundle, free, strdup(from)); omemo_bundle_request(sender->barejid, sid, omemo_start_device_session_handle_bundle, free, strdup(sender->barejid));
} }
} else { } else {
log_debug("OMEMO: decrypting message with existing session"); log_debug("OMEMO: decrypting message with existing session");
@ -707,7 +734,7 @@ omemo_on_message_recv(const char *const from, uint32_t sid,
} }
size_t plaintext_len = payload_len; size_t plaintext_len = payload_len;
unsigned char *plaintext = malloc(plaintext_len + 1); plaintext = malloc(plaintext_len + 1);
res = aes128gcm_decrypt(plaintext, &plaintext_len, payload, payload_len, iv, res = aes128gcm_decrypt(plaintext, &plaintext_len, payload, payload_len, iv,
signal_buffer_data(plaintext_key), signal_buffer_data(plaintext_key),
signal_buffer_data(plaintext_key) + AES128_GCM_KEY_LENGTH); signal_buffer_data(plaintext_key) + AES128_GCM_KEY_LENGTH);
@ -720,6 +747,9 @@ omemo_on_message_recv(const char *const from, uint32_t sid,
plaintext[plaintext_len] = '\0'; plaintext[plaintext_len] = '\0';
out:
jid_destroy(from);
jid_destroy(sender);
return (char *)plaintext; return (char *)plaintext;
} }

View File

@ -40,4 +40,4 @@ void omemo_start_device_session(const char *const jid, uint32_t device_id, GList
gboolean omemo_loaded(void); gboolean omemo_loaded(void);
gboolean omemo_on_message_send(ProfChatWin *chatwin, const char *const message, gboolean request_receipt); gboolean omemo_on_message_send(ProfChatWin *chatwin, const char *const message, gboolean request_receipt);
char * omemo_on_message_recv(const char *const from, uint32_t sid, const unsigned char *const iv, size_t iv_len, GList *keys, const unsigned char *const payload, size_t payload_len); char * omemo_on_message_recv(const char *const from, uint32_t sid, const unsigned char *const iv, size_t iv_len, GList *keys, const unsigned char *const payload, size_t payload_len, gboolean muc);

View File

@ -168,6 +168,7 @@ typedef struct prof_muc_win_t {
gboolean unread_mentions; gboolean unread_mentions;
gboolean unread_triggers; gboolean unread_triggers;
gboolean showjid; gboolean showjid;
gboolean is_omemo;
unsigned long memcheck; unsigned long memcheck;
char *enctext; char *enctext;
char *message_char; char *message_char;

View File

@ -723,10 +723,16 @@ _handle_groupchat(xmpp_stanza_t *const stanza)
return; return;
} }
message = xmpp_message_get_body(stanza); #ifdef HAVE_OMEMO
message = omemo_receive_message(stanza);
#endif
if (!message) { if (!message) {
jid_destroy(jid); message = xmpp_message_get_body(stanza);
return; if (!message) {
jid_destroy(jid);
return;
}
} }
// determine if the notifications happened whilst offline // determine if the notifications happened whilst offline

View File

@ -197,6 +197,8 @@ omemo_start_device_session_handle_bundle(xmpp_stanza_t *const stanza, void *cons
char * char *
omemo_receive_message(xmpp_stanza_t *const stanza) omemo_receive_message(xmpp_stanza_t *const stanza)
{ {
const char *type = xmpp_stanza_get_type(stanza);
xmpp_stanza_t *encrypted = xmpp_stanza_get_child_by_ns(stanza, STANZA_NS_OMEMO); xmpp_stanza_t *encrypted = xmpp_stanza_get_child_by_ns(stanza, STANZA_NS_OMEMO);
if (!encrypted) { if (!encrypted) {
return NULL; return NULL;
@ -264,11 +266,10 @@ skip:
} }
const char *from = xmpp_stanza_get_from(stanza); const char *from = xmpp_stanza_get_from(stanza);
Jid *jid = jid_create(from);
char *plaintext = omemo_on_message_recv(jid->barejid, sid, iv_raw, iv_len, keys, payload_raw, payload_len); char *plaintext = omemo_on_message_recv(from, sid, iv_raw, iv_len,
keys, payload_raw, payload_len,
jid_destroy(jid); g_strcmp0(type, STANZA_TYPE_GROUPCHAT) == 0);
return plaintext; return plaintext;
} }