about summary refs log tree commit diff stats
path: root/src/config/accounts.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/config/accounts.c')
-rw-r--r--src/config/accounts.c532
1 files changed, 532 insertions, 0 deletions
diff --git a/src/config/accounts.c b/src/config/accounts.c
new file mode 100644
index 00000000..6549819b
--- /dev/null
+++ b/src/config/accounts.c
@@ -0,0 +1,532 @@
+/*
+ * accounts.c
+ *
+ * Copyright (C) 2012, 2013 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 "accounts.h"
+
+#include "common.h"
+#include "files.h"
+#include "jid.h"
+#include "log.h"
+#include "tools/autocomplete.h"
+#include "xmpp/xmpp.h"
+
+static gchar *accounts_loc;
+static GKeyFile *accounts;
+
+static Autocomplete all_ac;
+static Autocomplete enabled_ac;
+
+static gchar *string_keys[] = {"jid", "server", "resource", "presence.last", "presence.login"};
+
+static void _fix_legacy_accounts(const char * const account_name);
+static void _save_accounts(void);
+
+void
+accounts_load(void)
+{
+    log_info("Loading accounts");
+    all_ac = autocomplete_new();
+    enabled_ac = autocomplete_new();
+    accounts_loc = files_get_accounts_file();
+
+    accounts = g_key_file_new();
+    g_key_file_load_from_file(accounts, accounts_loc, G_KEY_FILE_KEEP_COMMENTS,
+        NULL);
+
+    // create the logins searchable list for autocompletion
+    gsize naccounts;
+    gchar **account_names =
+        g_key_file_get_groups(accounts, &naccounts);
+
+    gsize i;
+    for (i = 0; i < naccounts; i++) {
+        autocomplete_add(all_ac, strdup(account_names[i]));
+        if (g_key_file_get_boolean(accounts, account_names[i], "enabled", NULL)) {
+            autocomplete_add(enabled_ac, strdup(account_names[i]));
+        }
+
+        _fix_legacy_accounts(account_names[i]);
+    }
+
+    for (i = 0; i < naccounts; i++) {
+        free(account_names[i]);
+    }
+    free(account_names);
+}
+
+void
+accounts_close(void)
+{
+    autocomplete_free(all_ac);
+    autocomplete_free(enabled_ac);
+    g_key_file_free(accounts);
+}
+
+char *
+accounts_find_enabled(char *prefix)
+{
+    return autocomplete_complete(enabled_ac, prefix);
+}
+
+char *
+accounts_find_all(char *prefix)
+{
+    return autocomplete_complete(all_ac, prefix);
+}
+
+void
+accounts_reset_all_search(void)
+{
+    autocomplete_reset(all_ac);
+}
+
+void
+accounts_reset_enabled_search(void)
+{
+    autocomplete_reset(enabled_ac);
+}
+
+void
+accounts_add(const char *account_name, const char *altdomain)
+{
+    // set account name and resource
+    const char *barejid = account_name;
+    const char *resource = "profanity";
+    Jid *jid = jid_create(account_name);
+    if (jid != NULL) {
+        barejid = jid->barejid;
+        if (jid->resourcepart != NULL) {
+            resource = jid->resourcepart;
+        }
+    }
+
+    // doesn't yet exist
+    if (!g_key_file_has_group(accounts, account_name)) {
+        g_key_file_set_boolean(accounts, account_name, "enabled", TRUE);
+        g_key_file_set_string(accounts, account_name, "jid", barejid);
+        g_key_file_set_string(accounts, account_name, "resource", resource);
+        if (altdomain != NULL) {
+            g_key_file_set_string(accounts, account_name, "server", altdomain);
+        }
+        g_key_file_set_string(accounts, account_name, "presence.last", "online");
+        g_key_file_set_string(accounts, account_name, "presence.login", "online");
+        g_key_file_set_integer(accounts, account_name, "priority.online", 0);
+        g_key_file_set_integer(accounts, account_name, "priority.chat", 0);
+        g_key_file_set_integer(accounts, account_name, "priority.away", 0);
+        g_key_file_set_integer(accounts, account_name, "priority.xa", 0);
+        g_key_file_set_integer(accounts, account_name, "priority.dnd", 0);
+
+        _save_accounts();
+        autocomplete_add(all_ac, strdup(account_name));
+        autocomplete_add(enabled_ac, strdup(account_name));
+    }
+
+    jid_destroy(jid);
+}
+
+gchar**
+accounts_get_list(void)
+{
+    return g_key_file_get_groups(accounts, NULL);
+}
+
+ProfAccount*
+accounts_get_account(const char * const name)
+{
+    if (!g_key_file_has_group(accounts, name)) {
+        return NULL;
+    } else {
+        ProfAccount *account = malloc(sizeof(ProfAccount));
+        account->name = strdup(name);
+
+        gchar *jid = g_key_file_get_string(accounts, name, "jid", NULL);
+        if (jid != NULL) {
+            account->jid = strdup(jid);
+        } else {
+            account->jid = strdup(name);
+            g_key_file_set_string(accounts, name, "jid", name);
+            _save_accounts();
+        }
+
+        account->enabled = g_key_file_get_boolean(accounts, name, "enabled", NULL);
+
+        gchar *server = g_key_file_get_string(accounts, name, "server", NULL);
+        if (server != NULL) {
+            account->server = strdup(server);
+        } else {
+            account->server = NULL;
+        }
+
+        gchar *resource = g_key_file_get_string(accounts, name, "resource", NULL);
+        if (resource != NULL) {
+            account->resource = strdup(resource);
+        } else {
+            account->resource = NULL;
+        }
+
+        gchar *presence = g_key_file_get_string(accounts, name, "presence.last", NULL);
+        if (presence == NULL || (!presence_valid_string(presence))) {
+            account->last_presence = strdup("online");
+        } else {
+            account->last_presence = strdup(presence);
+        }
+
+        presence = g_key_file_get_string(accounts, name, "presence.login", NULL);
+        if (presence == NULL) {
+            account->login_presence = strdup("online");
+        } else if (strcmp(presence, "last") == 0) {
+            account->login_presence = strdup("last");
+        } else if (!presence_valid_string(presence)) {
+            account->login_presence = strdup("online");
+        } else {
+            account->login_presence = strdup(presence);
+        }
+
+        account->priority_online = g_key_file_get_integer(accounts, name, "priority.online", NULL);
+        account->priority_chat = g_key_file_get_integer(accounts, name, "priority.chat", NULL);
+        account->priority_away = g_key_file_get_integer(accounts, name, "priority.away", NULL);
+        account->priority_xa = g_key_file_get_integer(accounts, name, "priority.xa", NULL);
+        account->priority_dnd = g_key_file_get_integer(accounts, name, "priority.dnd", NULL);
+
+        return account;
+    }
+}
+
+void
+accounts_free_account(ProfAccount *account)
+{
+    if (account != NULL) {
+        FREE_SET_NULL(account->name);
+        FREE_SET_NULL(account->jid);
+        FREE_SET_NULL(account->resource);
+        FREE_SET_NULL(account->server);
+        FREE_SET_NULL(account->last_presence);
+        FREE_SET_NULL(account->login_presence);
+        FREE_SET_NULL(account);
+    }
+}
+
+gboolean
+accounts_enable(const char * const name)
+{
+    if (g_key_file_has_group(accounts, name)) {
+        g_key_file_set_boolean(accounts, name, "enabled", TRUE);
+        _save_accounts();
+        autocomplete_add(enabled_ac, strdup(name));
+        return TRUE;
+    } else {
+        return FALSE;
+    }
+}
+
+gboolean
+accounts_disable(const char * const name)
+{
+    if (g_key_file_has_group(accounts, name)) {
+        g_key_file_set_boolean(accounts, name, "enabled", FALSE);
+        _save_accounts();
+        autocomplete_remove(enabled_ac, strdup(name));
+        return TRUE;
+    } else {
+        return FALSE;
+    }
+}
+
+gboolean
+accounts_rename(const char * const account_name, const char * const new_name)
+{
+    if (g_key_file_has_group(accounts, new_name)) {
+        return FALSE;
+    }
+
+    if (!g_key_file_has_group(accounts, account_name)) {
+        return FALSE;
+    }
+
+    g_key_file_set_boolean(accounts, new_name, "enabled",
+        g_key_file_get_boolean(accounts, account_name, "enabled", NULL));
+
+    g_key_file_set_integer(accounts, new_name, "priority.online",
+        g_key_file_get_boolean(accounts, account_name, "priority.online", NULL));
+    g_key_file_set_integer(accounts, new_name, "priority.chat",
+        g_key_file_get_boolean(accounts, account_name, "priority.chat", NULL));
+    g_key_file_set_integer(accounts, new_name, "priority.away",
+        g_key_file_get_boolean(accounts, account_name, "priority.away", NULL));
+    g_key_file_set_integer(accounts, new_name, "priority.xa",
+        g_key_file_get_boolean(accounts, account_name, "priority.xa", NULL));
+    g_key_file_set_integer(accounts, new_name, "priority.dnd",
+        g_key_file_get_boolean(accounts, account_name, "priority.dnd", NULL));
+
+    int i;
+    for (i = 0; i < ARRAY_SIZE(string_keys); i++) {
+        char *value = g_key_file_get_string(accounts, account_name, string_keys[i], NULL);
+        if (value != NULL) {
+            g_key_file_set_string(accounts, new_name, string_keys[i], value);
+            free(value);
+        }
+    }
+
+    g_key_file_remove_group(accounts, account_name, NULL);
+    _save_accounts();
+
+    autocomplete_remove(all_ac, strdup(account_name));
+    autocomplete_add(all_ac, strdup(new_name));
+    if (g_key_file_get_boolean(accounts, new_name, "enabled", NULL)) {
+        autocomplete_remove(enabled_ac, strdup(account_name));
+        autocomplete_add(enabled_ac, strdup(new_name));
+    }
+
+    return TRUE;
+}
+
+gboolean
+accounts_account_exists(const char * const account_name)
+{
+    return g_key_file_has_group(accounts, account_name);
+
+}
+
+void
+accounts_set_jid(const char * const account_name, const char * const value)
+{
+    Jid *jid = jid_create(value);
+    if (jid != NULL) {
+        if (accounts_account_exists(account_name)) {
+            g_key_file_set_string(accounts, account_name, "jid", jid->barejid);
+            if (jid->resourcepart != NULL) {
+                g_key_file_set_string(accounts, account_name, "resource", jid->resourcepart);
+            }
+            _save_accounts();
+        }
+    }
+}
+
+void
+accounts_set_server(const char * const account_name, const char * const value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_string(accounts, account_name, "server", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_resource(const char * const account_name, const char * const value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_string(accounts, account_name, "resource", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_priority_online(const char * const account_name, const gint value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_integer(accounts, account_name, "priority.online", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_priority_chat(const char * const account_name, const gint value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_integer(accounts, account_name, "priority.chat", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_priority_away(const char * const account_name, const gint value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_integer(accounts, account_name, "priority.away", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_priority_xa(const char * const account_name, const gint value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_integer(accounts, account_name, "priority.xa", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_priority_dnd(const char * const account_name, const gint value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_integer(accounts, account_name, "priority.dnd", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_priority_all(const char * const account_name, const gint value)
+{
+    if (accounts_account_exists(account_name)) {
+        accounts_set_priority_online(account_name, value);
+        accounts_set_priority_chat(account_name, value);
+        accounts_set_priority_away(account_name, value);
+        accounts_set_priority_xa(account_name, value);
+        accounts_set_priority_dnd(account_name, value);
+        _save_accounts();
+    }
+}
+
+gint
+accounts_get_priority_for_presence_type(const char * const account_name,
+    jabber_presence_t presence_type)
+{
+    gint result;
+
+    switch (presence_type)
+    {
+        case (PRESENCE_ONLINE):
+            result = g_key_file_get_integer(accounts, account_name, "priority.online", NULL);
+            break;
+        case (PRESENCE_CHAT):
+            result = g_key_file_get_integer(accounts, account_name, "priority.chat", NULL);
+            break;
+        case (PRESENCE_AWAY):
+            result = g_key_file_get_integer(accounts, account_name, "priority.away", NULL);
+            break;
+        case (PRESENCE_XA):
+            result = g_key_file_get_integer(accounts, account_name, "priority.xa", NULL);
+            break;
+        case (PRESENCE_DND):
+            result = g_key_file_get_integer(accounts, account_name, "priority.dnd", NULL);
+            break;
+        default:
+            result = 0;
+            break;
+    }
+
+    return result;
+}
+
+void
+accounts_set_last_presence(const char * const account_name, const char * const value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_string(accounts, account_name, "presence.last", value);
+        _save_accounts();
+    }
+}
+
+void
+accounts_set_login_presence(const char * const account_name, const char * const value)
+{
+    if (accounts_account_exists(account_name)) {
+        g_key_file_set_string(accounts, account_name, "presence.login", value);
+        _save_accounts();
+    }
+}
+
+jabber_presence_t
+accounts_get_last_presence(const char * const account_name)
+{
+    gchar *setting = g_key_file_get_string(accounts, account_name, "presence.last", NULL);
+    if (setting == NULL || (strcmp(setting, "online") == 0)) {
+        return PRESENCE_ONLINE;
+    } else if (strcmp(setting, "chat") == 0) {
+        return PRESENCE_CHAT;
+    } else if (strcmp(setting, "away") == 0) {
+        return PRESENCE_AWAY;
+    } else if (strcmp(setting, "xa") == 0) {
+        return PRESENCE_XA;
+    } else if (strcmp(setting, "dnd") == 0) {
+        return PRESENCE_DND;
+    } else {
+        log_warning("Error reading presence.last for account: '%s', value: '%s', defaulting to 'online'",
+            account_name, setting);
+        return PRESENCE_ONLINE;
+    }
+}
+
+jabber_presence_t
+accounts_get_login_presence(const char * const account_name)
+{
+    gchar *setting = g_key_file_get_string(accounts, account_name, "presence.login", NULL);
+    if (setting == NULL || (strcmp(setting, "online") == 0)) {
+        return PRESENCE_ONLINE;
+    } else if (strcmp(setting, "chat") == 0) {
+        return PRESENCE_CHAT;
+    } else if (strcmp(setting, "away") == 0) {
+        return PRESENCE_AWAY;
+    } else if (strcmp(setting, "xa") == 0) {
+        return PRESENCE_XA;
+    } else if (strcmp(setting, "dnd") == 0) {
+        return PRESENCE_DND;
+    } else if (strcmp(setting, "last") == 0) {
+        return accounts_get_last_presence(account_name);
+    } else {
+        log_warning("Error reading presence.login for account: '%s', value: '%s', defaulting to 'online'",
+            account_name, setting);
+        return PRESENCE_ONLINE;
+    }
+}
+
+static void
+_fix_legacy_accounts(const char * const account_name)
+{
+    // set barejid and resource
+    const char *barejid = account_name;
+    const char *resource = "profanity";
+    Jid *jid = jid_create(account_name);
+    if (jid != NULL) {
+        barejid = jid->barejid;
+        if (jid->resourcepart != NULL) {
+            resource = jid->resourcepart;
+        }
+    }
+
+    // accounts with no jid property
+    if (!g_key_file_has_key(accounts, account_name, "jid", NULL)) {
+        g_key_file_set_string(accounts, account_name, "jid", barejid);
+        _save_accounts();
+    }
+
+    // accounts with no resource, property
+    if (!g_key_file_has_key(accounts, account_name, "resource", NULL)) {
+        g_key_file_set_string(accounts, account_name, "resource", resource);
+        _save_accounts();
+    }
+
+    jid_destroy(jid);
+}
+
+static void
+_save_accounts(void)
+{
+    gsize g_data_size;
+    char *g_accounts_data = g_key_file_to_data(accounts, &g_data_size, NULL);
+    g_file_set_contents(accounts_loc, g_accounts_data, g_data_size, NULL);
+}