tg/loop.c

895 lines
24 KiB
C
Raw Normal View History

2013-10-23 18:26:17 +04:00
/*
This file is part of telegram-cli.
2013-10-23 18:26:17 +04:00
Telegram-cli is free software: you can redistribute it and/or modify
2013-10-23 18:26:17 +04:00
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-cli is distributed in the hope that it will be useful,
2013-10-23 18:26:17 +04:00
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-cli. If not, see <http://www.gnu.org/licenses/>.
2013-10-23 18:26:17 +04:00
Copyright Vitaly Valtman 2013-2014
2013-10-23 18:26:17 +04:00
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#define _GNU_SOURCE
#define READLINE_CALLBACKS
2013-10-03 16:38:25 +04:00
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifdef READLINE_GNU
2013-10-03 16:38:25 +04:00
#include <readline/readline.h>
#include <readline/history.h>
#else
2013-11-11 15:34:45 +04:00
#include <readline/readline.h>
#include <readline/history.h>
#endif
2013-10-03 16:38:25 +04:00
2013-10-03 20:09:06 +04:00
#include <errno.h>
2013-10-12 00:52:20 +04:00
#include <poll.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
2014-09-21 13:07:06 +04:00
#include <sys/socket.h>
2013-10-12 00:52:20 +04:00
#include <fcntl.h>
2014-09-21 13:10:22 +04:00
#include <netinet/in.h>
2013-10-03 20:09:06 +04:00
2014-09-07 23:12:39 +04:00
#ifdef EVENT_V2
2014-08-15 18:57:43 +04:00
#include <event2/event.h>
2014-09-21 02:22:35 +04:00
#include <event2/bufferevent.h>
2014-09-07 23:12:39 +04:00
#else
#include <event.h>
#include "event-old.h"
#endif
2014-08-15 18:57:43 +04:00
2013-10-03 16:38:25 +04:00
#include "interface.h"
2013-10-12 00:52:20 +04:00
#include "telegram.h"
2013-11-04 21:34:27 +04:00
#include "loop.h"
2014-01-12 04:43:29 +04:00
#include "lua-tg.h"
2014-08-13 16:56:55 +04:00
#include "tgl.h"
2014-08-21 07:24:52 +04:00
#include "binlog.h"
2013-10-12 00:52:20 +04:00
2014-08-15 18:57:43 +04:00
int verbosity;
extern int readline_disabled;
2014-08-15 18:57:43 +04:00
2014-08-21 19:38:51 +04:00
int binlog_read;
2013-10-03 16:38:25 +04:00
extern char *default_username;
extern char *auth_token;
void set_default_username (const char *s);
2013-11-13 04:11:25 +04:00
extern int binlog_enabled;
2013-10-03 16:38:25 +04:00
2013-10-23 14:24:59 +04:00
extern int unknown_user_list_pos;
extern int unknown_user_list[];
2013-11-21 23:35:49 +04:00
int register_mode;
extern int safe_quit;
2014-08-20 20:36:45 +04:00
extern int sync_from_start;
2013-10-03 16:38:25 +04:00
2014-09-09 21:05:36 +04:00
extern int disable_output;
2014-09-18 17:57:23 +04:00
extern int reset_authorization;
2014-09-09 21:05:36 +04:00
2014-09-21 02:22:35 +04:00
extern int sfd;
2014-09-23 02:23:40 +04:00
extern int usfd;
2014-09-21 02:22:35 +04:00
void got_it (char *line, int len);
2014-08-22 15:45:16 +04:00
void write_state_file (void);
2014-08-15 18:57:43 +04:00
static char *line_buffer;
static int line_buffer_size;
static int line_buffer_pos;
static int delete_stdin_event;
2014-09-23 02:23:40 +04:00
extern volatile int sigterm_cnt;
2014-10-02 22:54:06 +04:00
extern char *start_command;
static void stdin_read_callback_all (int arg, short what, struct event *self) {
if (!readline_disabled) {
if (((long)arg) & 1) {
rl_callback_read_char ();
} else {
char *line = 0;
size_t len = 0;
assert (getline (&line, &len, stdin) >= 0);
got_it (line, strlen (line));
}
2014-08-15 18:57:43 +04:00
} else {
while (1) {
if (line_buffer_pos == line_buffer_size) {
line_buffer = realloc (line_buffer, line_buffer_size * 2 + 100);
assert (line_buffer);
line_buffer_size = line_buffer_size * 2 + 100;
assert (line_buffer);
}
int r = read (0, line_buffer + line_buffer_pos, line_buffer_size - line_buffer_pos);
//logprintf ("r = %d, size = %d, pos = %d, what = 0x%x, fd = %d\n", r, line_buffer_size, line_buffer_pos, (int)what, fd);
if (r < 0) {
perror ("read");
2014-10-09 21:38:45 +04:00
delete_stdin_event = 1;
break;
}
if (r == 0) {
//struct event *ev = event_base_get_running_event (tgl_state.ev_base);
//event_del (ev);
//event_del (self);
delete_stdin_event = 1;
break;
}
line_buffer_pos += r;
while (1) {
int p = 0;
while (p < line_buffer_pos && line_buffer[p] != '\n') { p ++; }
if (p < line_buffer_pos) {
if (((long)arg) & 1) {
line_buffer[p] = 0;
interpreter (line_buffer);
} else {
got_it (line_buffer, p + 1);
}
memmove (line_buffer, line_buffer + p + 1, line_buffer_pos - p - 1);
line_buffer_pos -= (p + 1);
} else {
break;
}
}
2014-09-23 18:39:28 +04:00
break;
}
2014-08-15 18:57:43 +04:00
}
}
static void stdin_read_callback_char (evutil_socket_t fd, short what, void *arg) {
stdin_read_callback_all (1, what, arg);
}
static void stdin_read_callback_line (evutil_socket_t fd, short what, void *arg) {
stdin_read_callback_all (2, what, arg);
}
2014-09-23 17:18:42 +04:00
struct event *term_ev = 0;
2013-10-12 00:52:20 +04:00
void net_loop (int flags, int (*is_end)(void)) {
delete_stdin_event = 0;
2014-08-27 19:12:45 +04:00
if (verbosity >= E_DEBUG) {
2014-08-15 02:16:01 +04:00
logprintf ("Starting netloop\n");
}
2014-08-15 18:57:43 +04:00
if (flags & 3) {
if (flags & 1) {
2014-09-23 17:18:42 +04:00
term_ev = event_new (tgl_state.ev_base, 0, EV_READ | EV_PERSIST, stdin_read_callback_char, 0);
} else {
2014-09-23 17:18:42 +04:00
term_ev = event_new (tgl_state.ev_base, 0, EV_READ | EV_PERSIST, stdin_read_callback_line, 0);
}
2014-09-23 17:18:42 +04:00
event_add (term_ev, 0);
2014-08-15 18:57:43 +04:00
}
2014-09-21 14:13:40 +04:00
int last_get_state = time (0);
2014-08-15 18:57:43 +04:00
while (!is_end || !is_end ()) {
event_base_loop (tgl_state.ev_base, EVLOOP_ONCE);
2013-10-03 16:38:25 +04:00
2014-09-23 17:18:42 +04:00
if (term_ev && delete_stdin_event) {
event_free (term_ev);
term_ev = 0;
}
2014-01-12 04:43:29 +04:00
#ifdef USE_LUA
lua_do_all ();
#endif
2014-08-22 15:45:16 +04:00
if (safe_quit && !tgl_state.active_queries) {
printf ("All done. Exit\n");
2014-09-23 02:23:40 +04:00
do_halt (0);
}
if (sigterm_cnt > 0) {
do_halt (0);
}
2014-09-21 14:13:40 +04:00
if (time (0) - last_get_state > 3600) {
tgl_do_lookup_state ();
last_get_state = time (0);
}
2014-08-22 15:45:16 +04:00
write_state_file ();
2014-08-22 00:35:27 +04:00
update_prompt ();
2013-10-23 14:24:59 +04:00
if (unknown_user_list_pos) {
2014-08-15 18:57:43 +04:00
int i;
for (i = 0; i < unknown_user_list_pos; i++) {
tgl_do_get_user_info (TGL_MK_USER (unknown_user_list[i]), 0, 0, 0);
}
2013-10-23 14:24:59 +04:00
unknown_user_list_pos = 0;
}
2013-10-03 20:09:06 +04:00
}
2014-08-15 18:57:43 +04:00
2014-09-23 17:18:42 +04:00
if (term_ev) {
event_free (term_ev);
2014-09-24 14:35:27 +04:00
term_ev = 0;
2014-08-15 18:57:43 +04:00
}
2014-08-15 02:16:01 +04:00
2014-08-27 19:12:45 +04:00
if (verbosity >= E_DEBUG) {
2014-08-15 02:16:01 +04:00
logprintf ("End of netloop\n");
}
2013-10-12 00:52:20 +04:00
}
char **_s;
size_t *_l;
int got_it_ok;
void got_it (char *line, int len) {
2013-12-18 19:21:49 +04:00
assert (len > 0);
line[-- len] = 0; // delete end of line
*_s = line;
*_l = len;
got_it_ok = 1;
}
int is_got_it (void) {
return got_it_ok;
}
int net_getline (char **s, size_t *l) {
fflush (stdout);
// rl_already_prompted = 1;
got_it_ok = 0;
_s = s;
_l = l;
// rl_callback_handler_install (0, got_it);
net_loop (2, is_got_it);
return 0;
}
2013-10-12 00:52:20 +04:00
int main_loop (void) {
2014-08-15 18:57:43 +04:00
net_loop (1, 0);
2013-10-03 20:09:06 +04:00
return 0;
2013-10-03 16:38:25 +04:00
}
2014-08-21 07:24:52 +04:00
struct tgl_dc *cur_a_dc;
2014-08-15 18:57:43 +04:00
int is_authorized (void) {
return tgl_authorized_dc (cur_a_dc);
}
int all_authorized (void) {
int i;
for (i = 0; i <= tgl_state.max_dc_num; i++) if (tgl_state.DC_list[i]) {
if (!tgl_authorized_dc (tgl_state.DC_list[i])) {
return 0;
}
}
return 1;
}
2014-08-15 18:57:43 +04:00
int config_got;
int got_config (void) {
return config_got;
}
void on_get_config (void *extra, int success) {
if (!success) {
logprintf ("Can not get config.\n");
2014-09-23 02:23:40 +04:00
do_halt (1);
2014-08-15 18:57:43 +04:00
}
config_got = 1;
2013-10-12 00:52:20 +04:00
}
2014-08-15 18:57:43 +04:00
int should_register;
char *hash;
void sign_in_callback (void *extra, int success, int registered, const char *mhash) {
if (!success) {
logprintf ("Can not send code\n");
2014-09-23 02:23:40 +04:00
do_halt (1);
2014-08-15 18:57:43 +04:00
}
should_register = !registered;
hash = strdup (mhash);
assert (hash);
2014-08-15 18:57:43 +04:00
}
int signed_in_ok;
void sign_in_result (void *extra, int success, struct tgl_user *U) {
if (!success) {
logprintf ("Can not login\n");
2014-09-23 02:23:40 +04:00
do_halt (1);
2014-08-15 18:57:43 +04:00
}
signed_in_ok = 1;
}
2014-08-15 18:57:43 +04:00
int signed_in (void) {
return signed_in_ok;
}
int sent_code (void) {
return hash != 0;
2013-11-04 21:34:27 +04:00
}
2014-08-15 18:57:43 +04:00
int dc_signed_in (void) {
return tgl_signed_dc (cur_a_dc);
}
void export_auth_callback (void *DC, int success) {
if (!success) {
logprintf ("Can not export auth\n");
2014-09-23 02:23:40 +04:00
do_halt (1);
2014-08-15 18:57:43 +04:00
}
}
2014-08-15 02:16:01 +04:00
int d_got_ok;
void get_difference_callback (void *extra, int success) {
assert (success);
d_got_ok = 1;
}
int dgot (void) {
return d_got_ok;
}
2014-08-15 18:57:43 +04:00
int zero[512];
int readline_active;
int new_dc_num;
int wait_dialog_list;
2014-08-15 02:16:01 +04:00
extern struct tgl_update_callback upd_cb;
2014-08-21 07:24:52 +04:00
#define DC_SERIALIZED_MAGIC 0x868aa81d
2014-08-22 15:45:16 +04:00
#define STATE_FILE_MAGIC 0x28949a93
#define SECRET_CHAT_FILE_MAGIC 0x37a1988a
2014-08-22 15:45:16 +04:00
char *get_auth_key_filename (void);
char *get_state_filename (void);
char *get_secret_chat_filename (void);
2014-08-22 15:45:16 +04:00
void read_state_file (void) {
if (binlog_enabled) { return; }
int state_file_fd = open (get_state_filename (), O_CREAT | O_RDWR, 0600);
if (state_file_fd < 0) {
return;
}
int version, magic;
if (read (state_file_fd, &magic, 4) < 4) { close (state_file_fd); return; }
if (magic != (int)STATE_FILE_MAGIC) { close (state_file_fd); return; }
if (read (state_file_fd, &version, 4) < 4) { close (state_file_fd); return; }
assert (version >= 0);
int x[4];
if (read (state_file_fd, x, 16) < 16) {
close (state_file_fd);
return;
}
int pts = x[0];
int qts = x[1];
int seq = x[2];
int date = x[3];
close (state_file_fd);
bl_do_set_seq (seq);
bl_do_set_pts (pts);
bl_do_set_qts (qts);
bl_do_set_date (date);
}
void write_state_file (void) {
if (binlog_enabled) { return; }
static int wseq;
static int wpts;
static int wqts;
static int wdate;
if (wseq >= tgl_state.seq && wpts >= tgl_state.pts && wqts >= tgl_state.qts && wdate >= tgl_state.date) { return; }
wseq = tgl_state.seq; wpts = tgl_state.pts; wqts = tgl_state.qts; wdate = tgl_state.date;
int state_file_fd = open (get_state_filename (), O_CREAT | O_RDWR, 0600);
if (state_file_fd < 0) {
logprintf ("Can not write state file '%s': %m\n", get_state_filename ());
2014-09-23 02:23:40 +04:00
do_halt (1);
2014-08-22 15:45:16 +04:00
}
int x[6];
x[0] = STATE_FILE_MAGIC;
x[1] = 0;
x[2] = wpts;
x[3] = wqts;
x[4] = wseq;
x[5] = wdate;
assert (write (state_file_fd, x, 24) == 24);
close (state_file_fd);
}
2014-08-21 07:24:52 +04:00
void write_dc (struct tgl_dc *DC, void *extra) {
int auth_file_fd = *(int *)extra;
if (!DC) {
int x = 0;
assert (write (auth_file_fd, &x, 4) == 4);
return;
} else {
int x = 1;
assert (write (auth_file_fd, &x, 4) == 4);
}
assert (DC->has_auth);
assert (write (auth_file_fd, &DC->port, 4) == 4);
int l = strlen (DC->ip);
assert (write (auth_file_fd, &l, 4) == 4);
assert (write (auth_file_fd, DC->ip, l) == l);
assert (write (auth_file_fd, &DC->auth_key_id, 8) == 8);
assert (write (auth_file_fd, DC->auth_key, 256) == 256);
}
void write_auth_file (void) {
if (binlog_enabled) { return; }
int auth_file_fd = open (get_auth_key_filename (), O_CREAT | O_RDWR, 0600);
assert (auth_file_fd >= 0);
int x = DC_SERIALIZED_MAGIC;
assert (write (auth_file_fd, &x, 4) == 4);
assert (write (auth_file_fd, &tgl_state.max_dc_num, 4) == 4);
assert (write (auth_file_fd, &tgl_state.dc_working_num, 4) == 4);
tgl_dc_iterator_ex (write_dc, &auth_file_fd);
assert (write (auth_file_fd, &tgl_state.our_id, 4) == 4);
close (auth_file_fd);
}
void write_secret_chat (tgl_peer_t *_P, void *extra) {
struct tgl_secret_chat *P = (void *)_P;
if (tgl_get_peer_type (P->id) != TGL_PEER_ENCR_CHAT) { return; }
if (P->state != sc_ok) { return; }
int *a = extra;
int fd = a[0];
a[1] ++;
int id = tgl_get_peer_id (P->id);
assert (write (fd, &id, 4) == 4);
//assert (write (fd, &P->flags, 4) == 4);
int l = strlen (P->print_name);
assert (write (fd, &l, 4) == 4);
assert (write (fd, P->print_name, l) == l);
assert (write (fd, &P->user_id, 4) == 4);
assert (write (fd, &P->admin_id, 4) == 4);
assert (write (fd, &P->date, 4) == 4);
assert (write (fd, &P->ttl, 4) == 4);
assert (write (fd, &P->layer, 4) == 4);
assert (write (fd, &P->access_hash, 8) == 8);
assert (write (fd, &P->state, 4) == 4);
assert (write (fd, &P->key_fingerprint, 8) == 8);
assert (write (fd, &P->key, 256) == 256);
assert (write (fd, &P->in_seq_no, 4) == 4);
assert (write (fd, &P->last_in_seq_no, 4) == 4);
assert (write (fd, &P->out_seq_no, 4) == 4);
}
void write_secret_chat_file (void) {
if (binlog_enabled) { return; }
int secret_chat_fd = open (get_secret_chat_filename (), O_CREAT | O_RDWR, 0600);
assert (secret_chat_fd >= 0);
int x = SECRET_CHAT_FILE_MAGIC;
assert (write (secret_chat_fd, &x, 4) == 4);
x = 1;
assert (write (secret_chat_fd, &x, 4) == 4); // version
assert (write (secret_chat_fd, &x, 4) == 4); // num
int y[2];
y[0] = secret_chat_fd;
y[1] = 0;
tgl_peer_iterator_ex (write_secret_chat, y);
lseek (secret_chat_fd, 8, SEEK_SET);
assert (write (secret_chat_fd, &y[1], 4) == 4);
close (secret_chat_fd);
}
2014-08-21 07:24:52 +04:00
void read_dc (int auth_file_fd, int id, unsigned ver) {
int port = 0;
assert (read (auth_file_fd, &port, 4) == 4);
int l = 0;
assert (read (auth_file_fd, &l, 4) == 4);
assert (l >= 0 && l < 100);
char ip[100];
assert (read (auth_file_fd, ip, l) == l);
ip[l] = 0;
long long auth_key_id;
static unsigned char auth_key[256];
assert (read (auth_file_fd, &auth_key_id, 8) == 8);
assert (read (auth_file_fd, auth_key, 256) == 256);
//bl_do_add_dc (id, ip, l, port, auth_key_id, auth_key);
bl_do_dc_option (id, 2, "DC", l, ip, port);
bl_do_set_auth_key_id (id, auth_key);
bl_do_dc_signed (id);
}
void empty_auth_file (void) {
2014-09-27 21:58:28 +04:00
if (tgl_state.test_mode) {
bl_do_dc_option (1, 0, "", strlen (TG_SERVER_TEST_1), TG_SERVER_TEST_1, 443);
bl_do_dc_option (2, 0, "", strlen (TG_SERVER_TEST_2), TG_SERVER_TEST_2, 443);
bl_do_dc_option (3, 0, "", strlen (TG_SERVER_TEST_3), TG_SERVER_TEST_3, 443);
bl_do_set_working_dc (2);
} else {
bl_do_dc_option (1, 0, "", strlen (TG_SERVER_1), TG_SERVER_1, 443);
bl_do_dc_option (2, 0, "", strlen (TG_SERVER_2), TG_SERVER_2, 443);
bl_do_dc_option (3, 0, "", strlen (TG_SERVER_3), TG_SERVER_3, 443);
bl_do_dc_option (4, 0, "", strlen (TG_SERVER_4), TG_SERVER_4, 443);
bl_do_dc_option (5, 0, "", strlen (TG_SERVER_5), TG_SERVER_5, 443);
bl_do_set_working_dc (2);
}
2014-08-21 07:24:52 +04:00
}
int need_dc_list_update;
void read_auth_file (void) {
if (binlog_enabled) { return; }
int auth_file_fd = open (get_auth_key_filename (), O_CREAT | O_RDWR, 0600);
if (auth_file_fd < 0) {
empty_auth_file ();
return;
}
assert (auth_file_fd >= 0);
unsigned x;
unsigned m;
if (read (auth_file_fd, &m, 4) < 4 || (m != DC_SERIALIZED_MAGIC)) {
close (auth_file_fd);
empty_auth_file ();
return;
}
assert (read (auth_file_fd, &x, 4) == 4);
assert (x > 0);
int dc_working_num;
assert (read (auth_file_fd, &dc_working_num, 4) == 4);
int i;
for (i = 0; i <= (int)x; i++) {
int y;
assert (read (auth_file_fd, &y, 4) == 4);
if (y) {
read_dc (auth_file_fd, i, m);
}
}
bl_do_set_working_dc (dc_working_num);
int our_id;
int l = read (auth_file_fd, &our_id, 4);
if (l < 4) {
assert (!l);
}
if (our_id) {
bl_do_set_our_id (our_id);
}
close (auth_file_fd);
}
void read_secret_chat (int fd, int v) {
int id, l, user_id, admin_id, date, ttl, layer, state;
long long access_hash, key_fingerprint;
static char s[1000];
static unsigned char key[256];
assert (read (fd, &id, 4) == 4);
//assert (read (fd, &flags, 4) == 4);
assert (read (fd, &l, 4) == 4);
assert (l > 0 && l < 1000);
assert (read (fd, s, l) == l);
assert (read (fd, &user_id, 4) == 4);
assert (read (fd, &admin_id, 4) == 4);
assert (read (fd, &date, 4) == 4);
assert (read (fd, &ttl, 4) == 4);
assert (read (fd, &layer, 4) == 4);
assert (read (fd, &access_hash, 8) == 8);
assert (read (fd, &state, 4) == 4);
assert (read (fd, &key_fingerprint, 8) == 8);
assert (read (fd, &key, 256) == 256);
int in_seq_no = 0, out_seq_no = 0, last_in_seq_no = 0;
if (v >= 1) {
assert (read (fd, &in_seq_no, 4) == 4);
assert (read (fd, &last_in_seq_no, 4) == 4);
assert (read (fd, &out_seq_no, 4) == 4);
}
bl_do_encr_chat_create (id, user_id, admin_id, s, l);
struct tgl_secret_chat *P = (void *)tgl_peer_get (TGL_MK_ENCR_CHAT (id));
assert (P && (P->flags & FLAG_CREATED));
bl_do_encr_chat_set_date (P, date);
bl_do_encr_chat_set_ttl (P, ttl);
bl_do_encr_chat_set_layer (P, layer);
bl_do_encr_chat_set_access_hash (P, access_hash);
bl_do_encr_chat_set_state (P, state);
bl_do_encr_chat_set_key (P, key, key_fingerprint);
if (v >= 1) {
bl_do_encr_chat_set_seq (P, in_seq_no, last_in_seq_no, out_seq_no);
}
}
void read_secret_chat_file (void) {
if (binlog_enabled) { return; }
2014-08-31 21:24:39 +04:00
int secret_chat_fd = open (get_secret_chat_filename (), O_RDWR, 0600);
2014-08-31 21:15:49 +04:00
if (secret_chat_fd < 0) { return; }
//assert (secret_chat_fd >= 0);
int x;
2014-08-31 21:24:39 +04:00
if (read (secret_chat_fd, &x, 4) < 4) { close (secret_chat_fd); return; }
if (x != SECRET_CHAT_FILE_MAGIC) { close (secret_chat_fd); return; }
int v = 0;
assert (read (secret_chat_fd, &v, 4) == 4);
assert (v == 0 || v == 1); // version
assert (read (secret_chat_fd, &x, 4) == 4);
assert (x >= 0);
while (x --> 0) {
read_secret_chat (secret_chat_fd, v);
}
close (secret_chat_fd);
}
2014-08-25 04:11:41 +04:00
void dlist_cb (void *callback_extra, int success, int size, tgl_peer_id_t peers[], int last_msg_id[], int unread_count[]) {
d_got_ok = 1;
}
2014-09-21 02:22:35 +04:00
static void read_incoming (struct bufferevent *bev, void *_arg) {
vlogprintf (E_WARNING, "Read from incoming connection\n");
struct in_ev *ev = _arg;
assert (ev->bev == bev);
ev->in_buf_pos += bufferevent_read (bev, ev->in_buf + ev->in_buf_pos, 4096 - ev->in_buf_pos);
while (1) {
int pos = 0;
int ok = 0;
while (pos < ev->in_buf_pos) {
if (ev->in_buf[pos] == '\n') {
if (!ev->error) {
ev->in_buf[pos] = 0;
interpreter_ex (ev->in_buf, ev);
} else {
ev->error = 0;
}
ok = 1;
ev->in_buf_pos -= (pos + 1);
memmove (ev->in_buf, ev->in_buf + pos + 1, ev->in_buf_pos);
pos = 0;
} else {
pos ++;
}
}
if (ok) {
ev->in_buf_pos += bufferevent_read (bev, ev->in_buf + ev->in_buf_pos, 4096 - ev->in_buf_pos);
} else {
if (ev->in_buf_pos == 4096) {
ev->error = 1;
}
break;
}
}
}
static void event_incoming (struct bufferevent *bev, short what, void *_arg) {
struct in_ev *ev = _arg;
if (what & (BEV_EVENT_EOF | BEV_EVENT_ERROR)) {
vlogprintf (E_WARNING, "Closing incoming connection\n");
2014-09-21 13:18:12 +04:00
assert (ev->fd >= 0);
close (ev->fd);
2014-09-21 02:22:35 +04:00
bufferevent_free (bev);
ev->bev = 0;
2014-09-21 13:41:10 +04:00
if (!--ev->refcnt) { free (ev); }
2014-09-21 02:22:35 +04:00
}
}
static void accept_incoming (evutil_socket_t efd, short what, void *arg) {
2014-09-21 13:41:10 +04:00
vlogprintf (E_WARNING, "Accepting incoming connection\n");
2014-09-21 02:22:35 +04:00
unsigned clilen;
struct sockaddr_in cli_addr;
2014-09-23 02:23:40 +04:00
int fd = accept (efd, (struct sockaddr *)&cli_addr, &clilen);
2014-09-21 02:22:35 +04:00
assert (fd >= 0);
2014-09-21 13:10:22 +04:00
struct bufferevent *bev = bufferevent_socket_new (tgl_state.ev_base, fd, 0);
2014-09-21 02:22:35 +04:00
struct in_ev *e = malloc (sizeof (*e));
e->bev = bev;
e->refcnt = 1;
e->in_buf_pos = 0;
e->error = 0;
2014-09-21 13:18:12 +04:00
e->fd = fd;
2014-09-21 02:22:35 +04:00
bufferevent_setcb (bev, read_incoming, 0, event_incoming, e);
bufferevent_enable(bev, EV_READ|EV_WRITE);
}
2013-10-03 16:38:25 +04:00
int loop (void) {
2014-08-14 22:03:33 +04:00
//on_start ();
2014-08-15 02:16:01 +04:00
tgl_set_callback (&upd_cb);
//tgl_state.temp_key_expire_time = 60;
2014-08-14 22:03:33 +04:00
tgl_init ();
2014-08-21 07:24:52 +04:00
if (binlog_enabled) {
double t = tglt_get_double_time ();
2014-08-27 19:12:45 +04:00
if (verbosity >= E_DEBUG) {
logprintf ("replay log start\n");
}
2014-08-21 07:24:52 +04:00
tgl_replay_log ();
2014-08-27 19:12:45 +04:00
if (verbosity >= E_DEBUG) {
logprintf ("replay log end in %lf seconds\n", tglt_get_double_time () - t);
}
2014-08-21 07:24:52 +04:00
tgl_reopen_binlog_for_writing ();
} else {
read_auth_file ();
2014-08-22 15:45:16 +04:00
read_state_file ();
read_secret_chat_file ();
2014-08-21 07:24:52 +04:00
}
2014-09-21 02:22:35 +04:00
2014-08-21 19:38:51 +04:00
binlog_read = 1;
2014-08-14 22:03:33 +04:00
#ifdef USE_LUA
lua_binlog_end ();
#endif
2014-09-21 02:22:35 +04:00
if (sfd >= 0) {
2014-09-21 13:41:10 +04:00
struct event *ev = event_new (tgl_state.ev_base, sfd, EV_READ | EV_PERSIST, accept_incoming, 0);
2014-09-21 02:22:35 +04:00
event_add (ev, 0);
}
2014-09-23 02:23:40 +04:00
if (usfd >= 0) {
struct event *ev = event_new (tgl_state.ev_base, usfd, EV_READ | EV_PERSIST, accept_incoming, 0);
event_add (ev, 0);
}
update_prompt ();
2014-09-18 17:57:23 +04:00
if (reset_authorization) {
tgl_peer_t *P = tgl_peer_get (TGL_MK_USER (tgl_state.our_id));
if (P && P->user.phone && reset_authorization == 1) {
set_default_username (P->user.phone);
}
bl_do_reset_authorization ();
}
net_loop (0, all_authorized);
2014-08-15 18:57:43 +04:00
int i;
2014-08-15 18:57:43 +04:00
for (i = 0; i <= tgl_state.max_dc_num; i++) if (tgl_state.DC_list[i] && !tgl_authorized_dc (tgl_state.DC_list[i])) {
assert (0);
2013-10-12 00:52:20 +04:00
}
2014-08-15 18:57:43 +04:00
if (!tgl_signed_dc (tgl_state.DC_working)) {
2014-09-09 21:05:36 +04:00
if (disable_output) {
fprintf (stderr, "Can not login without output\n");
2014-09-23 02:23:40 +04:00
do_halt (1);
2014-09-09 21:05:36 +04:00
}
2013-10-12 00:52:20 +04:00
if (!default_username) {
size_t size = 0;
char *user = 0;
if (!user) {
2013-10-12 00:52:20 +04:00
printf ("Telephone number (with '+' sign): ");
if (net_getline (&user, &size) == -1) {
2013-10-12 00:52:20 +04:00
perror ("getline()");
2014-09-23 02:23:40 +04:00
do_halt (1);
2013-10-12 00:52:20 +04:00
}
set_default_username (user);
}
2013-10-03 20:10:53 +04:00
}
2014-08-15 18:57:43 +04:00
tgl_do_send_code (default_username, sign_in_callback, 0);
net_loop (0, sent_code);
2014-08-27 19:12:45 +04:00
if (verbosity >= E_DEBUG) {
logprintf ("%s\n", should_register ? "phone not registered" : "phone registered");
}
2014-08-15 18:57:43 +04:00
if (!should_register) {
char *code = 0;
size_t size = 0;
2014-02-24 15:27:30 +04:00
printf ("Code from sms (if you did not receive an SMS and want to be called, type \"call\"): ");
while (1) {
if (net_getline (&code, &size) == -1) {
perror ("getline()");
2014-09-23 02:23:40 +04:00
do_halt (1);
}
2014-02-24 15:27:30 +04:00
if (!strcmp (code, "call")) {
2014-02-25 23:31:27 +03:00
printf ("You typed \"call\", switching to phone system.\n");
2014-08-15 18:57:43 +04:00
tgl_do_phone_call (default_username, hash, 0, 0);
printf ("Calling you! Code: ");
continue;
}
2014-08-15 18:57:43 +04:00
if (tgl_do_send_code_result (default_username, hash, code, sign_in_result, 0) >= 0) {
break;
}
printf ("Invalid code. Try again: ");
2014-08-15 18:57:43 +04:00
free (code);
}
} else {
printf ("User is not registered. Do you want to register? [Y/n] ");
char *code;
size_t size;
if (net_getline (&code, &size) == -1) {
2013-10-12 00:52:20 +04:00
perror ("getline()");
2014-09-23 02:23:40 +04:00
do_halt (1);
2013-10-12 00:52:20 +04:00
}
2013-11-01 23:57:57 +04:00
if (!*code || *code == 'y' || *code == 'Y') {
printf ("Ok, starting registartion.\n");
} else {
printf ("Then try again\n");
2014-09-23 02:23:40 +04:00
do_halt (1);
}
char *first_name;
printf ("First name: ");
if (net_getline (&first_name, &size) == -1) {
perror ("getline()");
2014-09-23 02:23:40 +04:00
do_halt (1);
2013-10-12 00:52:20 +04:00
}
char *last_name;
printf ("Last name: ");
if (net_getline (&last_name, &size) == -1) {
perror ("getline()");
2014-09-23 02:23:40 +04:00
do_halt (1);
}
2014-02-24 15:27:30 +04:00
printf ("Code from sms (if you did not receive an SMS and want to be called, type \"call\"): ");
while (1) {
if (net_getline (&code, &size) == -1) {
perror ("getline()");
2014-09-23 02:23:40 +04:00
do_halt (1);
}
2014-02-24 15:27:30 +04:00
if (!strcmp (code, "call")) {
2014-02-25 23:31:27 +03:00
printf ("You typed \"call\", switching to phone system.\n");
2014-08-15 18:57:43 +04:00
tgl_do_phone_call (default_username, hash, 0, 0);
printf ("Calling you! Code: ");
continue;
}
2014-08-15 18:57:43 +04:00
if (tgl_do_send_code_result_auth (default_username, hash, code, first_name, last_name, sign_in_result, 0) >= 0) {
break;
}
printf ("Invalid code. Try again: ");
2014-08-15 18:57:43 +04:00
free (code);
}
2013-10-12 00:52:20 +04:00
}
2014-08-15 18:57:43 +04:00
net_loop (0, signed_in);
2014-08-15 02:16:01 +04:00
//bl_do_dc_signed (tgl_state.DC_working);
2013-10-03 16:38:25 +04:00
}
2013-10-12 00:52:20 +04:00
2014-08-15 18:57:43 +04:00
for (i = 0; i <= tgl_state.max_dc_num; i++) if (tgl_state.DC_list[i] && !tgl_signed_dc (tgl_state.DC_list[i])) {
tgl_do_export_auth (i, export_auth_callback, (void*)(long)tgl_state.DC_list[i]);
2014-08-21 07:24:52 +04:00
cur_a_dc = tgl_state.DC_list[i];
2014-08-15 18:57:43 +04:00
net_loop (0, dc_signed_in);
2014-08-21 07:24:52 +04:00
assert (tgl_signed_dc (tgl_state.DC_list[i]));
}
2014-08-21 07:24:52 +04:00
write_auth_file ();
2013-10-12 00:52:20 +04:00
fflush (stdout);
fflush (stderr);
2013-10-03 16:38:25 +04:00
2014-08-15 18:57:43 +04:00
//read_state_file ();
//read_secret_chat_file ();
set_interface_callbacks ();
tglm_send_all_unsent ();
2014-08-20 20:36:45 +04:00
tgl_do_get_difference (sync_from_start, get_difference_callback, 0);
2013-11-04 21:34:27 +04:00
net_loop (0, dgot);
2014-08-21 20:28:01 +04:00
assert (!(tgl_state.locks & TGL_LOCK_DIFF));
tgl_state.started = 1;
2014-08-25 04:11:41 +04:00
if (wait_dialog_list) {
d_got_ok = 0;
tgl_do_get_dialog_list (dlist_cb, 0);
net_loop (0, dgot);
}
2014-01-12 04:43:29 +04:00
#ifdef USE_LUA
lua_diff_end ();
#endif
2014-10-02 22:54:06 +04:00
if (start_command) {
safe_quit = 1;
while (*start_command) {
char *start = start_command;
while (*start_command && *start_command != '\n') {
start_command ++;
}
if (*start_command) {
*start_command = 0;
start_command ++;
}
interpreter_ex (start, 0);
}
}
2014-01-12 04:43:29 +04:00
2014-08-15 02:16:01 +04:00
/*tgl_do_get_dialog_list (get_dialogs_callback, 0);
if (wait_dialog_list) {
dialog_list_got = 0;
net_loop (0, dlgot);
2014-08-15 02:16:01 +04:00
}*/
2013-10-03 16:38:25 +04:00
return main_loop ();
}