tg/interface.c

374 lines
8.6 KiB
C
Raw Normal View History

2013-10-03 16:38:25 +04:00
#define _GNU_SOURCE
#include <readline/readline.h>
#include <readline/history.h>
#include <assert.h>
#include <stdio.h>
2013-10-12 00:52:20 +04:00
#include <stdarg.h>
2013-10-03 16:38:25 +04:00
#include <stdlib.h>
#include <string.h>
#include "include.h"
2013-10-12 00:52:20 +04:00
#include "queries.h"
#include "interface.h"
#include "telegram.h"
#include "structures.h"
#include "mtproto-common.h"
2013-10-03 16:38:25 +04:00
char *default_prompt = ">";
char *get_default_prompt (void) {
return default_prompt;
}
char *complete_none (const char *text UU, int state UU) {
return 0;
}
char *commands[] = {
"help",
"msg",
2013-10-12 00:52:20 +04:00
"contact_list",
"stats",
"history",
"dialog_list",
2013-10-03 16:38:25 +04:00
0 };
int commands_flags[] = {
070,
072,
07,
07,
072,
07,
2013-10-03 16:38:25 +04:00
};
2013-10-03 20:09:06 +04:00
2013-10-03 16:38:25 +04:00
char *a = 0;
char *user_list[MAX_USER_NUM + 1];
2013-10-03 16:38:25 +04:00
char **chat_list = &a;
int init_token (char **q) {
char *r = *q;
while (*r == ' ') { r ++; }
if (!*r) { return 0; }
2013-10-03 20:09:06 +04:00
*q = r;
2013-10-03 16:38:25 +04:00
return 1;
}
char *get_token (char **q, int *l) {
char *r = *q;
while (*r == ' ') { r ++; }
if (!*r) {
2013-10-03 20:09:06 +04:00
*q = r;
2013-10-03 16:38:25 +04:00
*l = 0;
return 0;
}
int neg = 0;
char *s = r;
while (*r && (*r != ' ' || neg)) {
if (*r == '\\') {
neg = 1 - neg;
} else {
neg = 0;
}
2013-10-03 20:09:06 +04:00
r++;
2013-10-03 16:38:25 +04:00
}
2013-10-03 20:09:06 +04:00
*q = r;
2013-10-03 16:38:25 +04:00
*l = r - s;
return s;
}
int get_complete_mode (void) {
char *q = rl_line_buffer;
if (!init_token (&q)) { return 0; }
int l = 0;
char *r = get_token (&q, &l);
if (!*q) { return 0; }
char **command = commands;
int n = 0;
int flags = -1;
while (*command) {
2013-10-03 20:09:06 +04:00
if (!strncmp (r, *command, l)) {
2013-10-03 16:38:25 +04:00
flags = commands_flags[n];
break;
}
n ++;
command ++;
}
if (flags == -1) {
return -1;
}
int s = 0;
while (1) {
get_token (&q, &l);
if (!*q) { return flags ? flags & 7 : 7; }
2013-10-03 16:38:25 +04:00
s ++;
if (s <= 4) { flags >>= 3; }
}
}
extern int user_num;
extern int chat_num;
extern union user_chat *Peers[];
int complete_user_list (int index, const char *text, int len, char **R) {
index ++;
while (index < user_num + chat_num && (!Peers[index]->print_name || strncmp (Peers[index]->print_name, text, len) || Peers[index]->id < 0)) {
index ++;
}
if (index < user_num + chat_num) {
*R = strdup (Peers[index]->print_name);
return index;
} else {
return -1;
}
}
int complete_user_chat_list (int index, const char *text, int len, char **R) {
index ++;
while (index < user_num + chat_num && (!Peers[index]->print_name || strncmp (Peers[index]->print_name, text, len))) {
index ++;
}
if (index < user_num + chat_num) {
*R = strdup (Peers[index]->print_name);
return index;
} else {
return -1;
}
}
2013-10-03 16:38:25 +04:00
int complete_string_list (char **list, int index, const char *text, int len, char **R) {
index ++;
2013-10-03 20:09:06 +04:00
while (list[index] && strncmp (list[index], text, len)) {
2013-10-03 16:38:25 +04:00
index ++;
}
2013-10-03 20:09:06 +04:00
if (list[index]) {
2013-10-03 16:38:25 +04:00
*R = strdup (list[index]);
return index;
} else {
*R = 0;
return -1;
}
}
char *command_generator (const char *text, int state) {
static int len, index, mode;
2013-10-03 20:09:06 +04:00
2013-10-03 16:38:25 +04:00
if (!state) {
len = strlen (text);
index = -1;
rl_line_buffer[rl_point] = '\0'; /* the effect should be such
2013-10-03 20:10:53 +04:00
* that the cursor position
* is at the end of line for
* the auto completion regex
* above (note the $ at end)
*/
2013-10-03 16:38:25 +04:00
mode = get_complete_mode ();
} else {
if (index == -1) { return 0; }
}
if (mode == -1) { return 0; }
char *R = 0;
switch (mode & 7) {
case 0:
index = complete_string_list (commands, index, text, len, &R);
return R;
case 1:
index = complete_user_list (index, text, len, &R);
2013-10-03 16:38:25 +04:00
return R;
case 2:
index = complete_user_chat_list (index, text, len, &R);
2013-10-03 16:38:25 +04:00
return R;
case 3:
return rl_filename_completion_function(text,state);
default:
return 0;
}
}
char **complete_text (char *text, int start UU, int end UU) {
return (char **) rl_completion_matches (text, command_generator);
}
void interpreter (char *line UU) {
if (line && *line) {
add_history (line);
}
2013-10-12 00:52:20 +04:00
if (!memcmp (line, "contact_list", 12)) {
do_update_contact_list ();
} else if (!memcmp (line, "dialog_list", 11)) {
do_get_dialog_list ();
} else if (!memcmp (line, "stats", 5)) {
static char stat_buf[1 << 15];
print_stat (stat_buf, (1 << 15) - 1);
printf ("%s\n", stat_buf);
2013-10-16 23:19:39 +04:00
} else if (!memcmp (line, "msg ", 4)) {
char *q = line + 4;
int len;
char *text = get_token (&q, &len);
int index = 0;
while (index < user_num + chat_num && (!Peers[index]->print_name || strncmp (Peers[index]->print_name, text, len))) {
2013-10-16 23:19:39 +04:00
index ++;
}
while (*q && (*q == ' ' || *q == '\t')) { q ++; }
if (*q && index < user_num + chat_num) {
do_send_message (Peers[index], q);
}
} else if (!memcmp (line, "history", 7)) {
char *q = line + 7;
int len;
char *text = get_token (&q, &len);
int index = 0;
2013-10-18 20:05:55 +04:00
while (index < user_num + chat_num && (!Peers[index]->print_name || strncmp (Peers[index]->print_name, text, len))) {
index ++;
}
if (index < user_num + chat_num) {
char *text = get_token (&q, &len);
int limit = 40;
if (text) {
limit = atoi (text);
if (limit <= 0 || limit >= 1000000) {
limit = 40;
}
}
do_get_history (Peers[index], limit);
}
2013-10-12 00:52:20 +04:00
}
}
int readline_active;
2013-10-12 00:52:20 +04:00
void rprintf (const char *format, ...) {
int saved_point = 0;
char *saved_line = 0;
if (readline_active) {
saved_point = rl_point;
saved_line = rl_copy_text(0, rl_end);
rl_save_prompt();
rl_replace_line("", 0);
rl_redisplay();
}
2013-10-12 00:52:20 +04:00
va_list ap;
va_start (ap, format);
vfprintf (stdout, format, ap);
va_end (ap);
2013-10-12 00:52:20 +04:00
if (readline_active) {
rl_restore_prompt();
rl_replace_line(saved_line, 0);
rl_point = saved_point;
rl_redisplay();
free(saved_line);
}
}
void hexdump (int *in_ptr, int *in_end) {
int saved_point = 0;
char *saved_line = 0;
if (readline_active) {
saved_point = rl_point;
saved_line = rl_copy_text(0, rl_end);
rl_save_prompt();
rl_replace_line("", 0);
rl_redisplay();
}
int *ptr = in_ptr;
while (ptr < in_end) { fprintf (stdout, " %08x", *(ptr ++)); }
fprintf (stdout, "\n");
if (readline_active) {
rl_restore_prompt();
rl_replace_line(saved_line, 0);
rl_point = saved_point;
rl_redisplay();
free(saved_line);
}
}
void logprintf (const char *format, ...) {
int saved_point = 0;
char *saved_line = 0;
if (readline_active) {
saved_point = rl_point;
saved_line = rl_copy_text(0, rl_end);
rl_save_prompt();
rl_replace_line("", 0);
rl_redisplay();
}
printf (COLOR_GREY " *** ");
2013-10-12 00:52:20 +04:00
va_list ap;
va_start (ap, format);
vfprintf (stdout, format, ap);
va_end (ap);
printf (COLOR_NORMAL);
2013-10-12 00:52:20 +04:00
if (readline_active) {
rl_restore_prompt();
rl_replace_line(saved_line, 0);
rl_point = saved_point;
rl_redisplay();
free(saved_line);
}
2013-10-03 16:38:25 +04:00
}
2013-10-16 23:19:39 +04:00
const char *message_media_type_str (struct message_media *M) {
static char buf[1000];
switch (M->type) {
case CODE_message_media_empty:
return "";
case CODE_message_media_photo:
return "[photo]";
case CODE_message_media_video:
return "[video]";
case CODE_message_media_geo:
sprintf (buf, "[geo] %.6lf:%.6lf", M->geo.latitude, M->geo.longitude);
return buf;
case CODE_message_media_contact:
snprintf (buf, 999, "[contact] " COLOR_RED "%s %s" COLOR_NORMAL " %s", M->first_name, M->last_name, M->phone);
return buf;
case CODE_message_media_unsupported:
return "[unsupported]";
default:
assert (0);
return "";
}
}
2013-10-16 23:19:39 +04:00
void print_message (struct message *M) {
if (M->service) {
rprintf ("Service message\n");
return;
}
if (M->to_id >= 0) {
if (M->out) {
union user_chat *U = user_chat_get (M->to_id);
assert (M->from_id >= 0);
if (U) {
rprintf (COLOR_RED "%s %s" COLOR_GREEN " <<< %s %s" COLOR_NORMAL "\n", U->user.first_name, U->user.last_name, M->message, message_media_type_str (&M->media));
} else {
rprintf (COLOR_RED "User #%d" COLOR_GREEN " <<< %s %s" COLOR_NORMAL "\n", M->from_id, M->message, message_media_type_str (&M->media));
}
2013-10-16 23:19:39 +04:00
} else {
union user_chat *U = user_chat_get (M->from_id);
assert (M->from_id >= 0);
if (U) {
rprintf (COLOR_RED "%s %s" COLOR_BLUE " >>> %s %s" COLOR_NORMAL "\n", U->user.first_name, U->user.last_name, M->message, message_media_type_str (&M->media));
} else {
rprintf (COLOR_RED "User #%d" COLOR_BLUE " >>> %s %s" COLOR_NORMAL "\n", M->from_id, M->message, message_media_type_str (&M->media));
}
}
} else {
rprintf ("Message to chat %d\n", -M->to_id);
union user_chat *C = user_chat_get (M->to_id);
if (C) {
rprintf ("Chat %s\n", C->chat.title);
2013-10-16 23:19:39 +04:00
}
}
}