/* This file is part of telegram-client. Telegram-client is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. Telegram-client is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this telegram-client. If not, see . Copyright Vitaly Valtman 2013 */ #include "config.h" #include #include #include "structures.h" #include "mtproto-common.h" #include "telegram.h" #include "tree.h" #include "loop.h" #include #include #include #include "queries.h" #include "binlog.h" #define sha1 SHA1 int verbosity; peer_t *Peers[MAX_USER_NUM]; int peer_num; int encr_chats_allocated; int geo_chats_allocated; extern int binlog_enabled; void fetch_skip_photo (void); #define code_assert(x) if (!(x)) { logprintf ("Can not parse at line %d\n", __LINE__); assert (0); return -1; } int fetch_file_location (struct file_location *loc) { int x = fetch_int (); code_assert (x == CODE_file_location_unavailable || x == CODE_file_location); if (x == CODE_file_location_unavailable) { loc->dc = -1; loc->volume = fetch_long (); loc->local_id = fetch_int (); loc->secret = fetch_long (); } else { loc->dc = fetch_int (); loc->volume = fetch_long (); loc->local_id = fetch_int (); loc->secret = fetch_long (); } return 0; } int fetch_skip_file_location (void) { int x = fetch_int (); code_assert (x == CODE_file_location_unavailable || x == CODE_file_location); if (x == CODE_file_location_unavailable) { in_ptr += 5; } else { in_ptr += 6; } return 0; } int fetch_user_status (struct user_status *S) { unsigned x = fetch_int (); code_assert (x == CODE_user_status_empty || x == CODE_user_status_online || x == CODE_user_status_offline); switch (x) { case CODE_user_status_empty: S->online = 0; S->when = 0; break; case CODE_user_status_online: S->online = 1; S->when = fetch_int (); break; case CODE_user_status_offline: S->online = -1; S->when = fetch_int (); break; default: assert (0); } return 0; } int our_id; char *create_print_name (peer_id_t id, const char *a1, const char *a2, const char *a3, const char *a4) { const char *d[4]; d[0] = a1; d[1] = a2; d[2] = a3; d[3] = a4; static char buf[10000]; int i; int p = 0; for (i = 0; i < 4; i++) { if (d[i] && strlen (d[i])) { p += snprintf (buf + p, 9999 - p, "%s%s", p ? "_" : "", d[i]); assert (p < 9990); } } char *s = buf; while (*s) { if (*s == ' ') { *s = '_'; } s++; } s = buf; int cc = 0; while (1) { int ok = 1; int i; for (i = 0; i < peer_num; i++) { if (cmp_peer_id (Peers[i]->id, id) && Peers[i]->print_name && !strcmp (Peers[i]->print_name, s)) { ok = 0; break; } } if (ok) { break; } cc ++; assert (cc <= 99); if (cc == 1) { int l = strlen (s); s[l + 2] = 0; s[l] = '#'; s[l + 1] = '1'; } else if (cc == 10) { int l = strlen (s); s[l + 1] = 0; s[l] = '0'; s[l - 1] = '1'; } else { int l = strlen (s); s[l - 1] ++; int cc = l - 1; while (s[cc] > '9') { s[cc] = '0'; s[cc - 1] ++; cc --; } } } return strdup (s); } long long fetch_user_photo (struct user *U) { unsigned x = fetch_int (); code_assert (x == CODE_user_profile_photo || x == CODE_user_profile_photo_old || x == CODE_user_profile_photo_empty); if (x == CODE_user_profile_photo_empty) { bl_do_set_user_profile_photo (U, 0, 0, 0); return 0; } long long photo_id = 1; if (x == CODE_user_profile_photo) { photo_id = fetch_long (); } struct file_location big; struct file_location small; if (fetch_file_location (&small) < 0) { return -1; } if (fetch_file_location (&big) < 0) { return -1; } bl_do_set_user_profile_photo (U, photo_id, &big, &small); return 0; } int fetch_user (struct user *U) { unsigned x = fetch_int (); code_assert (x == CODE_user_empty || x == CODE_user_self || x == CODE_user_contact || x == CODE_user_request || x == CODE_user_foreign || x == CODE_user_deleted); U->id = MK_USER (fetch_int ()); if ((U->flags & FLAG_CREATED) && x == CODE_user_empty) { return 0; } if (x == CODE_user_empty) { return 0; } if (x == CODE_user_self) { assert (!our_id || (our_id == get_peer_id (U->id))); if (!our_id) { bl_do_set_our_id (get_peer_id (U->id)); write_auth_file (); } } int new = 0; if (!(U->flags & FLAG_CREATED)) { new = 1; } if (new) { int l1 = prefetch_strlen (); char *s1 = fetch_str (l1); int l2 = prefetch_strlen (); char *s2 = fetch_str (l2); if (x == CODE_user_deleted && !(U->flags & FLAG_DELETED)) { bl_do_new_user (get_peer_id (U->id), s1, l1, s2, l2, 0, 0, 0, 0); bl_do_user_delete (U); } if (x != CODE_user_deleted) { long long access_token = 0; if (x != CODE_user_self) { access_token = fetch_long (); } int phone_len = 0; char *phone = 0; if (x != CODE_user_foreign) { phone_len = prefetch_strlen (); phone = fetch_str (phone_len); } bl_do_new_user (get_peer_id (U->id), s1, l1, s2, l2, access_token, phone, phone_len, x == CODE_user_contact); if (fetch_user_photo (U) < 0) { return -1; } if (fetch_user_status (&U->status) < 0) { return -1; } if (x == CODE_user_self) { fetch_bool (); } } } else { int l1 = prefetch_strlen (); char *s1 = fetch_str (l1); int l2 = prefetch_strlen (); char *s2 = fetch_str (l2); bl_do_set_user_name (U, s1, l1, s2, l2); if (x == CODE_user_deleted && !(U->flags & FLAG_DELETED)) { bl_do_user_delete (U); } if (x != CODE_user_deleted) { if (x != CODE_user_self) { bl_do_set_user_access_token (U, fetch_long ()); } if (x != CODE_user_foreign) { int l = prefetch_strlen (); char *s = fetch_str (l); bl_do_set_user_phone (U, s, l); } if (fetch_user_photo (U) < 0) { return -1; } fetch_user_status (&U->status); if (x == CODE_user_self) { fetch_bool (); } if (x == CODE_user_contact) { bl_do_set_user_friend (U, 1); } else { bl_do_set_user_friend (U, 0); } } } return 0; } void fetch_encrypted_chat (struct secret_chat *U) { unsigned x = fetch_int (); assert (x == CODE_encrypted_chat_empty || x == CODE_encrypted_chat_waiting || x == CODE_encrypted_chat_requested || x == CODE_encrypted_chat || x == CODE_encrypted_chat_discarded); U->id = MK_ENCR_CHAT (fetch_int ()); if (x == CODE_encrypted_chat_empty) { return; } int new = !(U->flags & FLAG_CREATED); if (x == CODE_encrypted_chat_discarded) { if (new) { logprintf ("Unknown chat in deleted state. May be we forgot something...\n"); return; } bl_do_encr_chat_delete (U); write_secret_chat_file (); return; } static char g_key[256]; static char nonce[256]; if (new) { long long access_hash = fetch_long (); int date = fetch_int (); int admin_id = fetch_int (); int user_id = fetch_int () + admin_id - our_id; if (x == CODE_encrypted_chat_waiting) { logprintf ("Unknown chat in waiting state. May be we forgot something...\n"); return; } if (x == CODE_encrypted_chat_requested || x == CODE_encrypted_chat) { memset (g_key, 0, sizeof (g_key)); memset (nonce, 0, sizeof (nonce)); } int l = prefetch_strlen (); char *s = fetch_str (l); if (l < 256) { memcpy (g_key + 256 - l, s, l); } else { memcpy (g_key, s + (l - 256), 256); } l = prefetch_strlen (); s = fetch_str (l); if (l < 256) { memcpy (nonce + 256 - l, s, l); } else { memcpy (nonce, s + (l - 256), 256); } if (x == CODE_encrypted_chat) { fetch_long (); // fingerprint } if (x == CODE_encrypted_chat) { logprintf ("Unknown chat in ok state. May be we forgot something...\n"); return; } bl_do_encr_chat_requested (U, access_hash, date, admin_id, user_id, (void *)g_key, (void *)nonce); write_secret_chat_file (); } else { bl_do_set_encr_chat_access_hash (U, fetch_long ()); bl_do_set_encr_chat_date (U, fetch_int ()); if (fetch_int () != U->admin_id) { logprintf ("Changed admin in secret chat. WTF?\n"); return; } if (U->user_id != U->admin_id + fetch_int () - our_id) { logprintf ("Changed partner in secret chat. WTF?\n"); return; } if (x == CODE_encrypted_chat_waiting) { bl_do_set_encr_chat_state (U, sc_waiting); write_secret_chat_file (); return; // We needed only access hash from here } if (x == CODE_encrypted_chat_requested || x == CODE_encrypted_chat) { memset (g_key, 0, sizeof (g_key)); memset (nonce, 0, sizeof (nonce)); } int l = prefetch_strlen (); char *s = fetch_str (l); if (l < 256) { memcpy (g_key + 256 - l, s, l); } else { memcpy (g_key, s + (l - 256), 256); } l = prefetch_strlen (); s = fetch_str (l); if (l < 256) { memcpy (nonce + 256 - l, s, l); } else { memcpy (nonce, s + (l - 256), 256); } if (x == CODE_encrypted_chat_requested) { return; // Duplicate? } bl_do_encr_chat_accepted (U, (void *)g_key, (void *)nonce, fetch_long ()); } write_secret_chat_file (); } void fetch_notify_settings (void); void fetch_user_full (struct user *U) { assert (fetch_int () == CODE_user_full); fetch_alloc_user (); unsigned x; assert (fetch_int () == (int)CODE_contacts_link); x = fetch_int (); assert (x == CODE_contacts_my_link_empty || x == CODE_contacts_my_link_requested || x == CODE_contacts_my_link_contact); if (x == CODE_contacts_my_link_requested) { fetch_bool (); } x = fetch_int (); assert (x == CODE_contacts_foreign_link_unknown || x == CODE_contacts_foreign_link_requested || x == CODE_contacts_foreign_link_mutual); if (x == CODE_contacts_foreign_link_requested) { fetch_bool (); } fetch_alloc_user (); int *start = in_ptr; fetch_skip_photo (); bl_do_set_user_full_photo (U, start, 4 * (in_ptr - start)); fetch_notify_settings (); bl_do_set_user_blocked (U, fetch_bool ()); int l1 = prefetch_strlen (); char *s1 = fetch_str (l1); int l2 = prefetch_strlen (); char *s2 = fetch_str (l2); bl_do_set_user_real_name (U, s1, l1, s2, l2); } void fetch_chat (struct chat *C) { unsigned x = fetch_int (); assert (x == CODE_chat_empty || x == CODE_chat || x == CODE_chat_forbidden); C->id = MK_CHAT (fetch_int ()); if (x == CODE_chat_empty) { return; } int new = !(C->flags & FLAG_CREATED); if (new) { int y = 0; if (x == CODE_chat_forbidden) { y |= FLAG_FORBIDDEN; } int l = prefetch_strlen (); char *s = fetch_str (l); struct file_location small; struct file_location big; memset (&small, 0, sizeof (small)); memset (&big, 0, sizeof (big)); int users_num = -1; int date = 0; int version = -1; if (x == CODE_chat) { unsigned y = fetch_int (); if (y == CODE_chat_photo_empty) { small.dc = -2; big.dc = -2; } else { assert (y == CODE_chat_photo); fetch_file_location (&small); fetch_file_location (&big); } users_num = fetch_int (); date = fetch_int (); if (fetch_bool ()) { y |= FLAG_CHAT_IN_CHAT; } version = fetch_int (); } else { small.dc = -2; big.dc = -2; users_num = -1; date = fetch_int (); version = -1; } bl_do_create_chat (C, y, s, l, users_num, date, version, &big, &small); } else { if (x == CODE_chat_forbidden) { bl_do_chat_forbid (C, 1); } else { bl_do_chat_forbid (C, 0); } int l = prefetch_strlen (); char *s = fetch_str (l); bl_do_set_chat_title (C, s, l); struct file_location small; struct file_location big; memset (&small, 0, sizeof (small)); memset (&big, 0, sizeof (big)); if (x == CODE_chat) { unsigned y = fetch_int (); if (y == CODE_chat_photo_empty) { small.dc = -2; big.dc = -2; } else { assert (y == CODE_chat_photo); fetch_file_location (&small); fetch_file_location (&big); } bl_do_set_chat_photo (C, &big, &small); int users_num = fetch_int (); bl_do_set_chat_date (C, fetch_int ()); bl_do_set_chat_set_in_chat (C, fetch_bool ()); bl_do_set_chat_version (C, users_num, fetch_int ()); } else { bl_do_set_chat_date (C, fetch_int ()); } } } void fetch_notify_settings (void) { unsigned x = fetch_int (); assert (x == CODE_peer_notify_settings || x == CODE_peer_notify_settings_empty || x == CODE_peer_notify_settings_old); if (x == CODE_peer_notify_settings_old) { fetch_int (); // mute_until int l = prefetch_strlen (); fetch_str (l); fetch_bool (); // show_previews fetch_int (); // peer notify events } if (x == CODE_peer_notify_settings) { fetch_int (); // mute_until int l = prefetch_strlen (); fetch_str (l); fetch_bool (); // show_previews fetch_int (); // events_mask } } void fetch_chat_full (struct chat *C) { unsigned x = fetch_int (); assert (x == CODE_messages_chat_full); assert (fetch_int () == CODE_chat_full); C->id = MK_CHAT (fetch_int ()); C->flags &= ~(FLAG_DELETED | FLAG_FORBIDDEN | FLAG_CHAT_IN_CHAT); C->flags |= FLAG_CREATED; x = fetch_int (); if (x == CODE_chat_participants) { assert (fetch_int () == get_peer_id (C->id)); C->admin_id = fetch_int (); assert (fetch_int () == CODE_vector); if (C->users) { free (C->users); } C->users_num = fetch_int (); C->users = malloc (sizeof (struct chat_user) * C->users_num); int i; for (i = 0; i < C->users_num; i++) { assert (fetch_int () == (int)CODE_chat_participant); C->users[i].user_id = fetch_int (); C->users[i].inviter_id = fetch_int (); C->users[i].date = fetch_int (); } C->version = fetch_int (); } else { C->flags |= FLAG_FORBIDDEN; assert (x == CODE_chat_participants_forbidden); } if (C->flags & FLAG_HAS_PHOTO) { free_photo (&C->photo); } fetch_photo (&C->photo); C->flags |= FLAG_HAS_PHOTO; fetch_notify_settings (); int n, i; assert (fetch_int () == CODE_vector); n = fetch_int (); for (i = 0; i < n; i++) { fetch_alloc_chat (); } assert (fetch_int () == CODE_vector); n = fetch_int (); for (i = 0; i < n; i++) { fetch_alloc_user (); } } void fetch_photo_size (struct photo_size *S) { memset (S, 0, sizeof (*S)); unsigned x = fetch_int (); assert (x == CODE_photo_size || x == CODE_photo_cached_size || x == CODE_photo_size_empty); S->type = fetch_str_dup (); if (x != CODE_photo_size_empty) { fetch_file_location (&S->loc); S->w = fetch_int (); S->h = fetch_int (); if (x == CODE_photo_size) { S->size = fetch_int (); } else { S->size = prefetch_strlen (); // S->data = malloc (S->size); // assert (S->data); fetch_str (S->size); // memcpy (S->data, fetch_str (S->size), S->size); } } } void fetch_skip_photo_size (void) { unsigned x = fetch_int (); assert (x == CODE_photo_size || x == CODE_photo_cached_size || x == CODE_photo_size_empty); int l = prefetch_strlen (); fetch_str (l); // type if (x != CODE_photo_size_empty) { fetch_skip_file_location (); in_ptr += 2; // w, h if (x == CODE_photo_size) { in_ptr ++; } else { l = prefetch_strlen (); fetch_str (l); } } } void fetch_geo (struct geo *G) { unsigned x = fetch_int (); if (x == CODE_geo_point) { G->longitude = fetch_double (); G->latitude = fetch_double (); } else { assert (x == CODE_geo_point_empty); G->longitude = 0; G->latitude = 0; } } void fetch_skip_geo (void) { unsigned x = fetch_int (); assert (x == CODE_geo_point || x == CODE_geo_point_empty); if (x == CODE_geo_point) { in_ptr += 2; } } void fetch_photo (struct photo *P) { memset (P, 0, sizeof (*P)); unsigned x = fetch_int (); assert (x == CODE_photo_empty || x == CODE_photo); P->id = fetch_long (); if (x == CODE_photo_empty) { return; } P->access_hash = fetch_long (); P->user_id = fetch_int (); P->date = fetch_int (); P->caption = fetch_str_dup (); fetch_geo (&P->geo); assert (fetch_int () == CODE_vector); P->sizes_num = fetch_int (); P->sizes = malloc (sizeof (struct photo_size) * P->sizes_num); int i; for (i = 0; i < P->sizes_num; i++) { fetch_photo_size (&P->sizes[i]); } } void fetch_skip_photo (void) { unsigned x = fetch_int (); assert (x == CODE_photo_empty || x == CODE_photo); in_ptr += 2; // id if (x == CODE_photo_empty) { return; } in_ptr += 2 +1 + 1; // access_hash, user_id, date int l = prefetch_strlen (); fetch_str (l); // caption fetch_skip_geo (); assert (fetch_int () == CODE_vector); int n = fetch_int (); int i; for (i = 0; i < n; i++) { fetch_skip_photo_size (); } } void fetch_video (struct video *V) { memset (V, 0, sizeof (*V)); unsigned x = fetch_int (); V->id = fetch_long (); if (x == CODE_video_empty) { return; } V->access_hash = fetch_long (); V->user_id = fetch_int (); V->date = fetch_int (); V->caption = fetch_str_dup (); V->duration = fetch_int (); V->size = fetch_int (); fetch_photo_size (&V->thumb); V->dc_id = fetch_int (); V->w = fetch_int (); V->h = fetch_int (); } void fetch_audio (struct audio *V) { memset (V, 0, sizeof (*V)); unsigned x = fetch_int (); V->id = fetch_long (); if (x == CODE_audio_empty) { return; } V->access_hash = fetch_long (); V->user_id = fetch_int (); V->date = fetch_int (); V->duration = fetch_int (); V->size = fetch_int (); V->dc_id = fetch_int (); } void fetch_document (struct document *V) { memset (V, 0, sizeof (*V)); unsigned x = fetch_int (); V->id = fetch_long (); if (x == CODE_document_empty) { return; } V->access_hash = fetch_long (); V->user_id = fetch_int (); V->date = fetch_int (); V->caption = fetch_str_dup (); V->mime_type = fetch_str_dup (); V->size = fetch_int (); fetch_photo_size (&V->thumb); V->dc_id = fetch_int (); } void fetch_message_action (struct message_action *M) { memset (M, 0, sizeof (*M)); unsigned x = fetch_int (); M->type = x; switch (x) { case CODE_message_action_empty: break; case CODE_message_action_geo_chat_create: { int l = prefetch_strlen (); // title char *s = fetch_str (l); int l2 = prefetch_strlen (); // checkin char *s2 = fetch_str (l2); logprintf ("Message action: Created geochat %.*s in address %.*s\n", l, s, l2, s2); } break; case CODE_message_action_geo_chat_checkin: break; case CODE_message_action_chat_create: M->title = fetch_str_dup (); assert (fetch_int () == (int)CODE_vector); M->user_num = fetch_int (); M->users = malloc (M->user_num * 4); fetch_ints (M->users, M->user_num); break; case CODE_message_action_chat_edit_title: M->new_title = fetch_str_dup (); break; case CODE_message_action_chat_edit_photo: fetch_photo (&M->photo); break; case CODE_message_action_chat_delete_photo: break; case CODE_message_action_chat_add_user: M->user = fetch_int (); break; case CODE_message_action_chat_delete_user: M->user = fetch_int (); break; default: assert (0); } } void fetch_message_short (struct message *M) { memset (M, 0, sizeof (*M)); M->id = fetch_int (); M->to_id = MK_USER (our_id); M->from_id = MK_USER (fetch_int ()); M->message = fetch_str_dup (); fetch_pts (); M->date = fetch_int (); fetch_seq (); M->media.type = CODE_message_media_empty; M->unread = 1; } void fetch_message_short_chat (struct message *M) { memset (M, 0, sizeof (*M)); M->id = fetch_int (); M->from_id = MK_USER (fetch_int ()); M->to_id = MK_CHAT (fetch_int ()); M->message = fetch_str_dup (); fetch_pts (); M->date = fetch_int (); fetch_seq (); M->media.type = CODE_message_media_empty; M->unread = 1; } void fetch_message_media (struct message_media *M) { memset (M, 0, sizeof (*M)); M->type = fetch_int (); switch (M->type) { case CODE_message_media_empty: break; case CODE_message_media_photo: fetch_photo (&M->photo); break; case CODE_message_media_video: fetch_video (&M->video); break; case CODE_message_media_audio: fetch_audio (&M->audio); break; case CODE_message_media_document: fetch_document (&M->document); break; case CODE_message_media_geo: fetch_geo (&M->geo); break; case CODE_message_media_contact: M->phone = fetch_str_dup (); M->first_name = fetch_str_dup (); M->last_name = fetch_str_dup (); M->user_id = fetch_int (); break; case CODE_message_media_unsupported: M->data = fetch_str_dup (); break; default: logprintf ("type = 0x%08x\n", M->type); assert (0); } } void fetch_message_media_encrypted (struct message_media *M) { memset (M, 0, sizeof (*M)); unsigned x = fetch_int (); int l; switch (x) { case CODE_decrypted_message_media_empty: M->type = CODE_message_media_empty; break; case CODE_decrypted_message_media_photo: M->type = x; l = prefetch_strlen (); fetch_str (l); // thumb fetch_int (); // thumb_w fetch_int (); // thumb_h M->encr_photo.w = fetch_int (); M->encr_photo.h = fetch_int (); M->encr_photo.size = fetch_int (); l = prefetch_strlen (); assert (l > 0); M->encr_photo.key = malloc (32); memset (M->encr_photo.key, 0, 32); if (l <= 32) { memcpy (M->encr_photo.key + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_photo.key, fetch_str (l) + (l - 32), 32); } M->encr_photo.iv = malloc (32); l = prefetch_strlen (); assert (l > 0); memset (M->encr_photo.iv, 0, 32); if (l <= 32) { memcpy (M->encr_photo.iv + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_photo.iv, fetch_str (l) + (l - 32), 32); } break; case CODE_decrypted_message_media_video: M->type = x; l = prefetch_strlen (); fetch_str (l); // thumb fetch_int (); // thumb_w fetch_int (); // thumb_h M->encr_video.duration = fetch_int (); M->encr_video.w = fetch_int (); M->encr_video.h = fetch_int (); M->encr_video.size = fetch_int (); l = prefetch_strlen (); assert (l > 0); M->encr_video.key = malloc (32); memset (M->encr_photo.key, 0, 32); if (l <= 32) { memcpy (M->encr_video.key + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_video.key, fetch_str (l) + (l - 32), 32); } M->encr_video.iv = malloc (32); l = prefetch_strlen (); assert (l > 0); memset (M->encr_video.iv, 0, 32); if (l <= 32) { memcpy (M->encr_video.iv + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_video.iv, fetch_str (l) + (l - 32), 32); } break; case CODE_decrypted_message_media_audio: M->type = x; M->encr_audio.duration = fetch_int (); M->encr_audio.size = fetch_int (); l = prefetch_strlen (); assert (l > 0); M->encr_video.key = malloc (32); memset (M->encr_photo.key, 0, 32); if (l <= 32) { memcpy (M->encr_video.key + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_video.key, fetch_str (l) + (l - 32), 32); } M->encr_video.iv = malloc (32); l = prefetch_strlen (); assert (l > 0); memset (M->encr_video.iv, 0, 32); if (l <= 32) { memcpy (M->encr_video.iv + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_video.iv, fetch_str (l) + (l - 32), 32); } break; case CODE_decrypted_message_media_document: M->type = x; l = prefetch_strlen (); fetch_str (l); // thumb fetch_int (); // thumb_w fetch_int (); // thumb_h M->encr_document.file_name = fetch_str_dup (); M->encr_document.mime_type = fetch_str_dup (); M->encr_video.size = fetch_int (); l = prefetch_strlen (); assert (l > 0); M->encr_video.key = malloc (32); memset (M->encr_photo.key, 0, 32); if (l <= 32) { memcpy (M->encr_video.key + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_video.key, fetch_str (l) + (l - 32), 32); } M->encr_video.iv = malloc (32); l = prefetch_strlen (); assert (l > 0); memset (M->encr_video.iv, 0, 32); if (l <= 32) { memcpy (M->encr_video.iv + (32 - l), fetch_str (l), l); } else { memcpy (M->encr_video.iv, fetch_str (l) + (l - 32), 32); } break; /* case CODE_decrypted_message_media_file: M->type = x; M->encr_file.filename = fetch_str_dup (); l = prefetch_strlen (); fetch_str (l); // thumb l = fetch_int (); assert (l > 0); M->encr_file.key = malloc (l); memcpy (M->encr_file.key, fetch_str (l), l); l = fetch_int (); assert (l > 0); M->encr_file.iv = malloc (l); memcpy (M->encr_file.iv, fetch_str (l), l); break; */ case CODE_decrypted_message_media_geo_point: M->geo.longitude = fetch_double (); M->geo.latitude = fetch_double (); M->type = CODE_message_media_geo; break; case CODE_decrypted_message_media_contact: M->type = CODE_message_media_contact; M->phone = fetch_str_dup (); M->first_name = fetch_str_dup (); M->last_name = fetch_str_dup (); M->user_id = fetch_int (); break; default: logprintf ("type = 0x%08x\n", M->type); assert (0); } } peer_id_t fetch_peer_id (void) { unsigned x =fetch_int (); if (x == CODE_peer_user) { return MK_USER (fetch_int ()); } else { assert (CODE_peer_chat); return MK_CHAT (fetch_int ()); } } void fetch_message (struct message *M) { memset (M, 0, sizeof (*M)); unsigned x = fetch_int (); assert (x == CODE_message_empty || x == CODE_message || x == CODE_message_forwarded || x == CODE_message_service); M->id = fetch_int (); if (x == CODE_message_empty) { M->flags |= 1; return; } if (x == CODE_message_forwarded) { M->fwd_from_id = MK_USER (fetch_int ()); M->fwd_date = fetch_int (); } M->from_id = MK_USER (fetch_int ()); M->to_id = fetch_peer_id (); M->out = fetch_bool (); M->unread = fetch_bool (); M->date = fetch_int (); if (x == CODE_message_service) { M->service = 1; fetch_message_action (&M->action); } else { M->message = fetch_str_dup (); fetch_message_media (&M->media); } } void fetch_geo_message (struct message *M) { memset (M, 0, sizeof (*M)); unsigned x = fetch_int (); assert (x == CODE_geo_chat_message_empty || x == CODE_geo_chat_message || x == CODE_geo_chat_message_service); M->to_id = MK_GEO_CHAT (fetch_int ()); M->id = fetch_int (); if (x == CODE_geo_chat_message_empty) { M->flags |= 1; return; } M->from_id = MK_USER (fetch_int ()); M->date = fetch_int (); if (x == CODE_geo_chat_message_service) { M->service = 1; fetch_message_action (&M->action); } else { M->message = fetch_str_dup (); fetch_message_media (&M->media); } } int *decr_ptr; int *decr_end; int decrypt_encrypted_message (struct secret_chat *E) { int *msg_key = decr_ptr; decr_ptr += 4; assert (decr_ptr < decr_end); static unsigned char sha1a_buffer[20]; static unsigned char sha1b_buffer[20]; static unsigned char sha1c_buffer[20]; static unsigned char sha1d_buffer[20]; static unsigned char buf[64]; memcpy (buf, msg_key, 16); memcpy (buf + 16, E->key, 32); sha1 (buf, 48, sha1a_buffer); memcpy (buf, E->key + 8, 16); memcpy (buf + 16, msg_key, 16); memcpy (buf + 32, E->key + 12, 16); sha1 (buf, 48, sha1b_buffer); memcpy (buf, E->key + 16, 32); memcpy (buf + 32, msg_key, 16); sha1 (buf, 48, sha1c_buffer); memcpy (buf, msg_key, 16); memcpy (buf + 16, E->key + 24, 32); sha1 (buf, 48, sha1d_buffer); static unsigned char key[32]; memcpy (key, sha1a_buffer + 0, 8); memcpy (key + 8, sha1b_buffer + 8, 12); memcpy (key + 20, sha1c_buffer + 4, 12); static unsigned char iv[32]; memcpy (iv, sha1a_buffer + 8, 12); memcpy (iv + 12, sha1b_buffer + 0, 8); memcpy (iv + 20, sha1c_buffer + 16, 4); memcpy (iv + 24, sha1d_buffer + 0, 8); AES_KEY aes_key; AES_set_decrypt_key (key, 256, &aes_key); AES_ige_encrypt ((void *)decr_ptr, (void *)decr_ptr, 4 * (decr_end - decr_ptr), &aes_key, iv, 0); int x = *(decr_ptr); if (x < 0 || (x & 3)) { return -1; } assert (x >= 0 && !(x & 3)); sha1 ((void *)decr_ptr, 4 + x, sha1a_buffer); if (memcmp (sha1a_buffer + 4, msg_key, 16)) { logprintf ("Sha1 mismatch\n"); return -1; } return 0; } void fetch_encrypted_message (struct message *M) { memset (M, 0, sizeof (*M)); unsigned x = fetch_int (); assert (x == CODE_encrypted_message || x == CODE_encrypted_message_service); unsigned sx = x; M->id = fetch_long (); peer_id_t chat = MK_ENCR_CHAT (fetch_int ()); M->from_id = MK_USER (our_id); M->to_id = chat; peer_t *P = user_chat_get (chat); M->flags &= ~(FLAG_MESSAGE_EMPTY | FLAG_DELETED); M->flags |= FLAG_ENCRYPTED; if (!P) { logprintf ("Encrypted message to unknown chat. Dropping\n"); M->flags |= FLAG_MESSAGE_EMPTY; } M->date = fetch_int (); int len = prefetch_strlen (); assert ((len & 15) == 8); decr_ptr = (void *)fetch_str (len); decr_end = decr_ptr + (len / 4); M->flags |= FLAG_ENCRYPTED; int ok = 0; if (P) { if (*(long long *)decr_ptr != P->encr_chat.key_fingerprint) { logprintf ("Encrypted message with bad fingerprint to chat %s\n", P->print_name); P = 0; } decr_ptr += 2; } if (P && decrypt_encrypted_message (&P->encr_chat) >= 0) { ok = 1; int *save_in_ptr = in_ptr; int *save_in_end = in_end; in_ptr = decr_ptr; int l = fetch_int (); in_end = in_ptr + l; unsigned x = fetch_int (); if (x == CODE_decrypted_message_layer) { int layer = fetch_int (); assert (layer >= 0); x = fetch_int (); } assert (x == CODE_decrypted_message || x == CODE_decrypted_message_service); assert (M->id = fetch_long ()); l = prefetch_strlen (); fetch_str (l); // random_bytes if (x == CODE_decrypted_message) { M->message = fetch_str_dup (); fetch_message_media_encrypted (&M->media); } else { assert (fetch_int () == (int)CODE_decrypted_message_action_set_message_t_t_l); M->action.type = CODE_decrypted_message_action_set_message_t_t_l; P->encr_chat.ttl = fetch_int (); M->action.ttl = P->encr_chat.ttl; M->service = 1; } in_ptr = save_in_ptr; in_end = save_in_end; } if (sx == CODE_encrypted_message) { if (ok) { fetch_encrypted_message_file (&M->media); } else { x = fetch_int (); if (x == CODE_encrypted_file) { fetch_skip (7); } else { assert (x == CODE_encrypted_file_empty); } M->media.type = CODE_message_media_empty; } } } void fetch_encrypted_message_file (struct message_media *M) { unsigned x = fetch_int (); assert (x == CODE_encrypted_file || x == CODE_encrypted_file_empty); if (x == CODE_encrypted_file_empty) { assert (M->type != CODE_decrypted_message_media_photo && M->type != CODE_decrypted_message_media_video); } else { assert (M->type == CODE_decrypted_message_media_photo || M->type == CODE_decrypted_message_media_video); M->encr_photo.id = fetch_long (); M->encr_photo.access_hash = fetch_long (); fetch_int (); //assert (M->encr_photo.size == fetch_int ()); //M->encr_photo.size = fetch_int (); // Why it is not the same? M->encr_photo.dc_id = fetch_int (); M->encr_photo.key_fingerprint = fetch_int (); } } static int id_cmp (struct message *M1, struct message *M2) { if (M1->id < M2->id) { return -1; } else if (M1->id > M2->id) { return 1; } else { return 0; } } #define peer_cmp(a,b) (cmp_peer_id (a->id, b->id)) DEFINE_TREE(peer,peer_t *,peer_cmp,0) DEFINE_TREE(message,struct message *,id_cmp,0) struct tree_peer *peer_tree; struct tree_message *message_tree; int users_allocated; int chats_allocated; int messages_allocated; struct message message_list = { .next_use = &message_list, .prev_use = &message_list }; struct user *fetch_alloc_user (void) { int data[2]; prefetch_data (data, 8); peer_t *U = user_chat_get (MK_USER (data[1])); if (!U) { users_allocated ++; U = malloc (sizeof (*U)); memset (U, 0, sizeof (*U)); U->id = MK_USER (data[1]); peer_tree = tree_insert_peer (peer_tree, U, lrand48 ()); Peers[peer_num ++] = U; } fetch_user (&U->user); return &U->user; } struct secret_chat *fetch_alloc_encrypted_chat (void) { int data[2]; prefetch_data (data, 8); peer_t *U = user_chat_get (MK_ENCR_CHAT (data[1])); if (!U) { U = malloc (sizeof (*U)); memset (U, 0, sizeof (*U)); U->id = MK_ENCR_CHAT (data[1]); encr_chats_allocated ++; peer_tree = tree_insert_peer (peer_tree, U, lrand48 ()); Peers[peer_num ++] = U; } fetch_encrypted_chat (&U->encr_chat); return &U->encr_chat; } void insert_encrypted_chat (peer_t *P) { encr_chats_allocated ++; peer_tree = tree_insert_peer (peer_tree, P, lrand48 ()); Peers[peer_num ++] = P; } void insert_user (peer_t *P) { users_allocated ++; peer_tree = tree_insert_peer (peer_tree, P, lrand48 ()); Peers[peer_num ++] = P; } void insert_chat (peer_t *P) { chats_allocated ++; peer_tree = tree_insert_peer (peer_tree, P, lrand48 ()); Peers[peer_num ++] = P; } struct user *fetch_alloc_user_full (void) { int data[3]; prefetch_data (data, 12); peer_t *U = user_chat_get (MK_USER (data[2])); if (U) { fetch_user_full (&U->user); return &U->user; } else { users_allocated ++; U = malloc (sizeof (*U)); memset (U, 0, sizeof (*U)); U->id = MK_USER (data[2]); peer_tree = tree_insert_peer (peer_tree, U, lrand48 ()); fetch_user_full (&U->user); Peers[peer_num ++] = U; return &U->user; } } void free_user (struct user *U) { if (U->first_name) { free (U->first_name); } if (U->last_name) { free (U->last_name); } if (U->print_name) { free (U->print_name); } if (U->phone) { free (U->phone); } } void free_photo_size (struct photo_size *S) { free (S->type); if (S->data) { free (S->data); } } void free_photo (struct photo *P) { if (!P->access_hash) { return; } if (P->caption) { free (P->caption); } if (P->sizes) { int i; for (i = 0; i < P->sizes_num; i++) { free_photo_size (&P->sizes[i]); } free (P->sizes); } } void free_video (struct video *V) { if (!V->access_hash) { return; } free (V->caption); free_photo_size (&V->thumb); } void free_document (struct document *D) { if (!D->access_hash) { return; } free (D->caption); free (D->mime_type); free_photo_size (&D->thumb); } void free_message_media (struct message_media *M) { switch (M->type) { case CODE_message_media_empty: case CODE_message_media_geo: case CODE_message_media_audio: return; case CODE_message_media_photo: free_photo (&M->photo); return; case CODE_message_media_video: free_video (&M->video); return; case CODE_message_media_contact: free (M->phone); free (M->first_name); free (M->last_name); return; case CODE_message_media_document: free_document (&M->document); return; case CODE_message_media_unsupported: free (M->data); return; case CODE_decrypted_message_media_photo: case CODE_decrypted_message_media_video: case CODE_decrypted_message_media_audio: case CODE_decrypted_message_media_document: free (M->encr_photo.key); free (M->encr_photo.iv); return; case 0: break; default: logprintf ("%08x\n", M->type); assert (0); } } void free_message_action (struct message_action *M) { switch (M->type) { case CODE_message_action_empty: break; case CODE_message_action_chat_create: free (M->title); free (M->users); break; case CODE_message_action_chat_edit_title: free (M->new_title); break; case CODE_message_action_chat_edit_photo: free_photo (&M->photo); break; case CODE_message_action_chat_delete_photo: break; case CODE_message_action_chat_add_user: break; case CODE_message_action_chat_delete_user: break; case 0: break; default: assert (0); } } void free_message (struct message *M) { if (!M->service) { if (M->message) { free (M->message); } free_message_media (&M->media); } else { free_message_action (&M->action); } } void message_del_use (struct message *M) { M->next_use->prev_use = M->prev_use; M->prev_use->next_use = M->next_use; } void message_add_use (struct message *M) { M->next_use = message_list.next_use; M->prev_use = &message_list; M->next_use->prev_use = M; M->prev_use->next_use = M; } void message_add_peer (struct message *M) { peer_id_t id; if (!cmp_peer_id (M->to_id, MK_USER (our_id))) { id = M->from_id; } else { id = M->to_id; } peer_t *P = user_chat_get (id); if (!P) { P = malloc (sizeof (*P)); memset (P, 0, sizeof (*P)); P->id = id; switch (get_peer_type (id)) { case PEER_USER: users_allocated ++; break; case PEER_CHAT: chats_allocated ++; break; case PEER_GEO_CHAT: geo_chats_allocated ++; break; case PEER_ENCR_CHAT: encr_chats_allocated ++; break; } peer_tree = tree_insert_peer (peer_tree, P, lrand48 ()); Peers[peer_num ++] = P; } M->next = P->last; if (M->next) { M->next->prev = M; } M->prev = 0; P->last = M; } void message_del_peer (struct message *M) { peer_id_t id; if (!cmp_peer_id (M->to_id, MK_USER (our_id))) { id = M->from_id; } else { id = M->to_id; } peer_t *P = user_chat_get (id); if (M->prev) { M->prev->next = M->next; } if (M->next) { M->next->prev = M->prev; } if (P && P->last == M) { P->last = M->next; } } struct message *fetch_alloc_message (void) { struct message *M = malloc (sizeof (*M)); fetch_message (M); struct message *M1 = tree_lookup_message (message_tree, M); messages_allocated ++; if (M1) { message_del_use (M1); message_del_peer (M1); free_message (M1); memcpy (M1, M, sizeof (*M)); free (M); message_add_use (M1); message_add_peer (M1); messages_allocated --; return M1; } else { message_add_use (M); message_add_peer (M); message_tree = tree_insert_message (message_tree, M, lrand48 ()); return M; } } struct message *fetch_alloc_geo_message (void) { struct message *M = malloc (sizeof (*M)); fetch_geo_message (M); struct message *M1 = tree_lookup_message (message_tree, M); messages_allocated ++; if (M1) { message_del_use (M1); message_del_peer (M1); free_message (M1); memcpy (M1, M, sizeof (*M)); free (M); message_add_use (M1); message_add_peer (M1); messages_allocated --; return M1; } else { message_add_use (M); message_add_peer (M); message_tree = tree_insert_message (message_tree, M, lrand48 ()); return M; } } struct message *fetch_alloc_encrypted_message (void) { struct message *M = malloc (sizeof (*M)); fetch_encrypted_message (M); struct message *M1 = tree_lookup_message (message_tree, M); messages_allocated ++; if (M1) { message_del_use (M1); message_del_peer (M1); free_message (M1); memcpy (M1, M, sizeof (*M)); free (M); message_add_use (M1); message_add_peer (M1); messages_allocated --; return M1; } else { message_add_use (M); message_add_peer (M); message_tree = tree_insert_message (message_tree, M, lrand48 ()); return M; } } struct message *fetch_alloc_message_short (void) { struct message *M = malloc (sizeof (*M)); fetch_message_short (M); struct message *M1 = tree_lookup_message (message_tree, M); messages_allocated ++; if (M1) { message_del_use (M1); message_del_peer (M1); free_message (M1); memcpy (M1, M, sizeof (*M)); free (M); message_add_use (M1); message_add_peer (M1); messages_allocated --; return M1; } else { message_add_use (M); message_add_peer (M); message_tree = tree_insert_message (message_tree, M, lrand48 ()); return M; } } struct message *fetch_alloc_message_short_chat (void) { struct message *M = malloc (sizeof (*M)); fetch_message_short_chat (M); if (verbosity >= 2) { logprintf ("Read message with id %lld\n", M->id); } struct message *M1 = tree_lookup_message (message_tree, M); messages_allocated ++; if (M1) { message_del_use (M1); message_del_peer (M1); free_message (M1); memcpy (M1, M, sizeof (*M)); free (M); message_add_use (M1); message_add_peer (M1); messages_allocated --; return M1; } else { message_add_use (M); message_add_peer (M); message_tree = tree_insert_message (message_tree, M, lrand48 ()); return M; } } struct chat *fetch_alloc_chat (void) { int data[2]; prefetch_data (data, 8); peer_t *U = user_chat_get (MK_CHAT (data[1])); if (!U) { chats_allocated ++; U = malloc (sizeof (*U)); memset (U, 0, sizeof (*U)); U->id = MK_CHAT (data[1]); peer_tree = tree_insert_peer (peer_tree, U, lrand48 ()); Peers[peer_num ++] = U; } fetch_chat (&U->chat); return &U->chat; } struct chat *fetch_alloc_chat_full (void) { int data[3]; prefetch_data (data, 12); peer_t *U = user_chat_get (MK_CHAT (data[2])); if (U) { fetch_chat_full (&U->chat); return &U->chat; } else { chats_allocated ++; U = malloc (sizeof (*U)); memset (U, 0, sizeof (*U)); U->id = MK_CHAT (data[2]); peer_tree = tree_insert_peer (peer_tree, U, lrand48 ()); fetch_chat_full (&U->chat); Peers[peer_num ++] = U; return &U->chat; } } void free_chat (struct chat *U) { if (U->title) { free (U->title); } if (U->print_title) { free (U->print_title); } } int print_stat (char *s, int len) { return snprintf (s, len, "users_allocated\t%d\n" "chats_allocated\t%d\n" "secret_chats_allocated\t%d\n" "peer_num\t%d\n" "messages_allocated\t%d\n", users_allocated, chats_allocated, encr_chats_allocated, peer_num, messages_allocated ); } peer_t *user_chat_get (peer_id_t id) { static peer_t U; U.id = id; return tree_lookup_peer (peer_tree, &U); } struct message *message_get (long long id) { struct message M; M.id = id; return tree_lookup_message (message_tree, &M); } void update_message_id (struct message *M, long long id) { message_tree = tree_delete_message (message_tree, M); M->id = id; message_tree = tree_insert_message (message_tree, M, lrand48 ()); } void message_insert (struct message *M) { message_add_use (M); message_add_peer (M); message_tree = tree_insert_message (message_tree, M, lrand48 ()); }