bitlbee-discord/src/discord.c

331 lines
9.1 KiB
C

/*
* Copyright 2015-2016 Artem Savkov <artem.savkov@gmail.com>
*
* This program 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.
*
* This program 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 program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "config.h"
#include "discord.h"
#include "discord-http.h"
#include "discord-util.h"
#include "discord-websockets.h"
#include "help.h"
#define HELPFILE_NAME "discord-help.txt"
static void discord_help_init()
{
/* Figure out where our help file is by looking at the global helpfile. */
gchar *dir = g_path_get_dirname (global.helpfile);
if (strcmp(dir, ".") == 0) {
log_message(LOGLVL_WARNING, "Error finding the directory of helpfile %s.", global.helpfile);
g_free(dir);
return;
}
gchar *df = g_strjoin("/", dir, HELPFILE_NAME, NULL);
g_free(dir);
/* Load help from our own help file. */
help_t *dh;
help_init(&dh, df);
if(dh == NULL) {
log_message(LOGLVL_WARNING, "Error opening helpfile: %s.", df);
g_free(df);
return;
}
g_free(df);
/* Link the last entry of global.help with first entry of our help. */
help_t *h, *l = NULL;
for (h = global.help; h; h = h->next) {
l = h;
}
if (l) {
l->next = dh;
} else {
/* No global help but ours? */
global.help = dh;
}
}
#ifdef BITLBEE_ABI_VERSION_CODE
struct plugin_info *init_plugin_info(void)
{
static struct plugin_info info = {
BITLBEE_ABI_VERSION_CODE,
"bitlbee-discord",
PACKAGE_VERSION,
"Bitlbee plugin for discordapp.com",
"Artem Savkov <artem.savkov@gmail.com>",
"https://github.com/sm00th/bitlbee-discord"
};
return &info;
}
#endif
static void discord_init(account_t *acc)
{
set_t *s;
s = set_add(&acc->set, "host", DISCORD_HOST, NULL, acc);
s->flags |= ACC_SET_OFFLINE_ONLY;
s = set_add(&acc->set, "voice_status_notify", "off", set_eval_bool, acc);
s = set_add(&acc->set, "send_acks", "on", set_eval_bool, acc);
s = set_add(&acc->set, "edit_prefix", "EDIT: ", NULL, acc);
s = set_add(&acc->set, "urlinfo_handle", "urlinfo", NULL, acc);
s = set_add(&acc->set, "mention_suffix", ":", NULL, acc);
s = set_add(&acc->set, "mention_ignorecase", "off", set_eval_bool, acc);
s = set_add(&acc->set, "incoming_me_translation", "on", set_eval_bool, acc);
s = set_add(&acc->set, "fetch_pinned", "off", set_eval_bool, acc);
s = set_add(&acc->set, "always_afk", "off", set_eval_bool, acc);
s = set_add(&acc->set, "emoji_urls", "on", set_eval_bool, acc);
s = set_add(&acc->set, "auto_join", "off", set_eval_bool, acc);
s->flags |= ACC_SET_OFFLINE_ONLY;
s = set_add(&acc->set, "auto_join_exclude", "", NULL, acc);
s->flags |= ACC_SET_OFFLINE_ONLY;
s = set_add(&acc->set, "max_backlog", "50", set_eval_int, acc);
s->flags |= ACC_SET_OFFLINE_ONLY;
s = set_add(&acc->set, "never_offline", "off", set_eval_bool, acc);
s->flags |= ACC_SET_OFFLINE_ONLY;
s = set_add(&acc->set, "server_prefix_len", "3", set_eval_int, acc);
s->flags |= ACC_SET_OFFLINE_ONLY;
s = set_add(&acc->set, "token_cache", NULL, NULL, acc);
s->flags |= SET_HIDDEN | SET_NULL_OK;
s = set_add(&acc->set, "friendship_mode", "on", set_eval_bool, acc);
s->flags |= ACC_SET_OFFLINE_ONLY;
s = set_add(&acc->set, "verbose", "off", set_eval_bool, acc);
acc->flags |= ACC_FLAG_AWAY_MESSAGE;
acc->flags |= ACC_FLAG_STATUS_MESSAGE;
discord_help_init();
}
static void discord_do_login(struct im_connection *ic)
{
if (set_getstr(&ic->acc->set,"token_cache")) {
discord_http_get_gateway(ic, set_getstr(&ic->acc->set,"token_cache"));
} else {
discord_http_login(ic->acc);
}
}
static void discord_login(account_t *acc)
{
struct im_connection *ic = imcb_new(acc);
discord_data *dd = g_new0(discord_data, 1);
dd->sent_message_ids = g_hash_table_new_full(g_str_hash, g_str_equal,
g_free, NULL);
dd->keepalive_interval = DEFAULT_KEEPALIVE_INTERVAL;
ic->proto_data = dd;
discord_do_login(ic);
}
static void discord_logout(struct im_connection *ic)
{
discord_data *dd = ic->proto_data;
discord_ws_cleanup(dd);
free_discord_data(dd);
g_slist_free(ic->chatlist);
}
void discord_soft_reconnect(struct im_connection *ic)
{
discord_data *dd = ic->proto_data;
if (set_getbool(&ic->acc->set, "verbose")) {
imcb_log(ic, "Performing soft-reconnect");
}
discord_ws_cleanup(dd);
dd->reconnecting = TRUE;
discord_do_login(ic);
}
static void discord_chat_msg(struct groupchat *gc, char *msg, int flags)
{
channel_info *cinfo = gc->data;
discord_http_send_msg(gc->ic, cinfo->id, msg);
}
static void discord_chat_list(struct im_connection *ic, const char *server)
{
imcb_chat_list_finish(ic);
}
static struct groupchat *discord_chat_join(struct im_connection *ic,
const char *room,
const char *nick,
const char *password,
set_t **sets)
{
return discord_chat_do_join(ic, room, FALSE);
}
struct groupchat *discord_chat_do_join(struct im_connection *ic,
const char *room,
gboolean is_auto_join)
{
discord_data *dd = ic->proto_data;
struct groupchat *gc = NULL;
server_info *sinfo = NULL;
channel_info *cinfo = get_channel(dd, room, NULL, SEARCH_FNAME);
if (cinfo != NULL && cinfo->type == CHANNEL_TEXT) {
sinfo = cinfo->to.channel.sinfo;
gc = imcb_chat_new(ic, cinfo->to.channel.name);
discord_ws_sync_channel(dd, sinfo->id, cinfo->id, 0);
if (is_auto_join) {
imcb_chat_name_hint(gc, room);
}
if (cinfo->to.channel.bci->topic != NULL) {
imcb_chat_topic(gc, "root", cinfo->to.channel.bci->topic, 0);
}
for (GSList *ul = sinfo->users; ul; ul = g_slist_next(ul)) {
user_info *uinfo = ul->data;
if (uinfo->flags & BEE_USER_ONLINE) {
imcb_chat_add_buddy(gc, uinfo->user->handle);
}
}
imcb_chat_add_buddy(gc, dd->uname);
cinfo->to.channel.gc = gc;
} else if (cinfo != NULL && cinfo->type == CHANNEL_GROUP_PRIVATE) {
gc = imcb_chat_new(ic, cinfo->to.group.name);
discord_ws_sync_private_group(dd, cinfo->id);
if (is_auto_join) {
imcb_chat_name_hint(gc, room);
}
for (GSList *ul = cinfo->to.group.users; ul; ul = g_slist_next(ul)) {
user_info *uinfo = ul->data;
imcb_chat_add_buddy(gc, uinfo->user->handle);
}
imcb_chat_add_buddy(gc, dd->uname);
cinfo->to.group.gc = gc;
} else {
return NULL;
}
gc->data = cinfo;
if (set_getbool(&ic->acc->set, "fetch_pinned")) {
discord_http_get_pinned(ic, cinfo->id);
}
if (set_getint(&ic->acc->set, "max_backlog") > 0 &&
cinfo->last_msg > cinfo->last_read) {
discord_http_get_backlog(ic, cinfo->id);
}
return gc;
}
static void discord_chat_leave(struct groupchat *gc)
{
channel_info *cinfo = gc->data;
imcb_chat_free(cinfo->to.channel.gc);
cinfo->to.channel.gc = NULL;
}
static int discord_buddy_msg(struct im_connection *ic, char *to, char *msg,
int flags)
{
discord_data *dd = ic->proto_data;
if (g_strcmp0(to, DISCORD_MFA_HANDLE) == 0) {
discord_http_mfa_auth(ic, msg);
return 0;
}
for (GSList *cl = dd->pchannels; cl; cl = g_slist_next(cl)) {
channel_info *cinfo = cl->data;
if (cinfo->type == CHANNEL_PRIVATE &&
g_strcmp0(cinfo->to.handle.name, to) == 0) {
discord_http_send_msg(ic, cinfo->id, msg);
return 0;
}
}
// If we are here we didn't find an appropriate channel, create it
discord_http_create_and_send_msg(ic, to, msg);
return 0;
}
gboolean discord_is_self(struct im_connection *ic, const char *who)
{
discord_data *dd = ic->proto_data;
return !g_strcmp0(dd->uname, who);
}
static GList *discord_away_states(struct im_connection *ic)
{
static GList *m = NULL;
m = g_list_prepend(m, "invisible");
m = g_list_prepend(m, "dnd");
m = g_list_prepend(m, "online");
m = g_list_prepend(m, "idle");
return m;
}
static void discord_set_away(struct im_connection *ic, char *state,
char *message)
{
discord_ws_set_status(ic, state, message);
}
G_MODULE_EXPORT void init_plugin(void)
{
struct prpl *dpp;
static const struct prpl pp = {
.name = "discord",
.init = discord_init,
.login = discord_login,
.logout = discord_logout,
.chat_msg = discord_chat_msg,
.chat_list = discord_chat_list,
.chat_join = discord_chat_join,
.chat_leave = discord_chat_leave,
.buddy_msg = discord_buddy_msg,
.handle_cmp = g_strcmp0,
.handle_is_self = discord_is_self,
.away_states = discord_away_states,
.set_away = discord_set_away
};
dpp = g_memdup(&pp, sizeof pp);
register_protocol(dpp);
}