566 lines
17 KiB
C++
566 lines
17 KiB
C++
#include "glib.h"
|
|
#include "purple.h"
|
|
#include <iostream>
|
|
|
|
#include "transport/config.h"
|
|
#include "transport/transport.h"
|
|
#include "transport/usermanager.h"
|
|
#include "transport/logger.h"
|
|
#include "transport/sqlite3backend.h"
|
|
#include "transport/userregistration.h"
|
|
#include "transport/user.h"
|
|
#include "transport/storagebackend.h"
|
|
#include "transport/rostermanager.h"
|
|
#include "transport/conversation.h"
|
|
#include "transport/networkplugin.h"
|
|
#include "spectrumeventloop.h"
|
|
#include "spectrumbuddy.h"
|
|
#include "spectrumconversation.h"
|
|
#include "geventloop.h"
|
|
|
|
#define Log(X, STRING) std::cout << "[SPECTRUM] " << X << " " << STRING << "\n";
|
|
|
|
|
|
using namespace Transport;
|
|
|
|
class SpectrumNetworkPlugin;
|
|
|
|
Logger *_logger;
|
|
SpectrumNetworkPlugin *np;
|
|
|
|
static gboolean nodaemon = FALSE;
|
|
static gchar *logfile = NULL;
|
|
static gchar *lock_file = NULL;
|
|
static gchar *host = NULL;
|
|
static int port = 10000;
|
|
static gboolean ver = FALSE;
|
|
static gboolean upgrade_db = FALSE;
|
|
static gboolean check_db_version = FALSE;
|
|
static gboolean list_purple_settings = FALSE;
|
|
|
|
static GOptionEntry options_entries[] = {
|
|
{ "nodaemon", 'n', 0, G_OPTION_ARG_NONE, &nodaemon, "Disable background daemon mode", NULL },
|
|
{ "logfile", 'l', 0, G_OPTION_ARG_STRING, &logfile, "Set file to log", NULL },
|
|
{ "pidfile", 'p', 0, G_OPTION_ARG_STRING, &lock_file, "File where to write transport PID", NULL },
|
|
{ "version", 'v', 0, G_OPTION_ARG_NONE, &ver, "Shows Spectrum version", NULL },
|
|
{ "list-purple-settings", 's', 0, G_OPTION_ARG_NONE, &list_purple_settings, "Lists purple settings which can be used in config file", NULL },
|
|
{ "host", 'h', 0, G_OPTION_ARG_STRING, &host, "Host to connect to", NULL },
|
|
{ "port", 'p', 0, G_OPTION_ARG_INT, &port, "Port to connect to", NULL },
|
|
{ NULL, 0, 0, G_OPTION_ARG_NONE, NULL, "", NULL }
|
|
};
|
|
|
|
class SpectrumNetworkPlugin : public NetworkPlugin {
|
|
public:
|
|
SpectrumNetworkPlugin(Config *config, SpectrumEventLoop *loop, const std::string &host, int port) : NetworkPlugin(loop, host, port) {
|
|
this->config = config;
|
|
}
|
|
|
|
void handleLoginRequest(const std::string &user, const std::string &legacyName, const std::string &password) {
|
|
PurpleAccount *account = NULL;
|
|
const char *protocol = CONFIG_STRING(config, "service.protocol").c_str();
|
|
if (purple_accounts_find(legacyName.c_str(), protocol) != NULL){
|
|
Log(user, "this account already exists");
|
|
account = purple_accounts_find(legacyName.c_str(), protocol);
|
|
// User *u = (User *) account->ui_data;
|
|
// if (u && u != user) {
|
|
// Log(userInfo.jid, "This account is already connected by another jid " << user->getJID());
|
|
// return;
|
|
// }
|
|
}
|
|
else {
|
|
Log(user, "creating new account");
|
|
account = purple_account_new(legacyName.c_str(), protocol);
|
|
|
|
purple_accounts_add(account);
|
|
}
|
|
|
|
purple_account_set_password(account, password.c_str());
|
|
purple_account_set_enabled(account, "spectrum", TRUE);
|
|
|
|
const PurpleStatusType *status_type = purple_account_get_status_type_with_primitive(account, PURPLE_STATUS_AVAILABLE);
|
|
if (status_type != NULL) {
|
|
purple_account_set_status(account, purple_status_type_get_id(status_type), TRUE, NULL);
|
|
}
|
|
m_accounts[account] = user;
|
|
}
|
|
|
|
void handleLogoutRequest(const std::string &user, const std::string &legacyName) {
|
|
const char *protocol = CONFIG_STRING(config, "service.protocol").c_str();
|
|
PurpleAccount *account = purple_accounts_find(legacyName.c_str(), protocol);
|
|
if (account) {
|
|
purple_account_set_enabled(account, "spectrum", FALSE);
|
|
|
|
// Remove conversations.
|
|
// This has to be called before m_account->ui_data = NULL;, because it uses
|
|
// ui_data to call SpectrumMessageHandler::purpleConversationDestroyed() callback.
|
|
GList *iter;
|
|
for (iter = purple_get_conversations(); iter; ) {
|
|
PurpleConversation *conv = (PurpleConversation*) iter->data;
|
|
iter = iter->next;
|
|
if (purple_conversation_get_account(conv) == account)
|
|
purple_conversation_destroy(conv);
|
|
}
|
|
|
|
g_free(account->ui_data);
|
|
account->ui_data = NULL;
|
|
m_accounts.erase(account);
|
|
}
|
|
}
|
|
|
|
std::map<PurpleAccount *, std::string> m_accounts;
|
|
private:
|
|
Config *config;
|
|
};
|
|
|
|
static std::string getAlias(PurpleBuddy *m_buddy) {
|
|
std::string alias;
|
|
if (purple_buddy_get_server_alias(m_buddy))
|
|
alias = (std::string) purple_buddy_get_server_alias(m_buddy);
|
|
else
|
|
alias = (std::string) purple_buddy_get_alias(m_buddy);
|
|
return alias;
|
|
}
|
|
|
|
static std::string getName(PurpleBuddy *m_buddy) {
|
|
std::string name(purple_buddy_get_name(m_buddy));
|
|
if (name.empty()) {
|
|
Log("getName", "Name is EMPTY!");
|
|
}
|
|
return name;
|
|
}
|
|
|
|
static bool getStatus(PurpleBuddy *m_buddy, Swift::StatusShow &status, std::string &statusMessage) {
|
|
PurplePresence *pres = purple_buddy_get_presence(m_buddy);
|
|
if (pres == NULL)
|
|
return false;
|
|
PurpleStatus *stat = purple_presence_get_active_status(pres);
|
|
if (stat == NULL)
|
|
return false;
|
|
int st = purple_status_type_get_primitive(purple_status_get_type(stat));
|
|
|
|
switch(st) {
|
|
case PURPLE_STATUS_AVAILABLE: {
|
|
break;
|
|
}
|
|
case PURPLE_STATUS_AWAY: {
|
|
status = Swift::StatusShow::Away;
|
|
break;
|
|
}
|
|
case PURPLE_STATUS_UNAVAILABLE: {
|
|
status = Swift::StatusShow::DND;
|
|
break;
|
|
}
|
|
case PURPLE_STATUS_EXTENDED_AWAY: {
|
|
status = Swift::StatusShow::XA;
|
|
break;
|
|
}
|
|
case PURPLE_STATUS_OFFLINE: {
|
|
status = Swift::StatusShow::None;
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
const char *message = purple_status_get_attr_string(stat, "message");
|
|
|
|
if (message != NULL) {
|
|
char *stripped = purple_markup_strip_html(message);
|
|
statusMessage = std::string(stripped);
|
|
g_free(stripped);
|
|
}
|
|
else
|
|
statusMessage = "";
|
|
return true;
|
|
}
|
|
|
|
static std::string getIconHash(PurpleBuddy *m_buddy) {
|
|
char *avatarHash = NULL;
|
|
PurpleBuddyIcon *icon = purple_buddy_icons_find(purple_buddy_get_account(m_buddy), purple_buddy_get_name(m_buddy));
|
|
if (icon) {
|
|
avatarHash = purple_buddy_icon_get_full_path(icon);
|
|
}
|
|
|
|
if (avatarHash) {
|
|
// Check if it's patched libpurple which saves icons to directories
|
|
char *hash = strrchr(avatarHash,'/');
|
|
std::string h;
|
|
if (hash) {
|
|
char *dot;
|
|
hash++;
|
|
dot = strchr(hash, '.');
|
|
if (dot)
|
|
*dot = '\0';
|
|
|
|
std::string ret(hash);
|
|
g_free(avatarHash);
|
|
return ret;
|
|
}
|
|
else {
|
|
std::string ret(avatarHash);
|
|
g_free(avatarHash);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
static std::vector<std::string> getGroups(PurpleBuddy *m_buddy) {
|
|
std::vector<std::string> groups;
|
|
groups.push_back(purple_group_get_name(purple_buddy_get_group(m_buddy)) ? std::string(purple_group_get_name(purple_buddy_get_group(m_buddy))) : std::string("Buddies"));
|
|
return groups;
|
|
}
|
|
|
|
static void buddyListNewNode(PurpleBlistNode *node) {
|
|
if (!PURPLE_BLIST_NODE_IS_BUDDY(node))
|
|
return;
|
|
PurpleBuddy *buddy = (PurpleBuddy *) node;
|
|
PurpleAccount *account = purple_buddy_get_account(buddy);
|
|
|
|
Swift::StatusShow status;
|
|
std::string message;
|
|
getStatus(buddy, status, message);
|
|
|
|
np->handleBuddyChanged(np->m_accounts[account], purple_buddy_get_name(buddy), getAlias(buddy), getGroups(buddy)[0], (int) status.getType(), message, getIconHash(buddy));
|
|
}
|
|
|
|
static void buddyStatusChanged(PurpleBuddy *buddy, PurpleStatus *status_, PurpleStatus *old_status) {
|
|
PurpleAccount *account = purple_buddy_get_account(buddy);
|
|
|
|
Swift::StatusShow status;
|
|
std::string message;
|
|
getStatus(buddy, status, message);
|
|
|
|
np->handleBuddyChanged(np->m_accounts[account], purple_buddy_get_name(buddy), getAlias(buddy), getGroups(buddy)[0], (int) status.getType(), message, getIconHash(buddy));
|
|
}
|
|
|
|
static void buddySignedOn(PurpleBuddy *buddy) {
|
|
PurpleAccount *account = purple_buddy_get_account(buddy);
|
|
|
|
Swift::StatusShow status;
|
|
std::string message;
|
|
getStatus(buddy, status, message);
|
|
|
|
np->handleBuddyChanged(np->m_accounts[account], purple_buddy_get_name(buddy), getAlias(buddy), getGroups(buddy)[0], (int) status.getType(), message, getIconHash(buddy));
|
|
}
|
|
|
|
static void buddySignedOff(PurpleBuddy *buddy) {
|
|
PurpleAccount *account = purple_buddy_get_account(buddy);
|
|
|
|
Swift::StatusShow status;
|
|
std::string message;
|
|
getStatus(buddy, status, message);
|
|
|
|
np->handleBuddyChanged(np->m_accounts[account], purple_buddy_get_name(buddy), getAlias(buddy), getGroups(buddy)[0], (int) status.getType(), message, getIconHash(buddy));
|
|
}
|
|
|
|
static void NodeRemoved(PurpleBlistNode *node, void *data) {
|
|
if (!PURPLE_BLIST_NODE_IS_BUDDY(node))
|
|
return;
|
|
PurpleBuddy *buddy = (PurpleBuddy *) node;
|
|
|
|
// PurpleAccount *account = purple_buddy_get_account(buddy);
|
|
// User *user = (User *) account->ui_data;
|
|
if (buddy->node.ui_data) {
|
|
SpectrumBuddy *s_buddy = (SpectrumBuddy *) buddy->node.ui_data;
|
|
s_buddy->removeBuddy(buddy);
|
|
buddy->node.ui_data = NULL;
|
|
if (s_buddy->getBuddiesCount() == 0) {
|
|
delete s_buddy;
|
|
}
|
|
}
|
|
}
|
|
|
|
static PurpleBlistUiOps blistUiOps =
|
|
{
|
|
NULL,
|
|
buddyListNewNode,
|
|
NULL,
|
|
NULL, // buddyListUpdate,
|
|
NULL, //NodeRemoved,
|
|
NULL,
|
|
NULL,
|
|
NULL, // buddyListAddBuddy,
|
|
NULL,
|
|
NULL,
|
|
NULL, //buddyListSaveNode,
|
|
NULL, //buddyListRemoveNode,
|
|
NULL, //buddyListSaveAccount,
|
|
NULL
|
|
};
|
|
|
|
static void conv_new(PurpleConversation *conv) {
|
|
PurpleAccount *account = purple_conversation_get_account(conv);
|
|
User *user = (User *) account->ui_data;
|
|
|
|
if (!user)
|
|
return;
|
|
|
|
std::string name = purple_conversation_get_name(conv);
|
|
size_t pos = name.find("/");
|
|
if (pos != std::string::npos)
|
|
name.erase((int) pos, name.length() - (int) pos);
|
|
|
|
SpectrumConversation *s_conv = new SpectrumConversation(user->getConversationManager(), name, conv);
|
|
conv->ui_data = s_conv;
|
|
}
|
|
|
|
static void conv_destroy(PurpleConversation *conv) {
|
|
SpectrumConversation *s_conv = (SpectrumConversation *) conv->ui_data;
|
|
if (s_conv) {
|
|
delete s_conv;
|
|
}
|
|
}
|
|
|
|
static void conv_write_im(PurpleConversation *conv, const char *who, const char *message, PurpleMessageFlags flags, time_t mtime) {
|
|
// Don't forwards our own messages.
|
|
if (flags & PURPLE_MESSAGE_SEND || flags & PURPLE_MESSAGE_SYSTEM)
|
|
return;
|
|
SpectrumConversation *s_conv = (SpectrumConversation *) conv->ui_data;
|
|
if (!s_conv)
|
|
return;
|
|
|
|
boost::shared_ptr<Swift::Message> msg(new Swift::Message());
|
|
|
|
char *striped = purple_markup_strip_html(message);
|
|
msg->setBody(message);
|
|
g_free(striped);
|
|
|
|
s_conv->handleMessage(msg);
|
|
}
|
|
|
|
static PurpleConversationUiOps conversation_ui_ops =
|
|
{
|
|
conv_new,
|
|
conv_destroy,
|
|
NULL,//conv_write_chat, /* write_chat */
|
|
conv_write_im, /* write_im */
|
|
NULL,//conv_write_conv, /* write_conv */
|
|
NULL,//conv_chat_add_users, /* chat_add_users */
|
|
NULL,//conv_chat_rename_user, /* chat_rename_user */
|
|
NULL,//conv_chat_remove_users, /* chat_remove_users */
|
|
NULL,//pidgin_conv_chat_update_user, /* chat_update_user */
|
|
NULL,//pidgin_conv_present_conversation, /* present */
|
|
NULL,//pidgin_conv_has_focus, /* has_focus */
|
|
NULL,//pidgin_conv_custom_smiley_add, /* custom_smiley_add */
|
|
NULL,//pidgin_conv_custom_smiley_write, /* custom_smiley_write */
|
|
NULL,//pidgin_conv_custom_smiley_close, /* custom_smiley_close */
|
|
NULL,//pidgin_conv_send_confirm, /* send_confirm */
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
};
|
|
|
|
static void connection_report_disconnect(PurpleConnection *gc, PurpleConnectionError reason, const char *text){
|
|
PurpleAccount *account = purple_connection_get_account(gc);
|
|
np->handleDisconnected(np->m_accounts[account], purple_account_get_username(account), (int) reason, text ? text : "");
|
|
}
|
|
|
|
static PurpleConnectionUiOps conn_ui_ops =
|
|
{
|
|
NULL,
|
|
NULL,
|
|
NULL,//connection_disconnected,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
connection_report_disconnect,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
};
|
|
|
|
static void transport_core_ui_init(void)
|
|
{
|
|
purple_blist_set_ui_ops(&blistUiOps);
|
|
// purple_accounts_set_ui_ops(&accountUiOps);
|
|
// purple_notify_set_ui_ops(¬ifyUiOps);
|
|
// purple_request_set_ui_ops(&requestUiOps);
|
|
// purple_xfers_set_ui_ops(getXferUiOps());
|
|
purple_connections_set_ui_ops(&conn_ui_ops);
|
|
purple_conversations_set_ui_ops(&conversation_ui_ops);
|
|
// #ifndef WIN32
|
|
// purple_dnsquery_set_ui_ops(getDNSUiOps());
|
|
// #endif
|
|
}
|
|
|
|
static PurpleCoreUiOps coreUiOps =
|
|
{
|
|
NULL,
|
|
// debug_init,
|
|
NULL,
|
|
transport_core_ui_init,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
};
|
|
|
|
static void printDebug(PurpleDebugLevel level, const char *category, const char *arg_s) {
|
|
std::string c("[LIBPURPLE");
|
|
|
|
if (category) {
|
|
c.push_back('/');
|
|
c.append(category);
|
|
}
|
|
|
|
c.push_back(']');
|
|
|
|
std::cout << c << " " << arg_s;
|
|
}
|
|
|
|
/*
|
|
* Ops....
|
|
*/
|
|
static PurpleDebugUiOps debugUiOps =
|
|
{
|
|
printDebug,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
};
|
|
|
|
static bool initPurple(Config &cfg) {
|
|
bool ret;
|
|
|
|
purple_util_set_user_dir("./");
|
|
remove("./accounts.xml");
|
|
remove("./blist.xml");
|
|
|
|
// if (m_configuration.logAreas & LOG_AREA_PURPLE)
|
|
purple_debug_set_ui_ops(&debugUiOps);
|
|
|
|
purple_core_set_ui_ops(&coreUiOps);
|
|
purple_eventloop_set_ui_ops(getEventLoopUiOps());
|
|
|
|
ret = purple_core_init("spectrum");
|
|
if (ret) {
|
|
static int conversation_handle;
|
|
static int conn_handle;
|
|
static int blist_handle;
|
|
|
|
purple_set_blist(purple_blist_new());
|
|
purple_blist_load();
|
|
|
|
purple_prefs_load();
|
|
|
|
/* Good default preferences */
|
|
/* The combination of these two settings mean that libpurple will never
|
|
* (of its own accord) set all the user accounts idle.
|
|
*/
|
|
purple_prefs_set_bool("/purple/away/away_when_idle", false);
|
|
/*
|
|
* This must be set to something not "none" for idle reporting to work
|
|
* for, e.g., the OSCAR prpl. We don't implement the UI ops, so this is
|
|
* okay for now.
|
|
*/
|
|
purple_prefs_set_string("/purple/away/idle_reporting", "system");
|
|
|
|
/* Disable all logging */
|
|
purple_prefs_set_bool("/purple/logging/log_ims", false);
|
|
purple_prefs_set_bool("/purple/logging/log_chats", false);
|
|
purple_prefs_set_bool("/purple/logging/log_system", false);
|
|
|
|
|
|
// purple_signal_connect(purple_conversations_get_handle(), "received-im-msg", &conversation_handle, PURPLE_CALLBACK(newMessageReceived), NULL);
|
|
// purple_signal_connect(purple_conversations_get_handle(), "buddy-typing", &conversation_handle, PURPLE_CALLBACK(buddyTyping), NULL);
|
|
// purple_signal_connect(purple_conversations_get_handle(), "buddy-typed", &conversation_handle, PURPLE_CALLBACK(buddyTyped), NULL);
|
|
// purple_signal_connect(purple_conversations_get_handle(), "buddy-typing-stopped", &conversation_handle, PURPLE_CALLBACK(buddyTypingStopped), NULL);
|
|
// purple_signal_connect(purple_connections_get_handle(), "signed-on", &conn_handle,PURPLE_CALLBACK(signed_on), NULL);
|
|
// purple_signal_connect(purple_blist_get_handle(), "buddy-removed", &blist_handle,PURPLE_CALLBACK(buddyRemoved), NULL);
|
|
purple_signal_connect(purple_blist_get_handle(), "buddy-signed-on", &blist_handle,PURPLE_CALLBACK(buddySignedOn), NULL);
|
|
purple_signal_connect(purple_blist_get_handle(), "buddy-signed-off", &blist_handle,PURPLE_CALLBACK(buddySignedOff), NULL);
|
|
purple_signal_connect(purple_blist_get_handle(), "buddy-status-changed", &blist_handle,PURPLE_CALLBACK(buddyStatusChanged), NULL);
|
|
purple_signal_connect(purple_blist_get_handle(), "blist-node-removed", &blist_handle,PURPLE_CALLBACK(NodeRemoved), NULL);
|
|
// purple_signal_connect(purple_conversations_get_handle(), "chat-topic-changed", &conversation_handle, PURPLE_CALLBACK(conv_chat_topic_changed), NULL);
|
|
//
|
|
// purple_commands_init();
|
|
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
GError *error = NULL;
|
|
GOptionContext *context;
|
|
context = g_option_context_new("config_file_name or profile name");
|
|
g_option_context_add_main_entries(context, options_entries, "");
|
|
if (!g_option_context_parse (context, &argc, &argv, &error)) {
|
|
std::cout << "option parsing failed: " << error->message << "\n";
|
|
return -1;
|
|
}
|
|
|
|
if (ver) {
|
|
// std::cout << VERSION << "\n";
|
|
std::cout << "verze\n";
|
|
g_option_context_free(context);
|
|
return 0;
|
|
}
|
|
|
|
if (argc != 2) {
|
|
#ifdef WIN32
|
|
std::cout << "Usage: spectrum.exe <configuration_file.cfg>\n";
|
|
#else
|
|
|
|
#if GLIB_CHECK_VERSION(2,14,0)
|
|
std::cout << g_option_context_get_help(context, FALSE, NULL);
|
|
#else
|
|
std::cout << "Usage: spectrum <configuration_file.cfg>\n";
|
|
std::cout << "See \"man spectrum\" for more info.\n";
|
|
#endif
|
|
|
|
#endif
|
|
}
|
|
else {
|
|
#ifndef WIN32
|
|
// signal(SIGPIPE, SIG_IGN);
|
|
//
|
|
// if (signal(SIGCHLD, spectrum_sigchld_handler) == SIG_ERR) {
|
|
// std::cout << "SIGCHLD handler can't be set\n";
|
|
// g_option_context_free(context);
|
|
// return -1;
|
|
// }
|
|
//
|
|
// if (signal(SIGINT, spectrum_sigint_handler) == SIG_ERR) {
|
|
// std::cout << "SIGINT handler can't be set\n";
|
|
// g_option_context_free(context);
|
|
// return -1;
|
|
// }
|
|
//
|
|
// if (signal(SIGTERM, spectrum_sigterm_handler) == SIG_ERR) {
|
|
// std::cout << "SIGTERM handler can't be set\n";
|
|
// g_option_context_free(context);
|
|
// return -1;
|
|
// }
|
|
//
|
|
// struct sigaction sa;
|
|
// memset(&sa, 0, sizeof(sa));
|
|
// sa.sa_handler = spectrum_sighup_handler;
|
|
// if (sigaction(SIGHUP, &sa, NULL)) {
|
|
// std::cout << "SIGHUP handler can't be set\n";
|
|
// g_option_context_free(context);
|
|
// return -1;
|
|
// }
|
|
#endif
|
|
Config config;
|
|
if (!config.load(argv[1])) {
|
|
std::cout << "Can't open " << argv[1] << " configuration file.\n";
|
|
return 1;
|
|
}
|
|
|
|
initPurple(config);
|
|
|
|
SpectrumEventLoop eventLoop;
|
|
np = new SpectrumNetworkPlugin(&config, &eventLoop, host, port);
|
|
eventLoop.run();
|
|
}
|
|
|
|
g_option_context_free(context);
|
|
}
|