diff options
-rw-r--r-- | Makefile.am | 5 | ||||
-rw-r--r-- | src/chat_session.c | 112 | ||||
-rw-r--r-- | src/chat_session.h | 38 | ||||
-rw-r--r-- | src/jabber.c | 20 | ||||
-rw-r--r-- | src/profanity.c | 2 | ||||
-rw-r--r-- | tests/test_chat_session.c | 73 | ||||
-rw-r--r-- | tests/testsuite.c | 1 | ||||
-rw-r--r-- | tests/testsuite.h | 1 |
8 files changed, 245 insertions, 7 deletions
diff --git a/Makefile.am b/Makefile.am index 72343a7b..63e46796 100644 --- a/Makefile.am +++ b/Makefile.am @@ -6,14 +6,15 @@ profanity_SOURCES = src/command.c src/contact.c src/history.c src/jabber.h \ src/contact_list.c src/input_win.c src/log.h src/profanity.c \ src/prof_history.c src/ui.h src/common.h src/ contact_list.h src/jabber.c \ src/main.c src/profanity.h src/prof_history.h src/chat_log.c \ - src/chat_log.h src/tinyurl.c src/tinyurl.h src/release.c src/release.h + src/chat_log.h src/tinyurl.c src/tinyurl.h src/chat_session.c \ + src/chat_session.h src/release.c src/release.h TESTS = tests/testsuite check_PROGRAMS = tests/testsuite tests_testsuite_SOURCES = tests/test_contact_list.c src/contact_list.c src/contact.c \ tests/test_common.c tests/test_prof_history.c src/prof_history.c src/common.c \ tests/test_prof_autocomplete.c src/prof_autocomplete.c tests/testsuite.c \ - src/log.c + tests/test_chat_session.c src/chat_session.c src/log.c tests_testsuite_LDADD = -lheadunit -lstdc++ man_MANS = docs/profanity.1 diff --git a/src/chat_session.c b/src/chat_session.c new file mode 100644 index 00000000..283860de --- /dev/null +++ b/src/chat_session.c @@ -0,0 +1,112 @@ +/* + * chat_session.c + * + * Copyright (C) 2012 James Booth <boothj5@gmail.com> + * + * This file is part of Profanity. + * + * Profanity 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 3 of the License, or + * (at your option) any later version. + * + * Profanity 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 Profanity. If not, see <http://www.gnu.org/licenses/>. + * + */ + +#include <stdlib.h> +#include <string.h> + +#include <glib.h> + +#include "chat_session.h" +#include "log.h" + +static ChatSession _chat_session_new(const char * const recipient, + gboolean recipient_supports); +static void _chat_session_free(ChatSession session); + +struct chat_session_t { + char *recipient; + gboolean recipient_supports; +}; + +static GHashTable *sessions; + +void +chat_sessions_init(void) +{ + sessions = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, + (GDestroyNotify)_chat_session_free); +} + +void +chat_sessions_clear(void) +{ + g_hash_table_remove_all(sessions); +} + +gboolean +chat_session_exists(const char * const recipient) +{ + ChatSession session = g_hash_table_lookup(sessions, recipient); + + if (session != NULL) { + return TRUE; + } else { + return FALSE; + } +} + +void +chat_session_start(const char * const recipient, gboolean recipient_supports) +{ + char *key = strdup(recipient); + ChatSession session = _chat_session_new(key, recipient_supports); + g_hash_table_insert(sessions, key, session); +} + +void +chat_session_end(const char * const recipient) +{ + g_hash_table_remove(sessions, recipient); +} + +gboolean +chat_session_recipient_supports(const char * const recipient) +{ + ChatSession session = g_hash_table_lookup(sessions, recipient); + + if (session == NULL) { + log_error("No chat session found for %s.", recipient); + return FALSE; + } else { + return session->recipient_supports; + } +} + +static ChatSession +_chat_session_new(const char * const recipient, gboolean recipient_supports) +{ + ChatSession new_session = malloc(sizeof(struct chat_session_t)); + new_session->recipient = strdup(recipient); + new_session->recipient_supports = recipient_supports; + + return new_session; +} + +static void +_chat_session_free(ChatSession session) +{ + if (session != NULL) { + g_free(session->recipient); + g_free(session); + } + session = NULL; +} diff --git a/src/chat_session.h b/src/chat_session.h new file mode 100644 index 00000000..7cfc8de5 --- /dev/null +++ b/src/chat_session.h @@ -0,0 +1,38 @@ +/* + * chat_session.h + * + * Copyright (C) 2012 James Booth <boothj5@gmail.com> + * + * This file is part of Profanity. + * + * Profanity 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 3 of the License, or + * (at your option) any later version. + * + * Profanity 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 Profanity. If not, see <http://www.gnu.org/licenses/>. + * + */ + +#ifndef CHAT_SESSION_H +#define CHAT_SESSION_H + +#include <glib.h> + +typedef struct chat_session_t *ChatSession; + +void chat_sessions_init(void); +void chat_sessions_clear(void); +void chat_session_start(const char * const recipient, + gboolean recipient_supports); +gboolean chat_session_exists(const char * const recipient); +void chat_session_end(const char * const recipient); +gboolean chat_session_recipient_supports(const char * const recipient); + +#endif diff --git a/src/jabber.c b/src/jabber.c index 8c9c9ba0..23f7f444 100644 --- a/src/jabber.c +++ b/src/jabber.c @@ -25,6 +25,7 @@ #include <strophe.h> +#include "chat_session.h" #include "common.h" #include "contact_list.h" #include "jabber.h" @@ -136,16 +137,16 @@ jabber_process_events(void) void jabber_send(const char * const msg, const char * const recipient) { + if (!chat_session_exists(recipient)) { + chat_session_start(recipient, TRUE); + } + char *coded_msg = str_replace(msg, "&", "&"); char *coded_msg2 = str_replace(coded_msg, "<", "<"); char *coded_msg3 = str_replace(coded_msg2, ">", ">"); xmpp_stanza_t *reply, *body, *text, *active; - active = xmpp_stanza_new(jabber_conn.ctx); - xmpp_stanza_set_name(active, "active"); - xmpp_stanza_set_ns(active, "http://jabber.org/protocol/chatstates"); - reply = xmpp_stanza_new(jabber_conn.ctx); xmpp_stanza_set_name(reply, "message"); xmpp_stanza_set_type(reply, "chat"); @@ -156,7 +157,14 @@ jabber_send(const char * const msg, const char * const recipient) text = xmpp_stanza_new(jabber_conn.ctx); xmpp_stanza_set_text(text, coded_msg3); - xmpp_stanza_add_child(reply, active); + + if (chat_session_recipient_supports(recipient)) { + active = xmpp_stanza_new(jabber_conn.ctx); + xmpp_stanza_set_name(active, "active"); + xmpp_stanza_set_ns(active, "http://jabber.org/protocol/chatstates"); + xmpp_stanza_add_child(reply, active); + } + xmpp_stanza_add_child(body, text); xmpp_stanza_add_child(reply, body); @@ -244,6 +252,7 @@ jabber_get_jid(void) void jabber_free_resources(void) { + chat_sessions_clear(); xmpp_conn_release(jabber_conn.conn); xmpp_ctx_free(jabber_conn.ctx); xmpp_shutdown(); @@ -335,6 +344,7 @@ _connection_handler(xmpp_conn_t * const conn, if (status == XMPP_CONN_CONNECT) { const char *jid = xmpp_conn_get_jid(conn); prof_handle_login_success(jid); + chat_sessions_init(); xmpp_handler_add(conn, _message_handler, NULL, "message", NULL, ctx); xmpp_handler_add(conn, _presence_handler, NULL, "presence", NULL, ctx); diff --git a/src/profanity.c b/src/profanity.c index e4888e97..885c55ea 100644 --- a/src/profanity.c +++ b/src/profanity.c @@ -87,6 +87,8 @@ prof_run(const int disable_tls, char *log_level) inp[size++] = '\0'; cmd_result = _process_input(inp); } + + g_timer_destroy(timer); } void diff --git a/tests/test_chat_session.c b/tests/test_chat_session.c new file mode 100644 index 00000000..0a4dd262 --- /dev/null +++ b/tests/test_chat_session.c @@ -0,0 +1,73 @@ +#include <stdio.h> +#include <head-unit.h> +#include "chat_session.h" + +void setup(void) +{ + chat_session_init(); +} + +void adding_new_sets_state_to_active(void) +{ + chat_session_start("prof1@panesar"); + chat_state_t state = chat_session_get_state("prof1@panesar"); + + assert_int_equals(ACTIVE, state); +} + +void set_inactive(void) +{ + chat_session_start("prof2@panesar"); + chat_session_set_state("prof2@panesar", INACTIVE); + chat_state_t state = chat_session_get_state("prof2@panesar"); + + assert_int_equals(INACTIVE, state); +} + +void set_gone(void) +{ + chat_session_start("prof3@panesar"); + chat_session_set_state("prof3@panesar", GONE); + chat_state_t state = chat_session_get_state("prof3@panesar"); + + assert_int_equals(GONE, state); +} + +void set_composing(void) +{ + chat_session_start("prof4@panesar"); + chat_session_set_state("prof4@panesar", COMPOSING); + chat_state_t state = chat_session_get_state("prof4@panesar"); + + assert_int_equals(COMPOSING, state); +} + +void set_paused(void) +{ + chat_session_start("prof5@panesar"); + chat_session_set_state("prof5@panesar", PAUSED); + chat_state_t state = chat_session_get_state("prof5@panesar"); + + assert_int_equals(PAUSED, state); +} + +void end_session(void) +{ + chat_session_start(strdup("prof6@panesar")); + chat_session_end("prof6@panesar"); + chat_state_t state = chat_session_get_state("prof5@panesat"); + + assert_int_equals(SESSION_ERR, state); +} + +void register_chat_session_tests(void) +{ + TEST_MODULE("chat_session_tests"); + SETUP(setup); + TEST(adding_new_sets_state_to_active); + TEST(set_inactive); + TEST(set_gone); + TEST(set_composing); + TEST(set_paused); + TEST(end_session); +} diff --git a/tests/testsuite.c b/tests/testsuite.c index e7a23005..b4001ef1 100644 --- a/tests/testsuite.c +++ b/tests/testsuite.c @@ -7,6 +7,7 @@ int main(void) register_contact_list_tests(); register_common_tests(); register_prof_autocomplete_tests(); + register_chat_session_tests(); run_suite(); return 0; } diff --git a/tests/testsuite.h b/tests/testsuite.h index 45145c9b..cd836622 100644 --- a/tests/testsuite.h +++ b/tests/testsuite.h @@ -5,5 +5,6 @@ void register_prof_history_tests(void); void register_contact_list_tests(void); void register_common_tests(void); void register_prof_autocomplete_tests(void); +void register_chat_session_tests(void); #endif |