#include #include #include #include #include #include #include "log.h" #include "muc.h" #include "ui/ui.h" #include "xmpp/connection.h" #include "xmpp/stanza.h" #include "xmpp/xmpp.h" #define BOOKMARK_TIMEOUT 5000 /* TODO: replace with a preference */ #define BOOKMARK_AUTOJOIN_MAX 5 static int autojoin_count; static int _bookmark_handle_result(xmpp_conn_t * const conn, xmpp_stanza_t * const stanza, void * const userdata); static int _bookmark_handle_delete(xmpp_conn_t * const conn, void * const userdata); void bookmark_request(void) { char *id; xmpp_conn_t *conn = connection_get_conn(); xmpp_ctx_t *ctx = connection_get_ctx(); xmpp_stanza_t *iq; id = get_unique_id(); if (!id) { return; } autojoin_count = 0; xmpp_timed_handler_add(conn, _bookmark_handle_delete, BOOKMARK_TIMEOUT, id); xmpp_id_handler_add(conn, _bookmark_handle_result, id, id); iq = stanza_create_storage_bookmarks(ctx); xmpp_stanza_set_id(iq, id); xmpp_send(conn, iq); xmpp_stanza_release(iq); } static int _bookmark_handle_result(xmpp_conn_t * const conn, xmpp_stanza_t * const stanza, void * const userdata) { xmpp_ctx_t *ctx = connection_get_ctx(); char *id = (char *)userdata; xmpp_stanza_t *ptr; xmpp_stanza_t *nick; char *name; char *jid; char *autojoin; Jid *my_jid; xmpp_timed_handler_delete(conn, _bookmark_handle_delete); g_free(id); name = xmpp_stanza_get_name(stanza); if (!name || strcmp(name, STANZA_NAME_IQ) != 0) { return 0; } ptr = xmpp_stanza_get_child_by_name(stanza, STANZA_NAME_QUERY); if (!ptr) { return 0; } ptr = xmpp_stanza_get_child_by_name(ptr, STANZA_NAME_STORAGE); if (!ptr) { return 0; } my_jid = jid_create(jabber_get_fulljid()); ptr = xmpp_stanza_get_children(ptr); while (ptr) { name = xmpp_stanza_get_name(ptr); if (strcmp(name, STANZA_NAME_CONFERENCE) != 0) { continue; } jid = xmpp_stanza_get_attribute(ptr, STANZA_ATTR_JID); if (!jid) { continue; } log_debug("Handle bookmark for %s", jid); autojoin = xmpp_stanza_get_attribute(ptr, "autojoin"); if (autojoin && strcmp(autojoin, "1") == 0) { name = NULL; nick = xmpp_stanza_get_child_by_name(ptr, "nick"); if (nick) { char *tmp; tmp = xmpp_stanza_get_text(nick); if (tmp) { name = strdup(tmp); xmpp_free(ctx, tmp); } } else { name = strdup(my_jid->localpart); } if (name) { if (autojoin_count < BOOKMARK_AUTOJOIN_MAX) { Jid *room_jid; log_debug("Autojoin %s with nick=%s", jid, name); ++autojoin_count; room_jid = jid_create_from_bare_and_resource(jid, name); if (!muc_room_is_active(room_jid)) { presence_join_room(room_jid); /* TODO: this should be removed after fixing #195 */ ui_room_join(room_jid); } jid_destroy(room_jid); } else { log_debug("Rejected autojoin %s (maximum has been reached)", jid); } free(name); } } /* TODO: add to autocompleter */ ptr = xmpp_stanza_get_next(ptr); } jid_destroy(my_jid); return 0; } static int _bookmark_handle_delete(xmpp_conn_t * const conn, void * const userdata) { char *id = (char *)userdata; assert(id != NULL); log_debug("Timeout for handler with id=%s", id); xmpp_id_handler_delete(conn, _bookmark_handle_result, id); g_free(id); return 0; }