diff options
Diffstat (limited to 'libs/tgl/src/binlog.c')
-rw-r--r-- | libs/tgl/src/binlog.c | 1830 |
1 files changed, 1830 insertions, 0 deletions
diff --git a/libs/tgl/src/binlog.c b/libs/tgl/src/binlog.c new file mode 100644 index 0000000000..74a47b38c2 --- /dev/null +++ b/libs/tgl/src/binlog.c @@ -0,0 +1,1830 @@ +/* + This file is part of tgl-library + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + This library 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 + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, write to the Free Software + Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + + Copyright Vitaly Valtman 2013-2015 +*/ + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +#if defined(WIN32) || defined(_WIN32) +#include <stdint.h> +#include <string.h> +#include <fcntl.h> +#define WIN32_LEAN_AND_MEAN +#include <windows.h> +#undef WIN32_LEAN_AND_MEAN +#include <io.h> +#include <sys/locking.h> +#else +#include <unistd.h> +#include <sys/file.h> +#endif +#include <sys/types.h> +#include <sys/stat.h> +#include <stdlib.h> +#include <fcntl.h> +#include <share.h> +#include <errno.h> +#include <stdio.h> +#include <string.h> +#include <assert.h> +#include <openssl/bn.h> + +#include "tgl-binlog.h" +#include "mtproto-common.h" +//#include "net.h" +#include "mtproto-client.h" +#include "mtproto-utils.h" + +#include "tgl.h" +#include "auto.h" +#include "auto/auto-types.h" +#include "auto/auto-skip.h" +#include "auto/auto-store-ds.h" +#include "auto/auto-fetch-ds.h" +#include "auto/auto-free-ds.h" + +#include "tgl-structures.h" +#include "tgl-methods-in.h" + +#include <openssl/sha.h> + +#define BINLOG_BUFFER_SIZE (1 << 20) +static int binlog_buffer[BINLOG_BUFFER_SIZE]; +static int *rptr; +static int *wptr; +//static int TLS->binlog_fd; +static int in_replay_log; // should be used ONLY for DEBUG + + +#define MAX_LOG_EVENT_SIZE (1 << 17) + +char *get_binlog_file_name (void); + +static void *alloc_log_event (int l) { + return binlog_buffer; +} + +static int mystreq1 (const char *a, const char *b, int l) { + if ((int)strlen (a) != l) { return 1; } + return memcmp (a, b, l); +} + +static long long binlog_pos; + +static int fetch_comb_binlog_start (struct tgl_state *TLS, void *extra) { + return 0; +} + +/* {{{ DC option */ +static int fetch_comb_binlog_dc_option (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + vlogprintf (E_NOTICE, "DC%d '%.*s' update: %.*s:%d\n", + DS_LVAL (DS_U->dc), + DS_RSTR (DS_U->name), + DS_RSTR (DS_U->ip), + DS_LVAL (DS_U->port) + ); + + tglmp_alloc_dc (TLS, + 0, + DS_LVAL (DS_U->dc), + DS_STR_DUP (DS_U->ip), + DS_LVAL (DS_U->port) + ); + return 0; +} + +static int fetch_comb_binlog_dc_option_new (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + vlogprintf (E_NOTICE, "DC%d '%.*s' update: %.*s:%d\n", + DS_LVAL (DS_U->dc), + DS_RSTR (DS_U->name), + DS_RSTR (DS_U->ip), + DS_LVAL (DS_U->port) + ); + + tglmp_alloc_dc (TLS, + DS_LVAL (DS_U->flags), + DS_LVAL (DS_U->dc), + DS_STR_DUP (DS_U->ip), + DS_LVAL (DS_U->port) + ); + return 0; +} +/* }}} */ + +/* {{{ Auth key */ +static int fetch_comb_binlog_auth_key (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + int num = DS_LVAL (DS_U->dc); + assert (num > 0 && num <= MAX_DC_ID); + assert (TLS->DC_list[num]); + + tglf_fetch_int_tuple ((void *)TLS->DC_list[num]->auth_key, DS_U->key->key, 64); + + static unsigned char sha1_buffer[20]; + SHA1 ((void *)TLS->DC_list[num]->auth_key, 256, sha1_buffer); + TLS->DC_list[num]->auth_key_id = *(long long *)(sha1_buffer + 12); + + TLS->DC_list[num]->flags |= TGLDCF_AUTHORIZED; + return 0; +} +/* }}} */ + +/* {{{ Default dc */ +static int fetch_comb_binlog_default_dc (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + int num = DS_LVAL (DS_U->dc); + assert (num > 0 && num <= MAX_DC_ID); + TLS->DC_working = TLS->DC_list[num]; + TLS->dc_working_num = num; + return 0; +} +/* }}} */ + +/* {{{ DC signed */ +static int fetch_comb_binlog_dc_signed (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + int num = DS_LVAL (DS_U->dc); + assert (num > 0 && num <= MAX_DC_ID); + assert (TLS->DC_list[num]); + TLS->DC_list[num]->flags |= TGLDCF_LOGGED_IN; + return 0; +} +/* }}} */ + +/* {{{ our user_id */ +static int fetch_comb_binlog_our_id (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + TLS->our_id = DS_LVAL (DS_U->id); + assert (TLS->our_id > 0); + if (TLS->callback.our_id) { + TLS->callback.our_id (TLS, TLS->our_id); + } + return 0; +} +/* }}} */ + +/* {{{ Set DH params */ +static int fetch_comb_binlog_set_dh_params (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + if (TLS->encr_prime) { tfree (TLS->encr_prime, 256); BN_free (TLS->encr_prime_bn); } + + TLS->encr_root = DS_LVAL (DS_U->root); + TLS->encr_prime = talloc (256); + tglf_fetch_int_tuple ((void *)TLS->encr_prime, DS_U->prime->key, 64); + + TLS->encr_prime_bn = BN_new (); + BN_bin2bn ((void *)TLS->encr_prime, 256, TLS->encr_prime_bn); + TLS->encr_param_version = DS_LVAL (DS_U->version); + + assert (tglmp_check_DH_params (TLS, TLS->encr_prime_bn, TLS->encr_root) >= 0); + + return 0; +} +/* }}} */ + +/* {{{ Set pts, qts, date, seq */ +static int fetch_comb_binlog_set_pts (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + TLS->pts = DS_LVAL (DS_U->pts); + return 0; +} + +static int fetch_comb_binlog_set_qts (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + TLS->qts = DS_LVAL (DS_U->qts); + return 0; +} + +static int fetch_comb_binlog_set_date (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + TLS->date = DS_LVAL (DS_U->date); + return 0; +} + +static int fetch_comb_binlog_set_seq (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + TLS->seq = DS_LVAL (DS_U->seq); + return 0; +} +/* }}} */ + +/* {{{ delete user */ +static int fetch_comb_binlog_user_delete (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_id_t id = TGL_MK_USER (DS_LVAL (DS_U->id)); + tgl_peer_t *U = tgl_peer_get (TLS, id); + assert (U); + U->flags |= TGLUF_DELETED; + + if (TLS->callback.user_update) { + TLS->callback.user_update (TLS, (void *)U, TGL_UPDATE_DELETED); + } + return 0; +} +/* }}} */ + +/* {{{ delete secret chat */ +static int fetch_comb_binlog_encr_chat_delete (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_id_t id = TGL_MK_ENCR_CHAT (DS_LVAL (DS_U->id)); + tgl_peer_t *_U = tgl_peer_get (TLS, id); + assert (_U); + struct tgl_secret_chat *U = &_U->encr_chat; + memset (U->key, 0, sizeof (U->key)); + U->flags |= TGLPF_DELETED; + U->state = sc_deleted; + if (U->g_key) { + tfree_secure (U->g_key, 256); + U->g_key = 0; + } + + if (TLS->callback.secret_chat_update) { + TLS->callback.secret_chat_update (TLS, U, TGL_UPDATE_DELETED); + } + return 0; +} +/* }}} */ + +static int fetch_comb_binlog_user_new (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_id_t id = TGL_MK_USER (DS_LVAL (DS_U->id)); + tgl_peer_t *_U = tgl_peer_get (TLS, id); + + int flags = DS_LVAL (DS_U->flags); + + unsigned updates = 0; + + if (flags & TGLPF_CREATE) { + if (!_U) { + _U = talloc0 (sizeof (*_U)); + _U->id = id; + tglp_insert_encrypted_chat (TLS, _U); + } else { + assert (!(_U->flags & TGLPF_CREATED)); + } + updates |= TGL_UPDATE_CREATED; + } else { + assert (_U->flags & TGLPF_CREATED); + } + + struct tgl_user *U = (void *)_U; + + if ((flags & 0xff) != (U->flags & 0xff)) { + updates |= TGL_UPDATE_FLAGS; + } + U->flags = flags & 0xffff; + + if (DS_U->access_hash) { + U->access_hash = DS_LVAL (DS_U->access_hash); + updates |= TGL_UPDATE_ACCESS_HASH; + } + + if (DS_U->first_name) { + if (U->first_name) { + tfree_str (U->first_name); + } + U->first_name = DS_STR_DUP (DS_U->first_name); + if (U->last_name) { + tfree_str (U->last_name); + } + U->last_name = DS_STR_DUP (DS_U->last_name); + + updates |= TGL_UPDATE_NAME; + + if (U->print_name) { + tglp_peer_delete_name (TLS, (void *)U); + tfree_str (U->print_name); + } + U->print_name = TLS->callback.create_print_name (TLS, U->id, U->first_name, U->last_name, 0, 0); + tglp_peer_insert_name (TLS, (void *)U); + } + + if (DS_U->phone) { + if (U->phone) { + tfree_str (U->phone); + } + U->phone = DS_STR_DUP (DS_U->phone); + updates |= TGL_UPDATE_PHONE; + } + + if (DS_U->username) { + if (U->username) { + tfree_str (U->username); + } + U->username = DS_STR_DUP (DS_U->username); + updates |= TGL_UPDATE_USERNAME; + } + + if (DS_U->photo) { + if (U->photo) { + tgls_free_photo (TLS, U->photo); + } + U->photo = tglf_fetch_alloc_photo_new (TLS, DS_U->photo); + U->flags |= TGLUF_HAS_PHOTO; + //updates |= TGL_UPDATE_PHOTO; + } + + if (DS_U->user_photo) { + U->photo_id = DS_LVAL (DS_U->user_photo->photo_id); + tglf_fetch_file_location_new (TLS, &U->photo_big, DS_U->user_photo->photo_big); + tglf_fetch_file_location_new (TLS, &U->photo_small, DS_U->user_photo->photo_small); + updates |= TGL_UPDATE_PHOTO; + } + + if (DS_U->real_first_name) { + if (U->real_first_name) { + tfree_str (U->real_first_name); + } + U->real_first_name = DS_STR_DUP (DS_U->real_first_name); + if (U->real_last_name) { + tfree_str (U->real_last_name); + } + U->real_last_name = DS_STR_DUP (DS_U->real_last_name); + + updates |= TGL_UPDATE_REAL_NAME; + } + + if (DS_U->last_read_in) { + U->last_read_in = DS_LVAL (DS_U->last_read_in); + tgls_messages_mark_read (TLS, U->last, 0, U->last_read_in); + } + + if (DS_U->last_read_out) { + U->last_read_out = DS_LVAL (DS_U->last_read_out); + tgls_messages_mark_read (TLS, U->last, TGLMF_OUT, U->last_read_out); + } + + if (DS_U->bot_info) { + if (U->bot_info) { + tgls_free_bot_info (TLS, U->bot_info); + } + U->bot_info = tglf_fetch_alloc_bot_info (TLS, DS_U->bot_info); + } + + if (TLS->callback.user_update && updates) { + TLS->callback.user_update (TLS, U, updates); + } + + return 0; +} + +static int fetch_comb_binlog_encr_chat_new (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_id_t id = TGL_MK_ENCR_CHAT (DS_LVAL (DS_U->id)); + tgl_peer_t *_U = tgl_peer_get (TLS, id); + + int flags = DS_LVAL (DS_U->flags); + + unsigned updates = 0; + + if (flags & TGLPF_CREATE) { + if (!_U) { + _U = talloc0 (sizeof (*_U)); + _U->id = id; + tglp_insert_encrypted_chat (TLS, _U); + } else { + assert (!(_U->flags & TGLPF_CREATED)); + } + updates |= TGL_UPDATE_CREATED; + } else { + assert (_U->flags & TGLPF_CREATED); + } + + struct tgl_secret_chat *U = (void *)_U; + + if ((flags & 0xff) != (U->flags & 0xff)) { + updates |= TGL_UPDATE_FLAGS; + } + U->flags = flags & 0xffff; + + if (DS_U->access_hash) { + U->access_hash = DS_LVAL (DS_U->access_hash); + updates |= TGL_UPDATE_ACCESS_HASH; + } + + if (DS_U->date) { + U->date = DS_LVAL (DS_U->date); + } + + if (DS_U->admin) { + U->admin_id = DS_LVAL (DS_U->admin); + } + + if (DS_U->user_id) { + U->user_id = DS_LVAL (DS_U->user_id); + } + + if (DS_U->key_fingerprint) { + U->key_fingerprint = DS_LVAL (DS_U->key_fingerprint); + } + + if (DS_U->in_seq_no) { + U->in_seq_no = DS_LVAL (DS_U->in_seq_no); + U->out_seq_no = DS_LVAL (DS_U->out_seq_no); + U->last_in_seq_no = DS_LVAL (DS_U->last_in_seq_no); + } + + tgl_peer_t *Us = tgl_peer_get (TLS, TGL_MK_USER (U->user_id)); + + if (!U->print_name) { + if (Us) { + U->print_name = TLS->callback.create_print_name (TLS, id, "!", Us->user.first_name, Us->user.last_name, 0); + } else { + static char buf[100]; + tsnprintf (buf, 99, "user#%d", U->user_id); + U->print_name = TLS->callback.create_print_name (TLS, id, "!", buf, 0, 0); + } + tglp_peer_insert_name (TLS, (void *)U); + } + + if (DS_U->g_key) { + if (!U->g_key) { + U->g_key = talloc (256); + } + tglf_fetch_int_tuple ((void *)U->g_key, DS_U->g_key->key, 64); + } + + if (DS_U->key) { + tglf_fetch_int_tuple (U->key, DS_U->key->key, 64); + } + + if (DS_U->state) { + if (U->state == sc_waiting && DS_LVAL (DS_U->state) == sc_ok) { + tgl_do_create_keys_end (TLS, U); + } + if ((int)U->state != DS_LVAL (DS_U->state)) { + switch (DS_LVAL (DS_U->state)) { + case sc_request: + updates |= TGL_UPDATE_REQUESTED; + break; + case sc_ok: + updates |= TGL_UPDATE_WORKING; + vlogprintf (E_WARNING, "Secret chat in ok state\n"); + break; + default: + break; + } + } + U->state = DS_LVAL (DS_U->state); + } + + if (TLS->callback.secret_chat_update && updates) { + TLS->callback.secret_chat_update (TLS, U, updates); + } + + return 0; +} + +static int fetch_comb_binlog_chat_new (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_id_t id = TGL_MK_CHAT (DS_LVAL (DS_U->id)); + tgl_peer_t *_U = tgl_peer_get (TLS, id); + + int flags = DS_LVAL (DS_U->flags); + + unsigned updates = 0; + + if (flags & (1 << 16)) { + if (!_U) { + _U = talloc0 (sizeof (*_U)); + _U->id = id; + tglp_insert_chat (TLS, _U); + } else { + assert (!(_U->flags & TGLPF_CREATED)); + } + updates |= TGL_UPDATE_CREATED; + } else { + assert (_U->flags & TGLPF_CREATED); + } + + struct tgl_chat *C = &_U->chat; + + if ((flags & 0xff) != (C->flags & 0xff)) { + updates |= TGL_UPDATE_FLAGS; + } + C->flags = flags & 0xffff; + + if (DS_U->title) { + if (C->title) { + tfree_str (C->title); + } + C->title = DS_STR_DUP (DS_U->title); + + if (C->print_title) { + tglp_peer_delete_name (TLS, (void *)C); + tfree_str (C->print_title); + } + C->print_title = TLS->callback.create_print_name (TLS, C->id, C->title, 0, 0, 0); + tglp_peer_insert_name (TLS, (void *)C); + + updates |= TGL_UPDATE_TITLE; + } + + if (DS_U->user_num) { + C->users_num = DS_LVAL (DS_U->user_num); + } + + if (DS_U->date) { + C->date = DS_LVAL (DS_U->date); + } + + if (DS_U->chat_photo) { + tglf_fetch_file_location_new (TLS, &C->photo_big, DS_U->chat_photo->photo_big); + tglf_fetch_file_location_new (TLS, &C->photo_small, DS_U->chat_photo->photo_small); + updates |= TGL_UPDATE_PHOTO; + } + + if (DS_U->photo) { + if (C->photo) { + tgls_free_photo (TLS, C->photo); + } + C->photo = tglf_fetch_alloc_photo_new (TLS, DS_U->photo); + C->flags |= TGLPF_HAS_PHOTO; + updates |= TGL_UPDATE_PHOTO; + } + + if (DS_U->admin) { + C->admin_id = DS_LVAL (DS_U->admin); + updates |= TGL_UPDATE_ADMIN; + } + + if (DS_U->version) { + C->version = DS_LVAL (DS_U->version); + + if (C->user_list) { tfree (C->user_list, 12 * C->user_list_size); } + + C->user_list_size = DS_LVAL (DS_U->participants->cnt); + C->user_list = talloc (12 * C->user_list_size); + + int i; + for (i = 0; i < C->user_list_size; i++) { + C->user_list[i].user_id = DS_LVAL (DS_U->participants->data[i]->user_id); + C->user_list[i].inviter_id = DS_LVAL (DS_U->participants->data[i]->inviter_id); + C->user_list[i].date = DS_LVAL (DS_U->participants->data[i]->date); + } + + updates |= TGL_UPDATE_MEMBERS; + } + + if (DS_U->last_read_in) { + C->last_read_in = DS_LVAL (DS_U->last_read_in); + tgls_messages_mark_read (TLS, C->last, 0, C->last_read_in); + } + + if (DS_U->last_read_out) { + C->last_read_out = DS_LVAL (DS_U->last_read_out); + tgls_messages_mark_read (TLS, C->last, TGLMF_OUT, C->last_read_out); + } + + + if (TLS->callback.chat_update && updates) { + TLS->callback.chat_update (TLS, C, updates); + } + return 0; +} + +static int fetch_comb_binlog_chat_add_participant (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_id_t id = TGL_MK_CHAT (DS_LVAL (DS_U->id)); + tgl_peer_t *_C = tgl_peer_get (TLS, id); + assert (_C && (_C->flags & TGLPF_CREATED)); + struct tgl_chat *C = &_C->chat; + + int version = DS_LVAL (DS_U->version); + int user = DS_LVAL (DS_U->user_id); + int inviter = DS_LVAL (DS_U->inviter_id); + int date = DS_LVAL (DS_U->date); + + + if (C->user_list_version > version) { return 0; } + + int i; + for (i = 0; i < C->user_list_size; i++) { + if (C->user_list[i].user_id == user) { + return 0; + } + } + + C->user_list_size ++; + C->user_list = trealloc (C->user_list, 12 * C->user_list_size - 12, 12 * C->user_list_size); + C->user_list[C->user_list_size - 1].user_id = user; + C->user_list[C->user_list_size - 1].inviter_id = inviter; + C->user_list[C->user_list_size - 1].date = date; + C->user_list_version = version; + + if (TLS->callback.chat_update) { + TLS->callback.chat_update (TLS, C, TGL_UPDATE_MEMBERS); + } + return 0; +} + +static int fetch_comb_binlog_chat_del_participant (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_id_t id = TGL_MK_CHAT (DS_LVAL (DS_U->id)); + tgl_peer_t *_C = tgl_peer_get (TLS, id); + assert (_C && (_C->flags & TGLPF_CREATED)); + struct tgl_chat *C = &_C->chat; + + int version = DS_LVAL (DS_U->version); + int user = DS_LVAL (DS_U->user_id); + if (C->user_list_version > version) { return 0; } + + int i; + for (i = 0; i < C->user_list_size; i++) { + if (C->user_list[i].user_id == user) { + struct tgl_chat_user t; + t = C->user_list[i]; + C->user_list[i] = C->user_list[C->user_list_size - 1]; + C->user_list[C->user_list_size - 1] = t; + } + } + if (C->user_list[C->user_list_size - 1].user_id != user) { return 0; } + + assert (C->user_list[C->user_list_size - 1].user_id == user); + C->user_list_size --; + C->user_list = trealloc (C->user_list, 12 * C->user_list_size + 12, 12 * C->user_list_size); + C->user_list_version = version; + + if (TLS->callback.chat_update) { + TLS->callback.chat_update (TLS, C, TGL_UPDATE_MEMBERS); + } + return 0; +} + +static int fetch_comb_binlog_message_new (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + struct tgl_message *M = tgl_message_get (TLS, DS_LVAL (DS_U->lid)); + int flags = DS_LVAL (DS_U->flags); + + if (flags & (1 << 16)) { + if (!M) { + M = tglm_message_alloc (TLS, DS_LVAL (DS_U->lid)); + } + assert (!(M->flags & TGLMF_CREATED)); + } else { + assert (M->flags & TGLMF_CREATED); + } + + assert (flags & TGLMF_CREATED); + assert (!(M->flags & TGLMF_ENCRYPTED)); + assert (!(flags & TGLMF_ENCRYPTED)); + + if ((M->flags & TGLMF_PENDING) && !(flags & TGLMF_PENDING)){ + tglm_message_remove_unsent (TLS, M); + } + if (!(M->flags & TGLMF_PENDING) && (flags & TGLMF_PENDING)){ + tglm_message_insert_unsent (TLS, M); + } + + if ((M->flags & TGLMF_UNREAD) && !(flags & TGLMF_UNREAD)) { + M->flags = (flags & 0xffff) | TGLMF_UNREAD; + } else { + M->flags = (flags & 0xffff); + } + + if (DS_U->from_id) { + M->from_id = TGL_MK_USER (DS_LVAL (DS_U->from_id)); + } + if (DS_U->to_type) { + assert (flags & 0x10000); + M->to_id = tgl_set_peer_id (DS_LVAL (DS_U->to_type), DS_LVAL (DS_U->to_id)); + assert (DS_LVAL (DS_U->to_type) != TGL_PEER_ENCR_CHAT); + } + + if (DS_U->date) { + M->date = DS_LVAL (DS_U->date); + } + + if (DS_U->fwd_from_id) { + M->fwd_from_id = TGL_MK_USER (DS_LVAL (DS_U->fwd_from_id)); + M->fwd_date = DS_LVAL (DS_U->fwd_date); + } + + if (DS_U->action) { + tglf_fetch_message_action_new (TLS, &M->action, DS_U->action); + M->flags |= TGLMF_SERVICE; + } + + if (DS_U->message) { + M->message_len = DS_U->message->len; + M->message = DS_STR_DUP (DS_U->message); + assert (!(M->flags & TGLMF_SERVICE)); + } + + if (DS_U->media) { + tglf_fetch_message_media_new (TLS, &M->media, DS_U->media); + assert (!(M->flags & TGLMF_SERVICE)); + } + + if (DS_U->reply_id) { + M->reply_id = DS_LVAL (DS_U->reply_id); + } + + if (flags & 0x10000) { + tglm_message_insert (TLS, M); + } + + if (!(flags & TGLMF_UNREAD) && (M->flags & TGLMF_UNREAD)) { + tgls_messages_mark_read (TLS, M, M->flags & TGLMF_OUT, M->id); + } + + if (DS_U->reply_markup) { + M->reply_markup = tglf_fetch_alloc_reply_markup (TLS, M->next, DS_U->reply_markup); + } + return 0; +} + +static int fetch_comb_binlog_message_encr_new (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + struct tgl_message *M = tgl_message_get (TLS, DS_LVAL (DS_U->lid)); + int flags = DS_LVAL (DS_U->flags); + + if (flags & (1 << 16)) { + if (!M) { + M = tglm_message_alloc (TLS, DS_LVAL (DS_U->lid)); + } else { + assert (!(M->flags & TGLMF_CREATED)); + } + assert (!(M->flags & TGLMF_CREATED)); + } else { + assert (M->flags & TGLMF_CREATED); + } + + assert (flags & TGLMF_CREATED); + assert (flags & TGLMF_ENCRYPTED); + + if ((M->flags & TGLMF_PENDING) && !(flags & TGLMF_PENDING)){ + tglm_message_remove_unsent (TLS, M); + } + if (!(M->flags & TGLMF_PENDING) && (flags & TGLMF_PENDING)){ + tglm_message_insert_unsent (TLS, M); + } + + M->flags = flags & 0xffff; + + if (DS_U->from_id) { + M->from_id = TGL_MK_USER (DS_LVAL (DS_U->from_id)); + } + if (DS_U->to_type) { + assert (flags & 0x10000); + M->to_id = tgl_set_peer_id (DS_LVAL (DS_U->to_type), DS_LVAL (DS_U->to_id)); + } + + if (DS_U->date) { + M->date = DS_LVAL (DS_U->date); + } + + struct tgl_secret_chat *E = (void *)tgl_peer_get (TLS, M->to_id); + assert (E); + + if (DS_U->message) { + M->message_len = DS_U->message->len; + M->message = DS_STR_DUP (DS_U->message); + assert (!(M->flags & TGLMF_SERVICE)); + } + + if (DS_U->encr_media) { + tglf_fetch_message_media_encrypted_new (TLS, &M->media, DS_U->encr_media); + assert (!(M->flags & TGLMF_SERVICE)); + } + + if (DS_U->encr_action) { + tglf_fetch_message_action_encrypted_new (TLS, &M->action, DS_U->encr_action); + M->flags |= TGLMF_SERVICE; + } + + if (DS_U->file) { + tglf_fetch_encrypted_message_file_new (TLS, &M->media, DS_U->file); + assert (!(M->flags & TGLMF_SERVICE)); + } + + if (DS_U->encr_action && !(M->flags & TGLMF_OUT) && M->action.type == tgl_message_action_notify_layer) { + E->layer = M->action.layer; + } + + if ((flags & TGLMF_CREATE) && (flags & TGLMF_OUT)) { + E->out_seq_no ++; + } + + if (flags & 0x10000) { + tglm_message_insert (TLS, M); + } + return 0; +} + +static int fetch_comb_binlog_set_msg_id (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + struct tgl_message *M = tgl_message_get (TLS, DS_LVAL (DS_U->old_id)); + assert (M); + if (M->flags & TGLMF_PENDING) { + tglm_message_remove_unsent (TLS, M); + M->flags &= ~TGLMF_PENDING; + } + + tglm_message_remove_tree (TLS, M); + tglm_message_del_peer (TLS, M); + + M->id = DS_LVAL (DS_U->new_id); + if (tgl_message_get (TLS, M->id)) { + tglm_message_del_use (TLS, M); + tgls_free_message (TLS, M); + } else { + tglm_message_insert_tree (TLS, M); + tglm_message_add_peer (TLS, M); + } + return 0; +} + +static int fetch_comb_binlog_message_delete (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + struct tgl_message *M = tgl_message_get (TLS, DS_LVAL (DS_U->lid)); + assert (M); + if (M->flags & TGLMF_PENDING) { + tglm_message_remove_unsent (TLS, M); + M->flags &= ~TGLMF_PENDING; + } + tglm_message_remove_tree (TLS, M); + tglm_message_del_peer (TLS, M); + tglm_message_del_use (TLS, M); + tgls_free_message (TLS, M); + return 0; +} + +static int fetch_comb_binlog_msg_update (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + struct tgl_message *M = tgl_message_get (TLS, DS_LVAL (DS_U->lid)); + if (!M) { return 0; } + assert (M); + + if (!(M->flags & TGLMF_ENCRYPTED)) { + if (TLS->max_msg_id < M->id) { + TLS->max_msg_id = M->id; + } + } + + if (TLS->callback.msg_receive) { + TLS->callback.msg_receive (TLS, M); + } + return 0; +} + +static int fetch_comb_binlog_reset_authorization (struct tgl_state *TLS, void *extra) { + int i; + for (i = 0; i <= TLS->max_dc_num; i++) if (TLS->DC_list[i]) { + struct tgl_dc *D = TLS->DC_list[i]; + D->flags = 0; + D->state = st_init; + D->auth_key_id = D->temp_auth_key_id = 0; + } + TLS->seq = 0; + TLS->qts = 0; + return 0; +} + +static int fetch_comb_binlog_encr_chat_exchange_new (struct tgl_state *TLS, struct tl_ds_binlog_update *DS_U) { + tgl_peer_t *P = tgl_peer_get (TLS, TGL_MK_ENCR_CHAT (DS_LVAL (DS_U->id))); + assert (P); + if (DS_U->state) { + P->encr_chat.exchange_state = DS_LVAL (DS_U->state); + } + if (DS_U->exchange_id) { + P->encr_chat.exchange_id = DS_LVAL (DS_U->exchange_id); + } + + static unsigned char sha_buffer[20]; + switch (P->encr_chat.exchange_state) { + case tgl_sce_requested: + tglf_fetch_int_tuple (P->encr_chat.exchange_key, DS_U->key->key, 64); + break; + case tgl_sce_accepted: + tglf_fetch_int_tuple (P->encr_chat.exchange_key, DS_U->key->key, 64); + + SHA1 ((unsigned char *)P->encr_chat.exchange_key, 256, sha_buffer); + P->encr_chat.exchange_key_fingerprint = *(long long *)(sha_buffer + 12); + break; + case tgl_sce_committed: + memcpy (P->encr_chat.exchange_key, P->encr_chat.key, 256); + P->encr_chat.exchange_key_fingerprint = P->encr_chat.key_fingerprint; + + tglf_fetch_int_tuple (P->encr_chat.key, DS_U->key->key, 64); + + SHA1 ((unsigned char *)P->encr_chat.key, 256, sha_buffer); + P->encr_chat.key_fingerprint = *(long long *)(sha_buffer + 12); + break; + case tgl_sce_confirmed: + P->encr_chat.exchange_state = tgl_sce_none; + if (P->encr_chat.exchange_state != tgl_sce_committed) { + memcpy (P->encr_chat.key, P->encr_chat.exchange_key, 256); + P->encr_chat.key_fingerprint = P->encr_chat.exchange_key_fingerprint; + } + break; + case tgl_sce_aborted: + P->encr_chat.exchange_state = tgl_sce_none; + if (P->encr_chat.exchange_state == tgl_sce_committed) { + memcpy (P->encr_chat.key, P->encr_chat.exchange_key, 256); + P->encr_chat.key_fingerprint = P->encr_chat.exchange_key_fingerprint; + } + break; + default: + assert (0); + } + return 0; +} + +#define FETCH_COMBINATOR_FUNCTION(NAME) \ + case CODE_ ## NAME:\ + ok = fetch_comb_ ## NAME (TLS, DS_U); \ + break; \ + + +static void replay_log_event (struct tgl_state *TLS) { + assert (rptr < wptr); + int op = *rptr; + + vlogprintf (E_DEBUG, "replay_log_event: log_pos=%"_PRINTF_INT64_"d, op=0x%08x\n", binlog_pos, op); + + in_ptr = rptr; + in_end = wptr; + if (skip_type_any (TYPE_TO_PARAM (binlog_update)) < 0) { + vlogprintf (E_ERROR, "Can not replay at %"_PRINTF_INT64_"d (magic = 0x%08x)\n", binlog_pos, *rptr); + assert (0); + } + int *end = in_ptr; + in_end = in_ptr; + in_ptr = rptr; + struct tl_ds_binlog_update *DS_U = fetch_ds_type_binlog_update (TYPE_TO_PARAM (binlog_update)); + assert (in_ptr == end); + + int ok = -1; + + switch (op) { + FETCH_COMBINATOR_FUNCTION (binlog_start) + FETCH_COMBINATOR_FUNCTION (binlog_dc_option) + FETCH_COMBINATOR_FUNCTION (binlog_dc_option_new) + FETCH_COMBINATOR_FUNCTION (binlog_auth_key) + FETCH_COMBINATOR_FUNCTION (binlog_default_dc) + FETCH_COMBINATOR_FUNCTION (binlog_dc_signed) + + FETCH_COMBINATOR_FUNCTION (binlog_our_id) + + FETCH_COMBINATOR_FUNCTION (binlog_set_dh_params) + FETCH_COMBINATOR_FUNCTION (binlog_set_pts) + FETCH_COMBINATOR_FUNCTION (binlog_set_qts) + FETCH_COMBINATOR_FUNCTION (binlog_set_date) + FETCH_COMBINATOR_FUNCTION (binlog_set_seq) + + FETCH_COMBINATOR_FUNCTION (binlog_user_new) + FETCH_COMBINATOR_FUNCTION (binlog_user_delete) + + FETCH_COMBINATOR_FUNCTION (binlog_chat_new) + //FETCH_COMBINATOR_FUNCTION (binlog_chat_delete) + + FETCH_COMBINATOR_FUNCTION (binlog_encr_chat_new) + FETCH_COMBINATOR_FUNCTION (binlog_encr_chat_delete) + FETCH_COMBINATOR_FUNCTION (binlog_chat_add_participant) + FETCH_COMBINATOR_FUNCTION (binlog_chat_del_participant) + + FETCH_COMBINATOR_FUNCTION (binlog_message_new) + FETCH_COMBINATOR_FUNCTION (binlog_message_encr_new) + FETCH_COMBINATOR_FUNCTION (binlog_message_delete) + FETCH_COMBINATOR_FUNCTION (binlog_set_msg_id) + + FETCH_COMBINATOR_FUNCTION (binlog_encr_chat_exchange_new) + + FETCH_COMBINATOR_FUNCTION (binlog_msg_update) + FETCH_COMBINATOR_FUNCTION (binlog_reset_authorization) + default: + vlogprintf (E_ERROR, "Unknown op 0x%08x\n", op); + assert (0); + } + assert (ok >= 0); + + free_ds_type_binlog_update (DS_U, TYPE_TO_PARAM (binlog_update)); + assert (in_ptr == end); + //assert (in_ptr == in_end); + binlog_pos += (in_ptr - rptr) * 4; + rptr = in_ptr; +} + +static void create_new_binlog (struct tgl_state *TLS) { + clear_packet (); + //static int s[1000]; + + //packet_ptr = s; + out_int (CODE_binlog_start); + if (TLS->test_mode) { + out_int (CODE_binlog_dc_option); + out_int (1); + out_string (""); + out_string (TG_SERVER_TEST_1); + out_int (443); + out_int (CODE_binlog_dc_option); + out_int (2); + out_string (""); + out_string (TG_SERVER_TEST_2); + out_int (443); + out_int (CODE_binlog_dc_option); + out_int (3); + out_string (""); + out_string (TG_SERVER_TEST_3); + out_int (443); + out_int (CODE_binlog_default_dc); + out_int (2); + } else { + out_int (CODE_binlog_dc_option); + out_int (1); + out_string (""); + out_string (TG_SERVER_1); + out_int (443); + out_int (CODE_binlog_dc_option); + out_int (2); + out_string (""); + out_string (TG_SERVER_2); + out_int (443); + out_int (CODE_binlog_dc_option); + out_int (3); + out_string (""); + out_string (TG_SERVER_3); + out_int (443); + out_int (CODE_binlog_dc_option); + out_int (4); + out_string (""); + out_string (TG_SERVER_4); + out_int (443); + out_int (CODE_binlog_dc_option); + out_int (5); + out_string (""); + out_string (TG_SERVER_5); + out_int (443); + out_int (CODE_binlog_default_dc); + out_int (2); + } + +#if defined(_MSC_VER) && _MSC_VER >= 1400 + int fd = 0; + if(_sopen_s (&fd, TLS->binlog_name, _O_WRONLY | _O_EXCL | _O_CREAT, _SH_DENYNO, _S_IREAD | _S_IWRITE) != 0 ) { +#else + int fd = open (TLS->binlog_name, O_WRONLY | O_EXCL | O_CREAT, 0600); + if (fd < 0) { +#endif + perror ("Write new binlog"); + exit (2); + } + assert (write (fd, packet_buffer, (packet_ptr - packet_buffer) * 4) == (packet_ptr - packet_buffer) * 4); + close (fd); +} + + +void tgl_replay_log (struct tgl_state *TLS) { + if (!TLS->binlog_enabled) { return; } +#if defined(WIN32) || defined(_WIN32) + if (INVALID_FILE_ATTRIBUTES == GetFileAttributesA (TLS->binlog_name) && GetLastError () == ERROR_FILE_NOT_FOUND) { +#else + if (access (TLS->binlog_name, F_OK) < 0) { +#endif + printf ("No binlog found. Creating new one\n"); + create_new_binlog (TLS); + } +#if defined(_MSC_VER) && _MSC_VER >= 1400 + int fd = 0; + if (_sopen_s(&fd, TLS->binlog_name, _O_RDONLY | _O_BINARY, _SH_DENYNO, _S_IREAD | _S_IWRITE) != 0) { +#elif defined(WIN32) || defined(_WIN32) + int fd = open (TLS->binlog_name, O_RDONLY | O_BINARY); + if (fd < 0) { +#else + int fd = open (TLS->binlog_name, O_RDONLY); + if (fd < 0) { +#endif + perror ("binlog open"); + exit (2); + } + int end = 0; + in_replay_log = 1; + while (1) { + if (!end && wptr - rptr < MAX_LOG_EVENT_SIZE / 4) { + if (wptr == rptr) { + wptr = rptr = binlog_buffer; + } else { + int x = wptr - rptr; + memcpy (binlog_buffer, rptr, 4 * x); + wptr -= (rptr - binlog_buffer); + rptr = binlog_buffer; + } + int l = (binlog_buffer + BINLOG_BUFFER_SIZE - wptr) * 4; + int k = read (fd, wptr, l); + if (k < 0) { + perror ("read binlog"); + exit (2); + } + assert (!(k & 3)); + if (k < l) { + end = 1; + } + wptr += (k / 4); + } + if (wptr == rptr) { break; } + replay_log_event (TLS); + } + in_replay_log = 0; + close (fd); +} + +//static int b_packet_buffer[PACKET_BUFFER_SIZE]; + +void tgl_reopen_binlog_for_writing (struct tgl_state *TLS) { +#if defined(_MSC_VER) && _MSC_VER >= 1400 + if (_sopen_s (&TLS->binlog_fd, TLS->binlog_name, _O_WRONLY | _O_BINARY, _SH_DENYNO, _S_IREAD | _S_IWRITE) != 0) { +#elif defined(WIN32) || defined(_WIN32) + TLS->binlog_fd = open (TLS->binlog_name, O_WRONLY | _O_BINARY); + if (TLS->binlog_fd < 0) { +#else + TLS->binlog_fd = open (TLS->binlog_name, O_WRONLY); + if (TLS->binlog_fd < 0) { +#endif + perror ("binlog open"); + exit (2); + } + + assert (lseek (TLS->binlog_fd, binlog_pos, SEEK_SET) == binlog_pos); +#if defined(WIN32) || defined(_WIN32) + HANDLE h = INVALID_HANDLE_VALUE; + DWORD size_lower, size_upper; + DWORD err = 0; + OVERLAPPED ovlp; + int flags = 0; + + h = (HANDLE)_get_osfhandle(TLS->binlog_fd); + if (h == INVALID_HANDLE_VALUE) { + errno = EBADF; + goto error; + } + + size_lower = GetFileSize (h, &size_upper); + if (size_lower == INVALID_FILE_SIZE) { + goto get_err; + } + + memset (&ovlp, 0, sizeof ovlp); + flags |= LOCKFILE_EXCLUSIVE_LOCK | LOCKFILE_FAIL_IMMEDIATELY; + + if (!LockFileEx (h, flags, 0, size_lower, size_upper, &ovlp)) { + goto get_err; + } + return; + +error: + perror ("get lock"); + exit(2); + +get_err: + err = GetLastError(); + switch (err) + { + case ERROR_LOCK_VIOLATION: + errno = EAGAIN; + break; + case ERROR_NOT_ENOUGH_MEMORY: + errno = ENOMEM; + break; + case ERROR_BAD_COMMAND: + errno = EINVAL; + break; + default: + errno = err; + } + goto error; +#else + if (flock (TLS->binlog_fd, LOCK_EX | LOCK_NB) < 0) { + perror ("get lock"); + exit (2); + } +#endif +} + +static void add_log_event (struct tgl_state *TLS, const int *data, int len) { + vlogprintf (E_DEBUG, "Add log event: magic = 0x%08x, len = %d\n", data[0], len); + assert (!(len & 3)); + int *ev = talloc (len); + memcpy (ev, data, len); + rptr = (void *)ev; + wptr = rptr + (len / 4); + int *in = in_ptr; + int *end = in_end; + replay_log_event (TLS); + if (rptr != wptr) { + vlogprintf (E_ERROR, "Unread %"_PRINTF_INT64_"d ints. Len = %d\n", (long long)(wptr - rptr), len); + assert (rptr == wptr); + } + if (TLS->binlog_enabled) { + assert (TLS->binlog_fd > 0); + assert (write (TLS->binlog_fd, ev, len) == len); + } + tfree (ev, len); + in_ptr = in; + in_end = end; +} + +void bl_do_dc_option_new (struct tgl_state *TLS, int flags, int id, const char *name, int l1, const char *ip, int l2, int port) { + struct tgl_dc *DC = TLS->DC_list[id]; + + if (DC) { + struct tgl_dc_option *O = DC->options[flags & 3]; + while (O) { + if (!strncmp (O->ip, ip, l2)) { + return; + } + O = O->next; + } + } + + clear_packet (); + out_int (CODE_binlog_dc_option_new); + out_int (flags); + out_int (id); + + out_cstring (name, l1); + out_cstring (ip, l2); + out_int (port); + + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_dc_option (struct tgl_state *TLS, int id, const char *name, int l1, const char *ip, int l2, int port) { + bl_do_dc_option_new (TLS, 0, id, name, l1, ip, l2, port); +} + +void bl_do_set_working_dc (struct tgl_state *TLS, int num) { + int *ev = alloc_log_event (8); + ev[0] = CODE_binlog_default_dc; + ev[1] = num; + add_log_event (TLS, ev, 8); +} + +void bl_do_dc_signed (struct tgl_state *TLS, int id) { + clear_packet (); + out_int (CODE_binlog_dc_signed); + out_int (id); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_our_id (struct tgl_state *TLS, int id) { + if (TLS->our_id) { + assert (TLS->our_id == id); + return; + } + + clear_packet (); + out_int (CODE_binlog_our_id); + out_int (id); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_dh_params (struct tgl_state *TLS, int root, unsigned char prime[], int version) { + clear_packet (); + out_int (CODE_binlog_set_dh_params); + out_int (root); + out_ints ((void *)prime, 64); + out_int (version); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_pts (struct tgl_state *TLS, int pts) { + if (TLS->locks & TGL_LOCK_DIFF) { return; } + if (pts <= TLS->pts) { return; } + + clear_packet (); + out_int (CODE_binlog_set_pts); + out_int (pts); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_qts (struct tgl_state *TLS, int qts) { + if (TLS->locks & TGL_LOCK_DIFF) { return; } + if (qts <= TLS->qts) { return; } + + clear_packet (); + out_int (CODE_binlog_set_qts); + out_int (qts); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_date (struct tgl_state *TLS, int date) { + if (TLS->locks & TGL_LOCK_DIFF) { return; } + if (date <= TLS->date) { return; } + + clear_packet (); + out_int (CODE_binlog_set_date); + out_int (date); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_seq (struct tgl_state *TLS, int seq) { + if (TLS->locks & TGL_LOCK_DIFF) { return; } + if (seq <= TLS->seq) { return; } + + clear_packet (); + out_int (CODE_binlog_set_seq); + out_int (seq); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_msg_id (struct tgl_state *TLS, struct tgl_message *M, int id) { + if (M->id == id) { return; } + + clear_packet (); + out_int (CODE_binlog_set_msg_id); + out_long (M->id); + out_int (id); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_user_delete (struct tgl_state *TLS, struct tgl_user *U) { + if (U->flags & TGLUF_DELETED) { return; } + + clear_packet (); + out_int (CODE_binlog_user_delete); + out_int (tgl_get_peer_id (U->id)); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_encr_chat_delete (struct tgl_state *TLS, struct tgl_secret_chat *U) { + if (!(U->flags & TGLPF_CREATED) || U->state == sc_deleted || U->state == sc_none) { return; } + + clear_packet (); + out_int (CODE_binlog_encr_chat_delete); + out_int (tgl_get_peer_id (U->id)); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_chat_add_user (struct tgl_state *TLS, struct tgl_chat *C, int version, int user, int inviter, int date) { + if (C->user_list_version >= version || !C->user_list_version) { return; } + + clear_packet (); + out_int (CODE_binlog_chat_add_participant); + out_int (tgl_get_peer_id (C->id)); + out_int (version); + out_int (user); + out_int (inviter); + out_int (date); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_chat_del_user (struct tgl_state *TLS, struct tgl_chat *C, int version, int user) { + if (C->user_list_version >= version || !C->user_list_version) { return; } + + clear_packet (); + out_int (CODE_binlog_chat_del_participant); + out_int (tgl_get_peer_id (C->id)); + out_int (version); + out_int (user); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_create_message_new (struct tgl_state *TLS, long long id, int *from_id, int *to_type, int *to_id, int *fwd_from_id, int *fwd_date, int *date, const char *message, int message_len, struct tl_ds_message_media *media, struct tl_ds_message_action *action, int *reply_id, struct tl_ds_reply_markup *reply_markup, int flags) { + clear_packet (); + assert (!(flags & 0xfffe0000)); + + out_int (CODE_binlog_message_new); + int *flags_p = packet_ptr; + out_int (flags); + assert (*flags_p == flags); + + out_long (id); + + if (from_id) { + assert (to_type); + assert (to_id); + (*flags_p) |= (1 << 17); + out_int (*from_id); + out_int (*to_type); + out_int (*to_id); + } + + if (fwd_from_id) { + assert (fwd_date); + (*flags_p) |= (1 << 18); + out_int (*fwd_from_id); + out_int (*fwd_date); + } + + if (date) { + (*flags_p) |= (1 << 19); + out_int (*date); + } + + if (message) { + (*flags_p) |= (1 << 20); + out_cstring (message, message_len); + } + + if (media) { + (*flags_p) |= (1 << 21); + store_ds_type_message_media (media, TYPE_TO_PARAM (message_media)); + } + + if (action) { + (*flags_p) |= (1 << 22); + + store_ds_type_message_action (action, TYPE_TO_PARAM (message_action)); + } + + if (reply_id) { + (*flags_p) |= (1 << 23); + out_int (*reply_id); + } + + if (reply_markup) { + (*flags_p) |= (1 << 24); + store_ds_type_reply_markup (reply_markup, TYPE_TO_PARAM (reply_markup)); + } + + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_create_message_encr_new (struct tgl_state *TLS, long long id, int *from_id, int *to_type, int *to_id, int *date, const char *message, int message_len, struct tl_ds_decrypted_message_media *media, struct tl_ds_decrypted_message_action *action, struct tl_ds_encrypted_file *file, int flags) { + clear_packet (); + assert (!(flags & 0xfffe0000)); + + out_int (CODE_binlog_message_encr_new); + int *flags_p = packet_ptr; + out_int (flags); + assert (flags & TGLMF_ENCRYPTED); + assert (*flags_p == flags); + + out_long (id); + + if (from_id) { + assert (to_id); + assert (to_type); + (*flags_p) |= (1 << 17); + out_int (*from_id); + out_int (*to_type); + out_int (*to_id); + } + + if (date) { + (*flags_p) |= (1 << 19); + out_int (*date); + } + + if (message) { + (*flags_p) |= (1 << 20); + out_cstring (message, message_len); + } + + if (media) { + (*flags_p) |= (1 << 21); + store_ds_type_decrypted_message_media (media, TYPE_TO_PARAM (decrypted_message_media)); + } + + if (action) { + (*flags_p) |= (1 << 22); + store_ds_type_decrypted_message_action (action, TYPE_TO_PARAM (decrypted_message_action)); + } + + if (file) { + (*flags_p) |= (1 << 23); + store_ds_type_encrypted_file (file, TYPE_TO_PARAM (encrypted_file)); + } + + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_message_delete (struct tgl_state *TLS, struct tgl_message *M) { + clear_packet (); + out_int (CODE_binlog_message_delete); + out_long (M->id); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_msg_update (struct tgl_state *TLS, long long id) { + clear_packet (); + out_int (CODE_binlog_msg_update); + out_long (id); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_reset_authorization (struct tgl_state *TLS) { + clear_packet (); + out_int (CODE_binlog_reset_authorization); + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_encr_chat_exchange_new (struct tgl_state *TLS, struct tgl_secret_chat *E, long long *exchange_id, const void *key, int *state) { + clear_packet (); + + out_int (CODE_binlog_encr_chat_exchange_new); + out_int (tgl_get_peer_id (E->id)); + + int *flags_p = packet_ptr; + out_int (0); + + if (exchange_id) { + *flags_p |= (1 << 17); + out_long (*exchange_id); + } + + if (key) { + *flags_p |= (1 << 18); + out_ints ((void *)key, 64); + } + + if (state) { + *flags_p |= (1 << 19); + out_int (*state); + } + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); +} + +void bl_do_set_auth_key (struct tgl_state *TLS, int num, unsigned char *buf) { + int *ev = alloc_log_event (8 + 8 + 256); + ev[0] = CODE_binlog_auth_key; + ev[1] = num; + //*(long long *)(ev + 2) = fingerprint; + memcpy (ev + 2, buf, 256); + add_log_event (TLS, ev, 8 + 256); +} + +void bl_do_user_new (struct tgl_state *TLS, int id, long long *access_hash, const char *first_name, int first_name_len, const char *last_name, int last_name_len, const char *phone, int phone_len, const char *username, int username_len, struct tl_ds_photo *photo, const char *real_first_name, int real_first_name_len, const char *real_last_name, int real_last_name_len, struct tl_ds_user_profile_photo *profile_photo, int *last_read_in, int *last_read_out, struct tl_ds_bot_info *bot_info, int flags) { + tgl_peer_t *PP = tgl_peer_get (TLS, TGL_MK_USER (id)); + struct tgl_user *P = &PP->user; + + if (flags == TGL_FLAGS_UNCHANGED) { + flags = P->flags & 0xffff; + } + + clear_packet (); + out_int (CODE_binlog_user_new); + + int *flags_p = packet_ptr; + + assert (!(flags & 0xfffe0000)); + out_int (flags); + out_int (id); + + if (access_hash) { + if (!P || P->access_hash != *access_hash) { + out_long (*access_hash); + (*flags_p) |= (1 << 17); + } + } + + if (first_name) { + if (!P || !P->first_name || !P->last_name || mystreq1 (P->first_name, first_name, first_name_len) || mystreq1 (P->last_name, last_name, last_name_len)) { + out_cstring (first_name, first_name_len); + out_cstring (last_name, last_name_len); + + (*flags_p) |= (1 << 18); + } + } + + if (phone) { + if (!P || !P->phone || mystreq1 (P->phone, phone, phone_len)) { + out_cstring (phone, phone_len); + (*flags_p) |= (1 << 19); + } + } + + if (username) { + if (!P || !P->username || mystreq1 (P->username, username, username_len)) { + out_cstring (username, username_len); + (*flags_p) |= (1 << 20); + } + } + + if (photo) { + if (!P || !P->photo || P->photo->id != DS_LVAL (photo->id)) { + store_ds_type_photo (photo, TYPE_TO_PARAM (photo)); + (*flags_p) |= (1 << 21); + } + } + + if (real_first_name) { + assert (real_last_name); + if (!P || !P->real_first_name || !P->real_last_name || mystreq1 (P->real_first_name, real_first_name, real_first_name_len) || mystreq1 (P->real_last_name, real_last_name, real_last_name_len)) { + out_cstring (real_first_name, real_first_name_len); + out_cstring (real_last_name, real_last_name_len); + + (*flags_p) |= (1 << 22); + } + } + + if (profile_photo) { + if (!P || P->photo_id != DS_LVAL (profile_photo->photo_id)) { + store_ds_type_user_profile_photo (profile_photo, TYPE_TO_PARAM (user_profile_photo)); + (*flags_p) |= (1 << 23); + } + } + + if (last_read_in) { + if (!P || P->last_read_in < *last_read_in) { + out_int (*last_read_in); + (*flags_p) |= (1 << 24); + } + } + + if (last_read_out) { + if (!P || P->last_read_out < *last_read_out) { + out_int (*last_read_out); + (*flags_p) |= (1 << 25); + } + } + + if (bot_info) { + if (!P || !P->bot_info || P->bot_info->version != DS_LVAL (bot_info->version)) { + store_ds_type_bot_info (bot_info, TYPE_TO_PARAM (bot_info)); + (*flags_p) |= (1 << 26); + } + } + + if (((*flags_p) & 0xffff0000) || !P || (P->flags & 0xffff) != flags) { + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); + } +} + +void bl_do_chat_new (struct tgl_state *TLS, int id, const char *title, int title_len, int *user_num, int *date, int *version, struct tl_ds_vector *participants, struct tl_ds_chat_photo *chat_photo, struct tl_ds_photo *photo, int *admin, int *last_read_in, int *last_read_out, int flags) { + tgl_peer_t *PP = tgl_peer_get (TLS, TGL_MK_CHAT (id)); + struct tgl_chat *P = &PP->chat; + + if (flags == TGL_FLAGS_UNCHANGED) { + flags = P ? (P->flags & 0xffff) : 0; + } + + clear_packet (); + out_int (CODE_binlog_chat_new); + + int *flags_p = packet_ptr; + + assert (!(flags & 0xfffe0000)); + out_int (flags); + out_int (id); + + if (title) { + if (!P || !P->title || mystreq1 (P->title, title, title_len)) { + out_cstring (title, title_len); + (*flags_p) |= (1 << 17); + } + } + + if (user_num) { + if (!P || P->users_num != *user_num) { + out_int (*user_num); + (*flags_p) |= (1 << 18); + } + } + + if (date) { + if (!P || P->date != *date) { + out_int (*date); + (*flags_p) |= (1 << 19); + } + } + + if (version) { + assert (participants); + if (!P || *version != P->version) { + out_int (*version); + store_ds_type_vector (participants, TYPE_TO_PARAM_1 (vector, TYPE_TO_PARAM (chat_participant))); + (*flags_p) |= (1 << 20); + } + } + + if (chat_photo && chat_photo->photo_big) { + if (!P || DS_LVAL (chat_photo->photo_big->secret) != P->photo_big.secret) { + store_ds_type_chat_photo (chat_photo, TYPE_TO_PARAM (chat_photo)); + (*flags_p) |= (1 << 21); + } + } + + if (photo) { + if (!P || !P->photo || P->photo->id != DS_LVAL (photo->id)) { + store_ds_type_photo (photo, TYPE_TO_PARAM (photo)); + (*flags_p) |= (1 << 22); + } + } + + if (admin) { + if (!P || P->admin_id != *admin) { + out_int (*admin); + (*flags_p) |= (1 << 23); + } + } + + if (last_read_in) { + if (!P || P->last_read_in < *last_read_in) { + out_int (*last_read_in); + (*flags_p) |= (1 << 24); + } + } + + if (last_read_out) { + if (!P || P->last_read_out < *last_read_out) { + out_int (*last_read_out); + (*flags_p) |= (1 << 25); + } + } + + if (((*flags_p) & 0xffff0000) || !P || (P->flags & 0xffff) != flags) { + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); + } +} + +void bl_do_encr_chat_new (struct tgl_state *TLS, int id, long long *access_hash, int *date, int *admin, int *user_id, void *key, void *g_key, void *first_key_id, int *state, int *ttl, int *layer, int *in_seq_no, int *last_in_seq_no, int *out_seq_no, long long *key_fingerprint, int flags) { + tgl_peer_t *PP = tgl_peer_get (TLS, TGL_MK_ENCR_CHAT (id)); + struct tgl_secret_chat *P = PP ? &PP->encr_chat : NULL; + + if (flags == TGL_FLAGS_UNCHANGED) { + flags = P->flags & 0xffff; + } + + clear_packet (); + out_int (CODE_binlog_encr_chat_new); + + int *flags_p = packet_ptr; + + assert (!(flags & 0xfffe0000)); + out_int (flags); + out_int (id); + + if (access_hash) { + if (!P || P->access_hash != *access_hash) { + out_long (*access_hash); + (*flags_p) |= (1 << 17); + } + } + + if (date) { + if (!P || P->date != *date) { + out_int (*date); + (*flags_p) |= (1 << 18); + } + } + + if (admin) { + if (!P || P->admin_id != *admin) { + out_int (*admin); + (*flags_p) |= (1 << 19); + } + } + + if (user_id) { + if (!P || P->user_id != *user_id) { + out_int (*user_id); + (*flags_p) |= (1 << 20); + } + } + + if (key) { + if (!P || memcmp (P->key, key, 256)) { + out_ints (key, 64); + (*flags_p) |= (1 << 21); + } + } + + if (g_key) { + if (!P || !P->g_key || memcmp (P->g_key, g_key, 256)) { + out_ints (g_key, 64); + (*flags_p) |= (1 << 22); + } + } + + if (state) { + if (!P || (int)P->state != *state) { + out_int (*state); + (*flags_p) |= (1 << 23); + } + } + + if (ttl) { + if (!P || P->ttl != *ttl) { + out_int (*ttl); + (*flags_p) |= (1 << 24); + } + } + + if (layer) { + if (!P || P->layer != *layer) { + out_int (*layer); + (*flags_p) |= (1 << 25); + } + } + + if (in_seq_no || last_in_seq_no || out_seq_no) { + if (!P || (in_seq_no && P->in_seq_no != *in_seq_no) || + (out_seq_no && P->out_seq_no != *out_seq_no) || + (last_in_seq_no && P->last_in_seq_no != *last_in_seq_no)) { + + out_int (in_seq_no ? *in_seq_no : P ? P->in_seq_no : 0); + out_int (last_in_seq_no ? *last_in_seq_no : P ? P->last_in_seq_no : 0); + out_int (out_seq_no ? *out_seq_no : P ? P->out_seq_no : 0); + (*flags_p) |= (1 << 26); + } + } + + if (key_fingerprint) { + if (!P || P->key_fingerprint != *key_fingerprint) { + out_long (*key_fingerprint); + (*flags_p) |= (1 << 27); + } + } + + if (((*flags_p) & 0xffff0000) || !P || (P->flags & 0xffff) != flags) { + add_log_event (TLS, packet_buffer, 4 * (packet_ptr - packet_buffer)); + } +} |