mirror of
https://github.com/hexchat/hexchat.git
synced 2024-11-05 10:42:33 +01:00
279 lines
7.2 KiB
C
279 lines
7.2 KiB
C
/*
|
|
|
|
Copyright (c) 2010 Samuel Lidén Borell <samuel@kodafritt.se>
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <glib.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include "irc.h"
|
|
#include "fish.h"
|
|
#include "keystore.h"
|
|
#include "plugin_hexchat.h"
|
|
|
|
|
|
static char *keystore_password = NULL;
|
|
|
|
|
|
/**
|
|
* Opens the key store file: ~/.config/hexchat/addon_fishlim.conf
|
|
*/
|
|
static GKeyFile *getConfigFile(void) {
|
|
gchar *filename = get_config_filename();
|
|
|
|
GKeyFile *keyfile = g_key_file_new();
|
|
g_key_file_load_from_file(keyfile, filename,
|
|
G_KEY_FILE_KEEP_COMMENTS |
|
|
G_KEY_FILE_KEEP_TRANSLATIONS, NULL);
|
|
|
|
g_free(filename);
|
|
return keyfile;
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns the key store password, or the default.
|
|
*/
|
|
static const char *get_keystore_password(void) {
|
|
return (keystore_password != NULL ?
|
|
keystore_password :
|
|
/* Silly default value... */
|
|
"blowinikey");
|
|
}
|
|
|
|
|
|
static char *escape_nickname(const char *nick) {
|
|
char *escaped = g_strdup(nick);
|
|
char *p = escaped;
|
|
|
|
while (*p) {
|
|
if (*p == '[')
|
|
*p = '~';
|
|
else if (*p == ']')
|
|
*p = '!';
|
|
|
|
++p;
|
|
}
|
|
|
|
return escaped;
|
|
}
|
|
|
|
/**
|
|
* Gets a value for a nick/channel from addon_fishlim.conf. Unlike
|
|
* g_key_file_get_string, this function is case insensitive.
|
|
*/
|
|
static gchar *get_nick_value(GKeyFile *keyfile, const char *nick, const char *item) {
|
|
gchar **group;
|
|
gchar **groups = g_key_file_get_groups(keyfile, NULL);
|
|
gchar *result = NULL;
|
|
|
|
for (group = groups; *group != NULL; group++) {
|
|
if (!irc_nick_cmp(*group, nick)) {
|
|
result = g_key_file_get_string(keyfile, *group, item, NULL);
|
|
break;
|
|
}
|
|
}
|
|
|
|
g_strfreev(groups);
|
|
return result;
|
|
}
|
|
|
|
|
|
/**
|
|
* Extracts a key from the key store file.
|
|
*/
|
|
char *keystore_get_key(const char *nick, enum fish_mode *mode) {
|
|
GKeyFile *keyfile;
|
|
char *escaped_nick;
|
|
gchar *value, *key_mode;
|
|
int encrypted_mode;
|
|
char *password;
|
|
char *encrypted;
|
|
char *decrypted;
|
|
|
|
/* Get the key */
|
|
keyfile = getConfigFile();
|
|
escaped_nick = escape_nickname(nick);
|
|
value = get_nick_value(keyfile, escaped_nick, "key");
|
|
key_mode = get_nick_value(keyfile, escaped_nick, "mode");
|
|
g_key_file_free(keyfile);
|
|
g_free(escaped_nick);
|
|
|
|
/* Determine cipher mode */
|
|
*mode = FISH_ECB_MODE;
|
|
if (key_mode) {
|
|
if (*key_mode == '1')
|
|
*mode = FISH_ECB_MODE;
|
|
else if (*key_mode == '2')
|
|
*mode = FISH_CBC_MODE;
|
|
g_free(key_mode);
|
|
}
|
|
|
|
if (!value)
|
|
return NULL;
|
|
|
|
if (strncmp(value, "+OK ", 4) == 0) {
|
|
/* Key is encrypted */
|
|
encrypted = (char *) value;
|
|
encrypted += 4;
|
|
|
|
encrypted_mode = FISH_ECB_MODE;
|
|
|
|
if (*encrypted == '*') {
|
|
++encrypted;
|
|
encrypted_mode = FISH_CBC_MODE;
|
|
}
|
|
|
|
password = (char *) get_keystore_password();
|
|
decrypted = fish_decrypt_str((const char *) password, strlen(password), (const char *) encrypted, encrypted_mode);
|
|
g_free(value);
|
|
return decrypted;
|
|
} else {
|
|
/* Key is stored in plaintext */
|
|
return value;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Deletes a nick and the associated key in the key store file.
|
|
*/
|
|
static gboolean delete_nick(GKeyFile *keyfile, const char *nick) {
|
|
gchar **group;
|
|
gchar **groups = g_key_file_get_groups(keyfile, NULL);
|
|
gboolean ok = FALSE;
|
|
|
|
for (group = groups; *group != NULL; group++) {
|
|
if (!irc_nick_cmp(*group, nick)) {
|
|
ok = g_key_file_remove_group(keyfile, *group, NULL);
|
|
break;
|
|
}
|
|
}
|
|
|
|
g_strfreev(groups);
|
|
return ok;
|
|
}
|
|
|
|
#if !GLIB_CHECK_VERSION(2,40,0)
|
|
/**
|
|
* Writes the key store file to disk.
|
|
*/
|
|
static gboolean keyfile_save_to_file (GKeyFile *keyfile, char *filename) {
|
|
gboolean ok;
|
|
|
|
/* Serialize */
|
|
gsize file_length;
|
|
gchar *file_data = g_key_file_to_data(keyfile, &file_length, NULL);
|
|
if (!file_data)
|
|
return FALSE;
|
|
|
|
/* Write to file */
|
|
ok = g_file_set_contents (filename, file_data, file_length, NULL);
|
|
g_free(file_data);
|
|
return ok;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* Writes the key store file to disk.
|
|
*/
|
|
static gboolean save_keystore(GKeyFile *keyfile) {
|
|
char *filename;
|
|
gboolean ok;
|
|
|
|
filename = get_config_filename();
|
|
#if !GLIB_CHECK_VERSION(2,40,0)
|
|
ok = keyfile_save_to_file (keyfile, filename);
|
|
#else
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS /* Hide false positive */
|
|
ok = g_key_file_save_to_file (keyfile, filename, NULL);
|
|
G_GNUC_END_IGNORE_DEPRECATIONS
|
|
#endif
|
|
g_free (filename);
|
|
|
|
return ok;
|
|
}
|
|
|
|
/**
|
|
* Sets a key in the key store file.
|
|
*/
|
|
gboolean keystore_store_key(const char *nick, const char *key, enum fish_mode mode) {
|
|
const char *password;
|
|
char *encrypted;
|
|
char *wrapped;
|
|
gboolean ok = FALSE;
|
|
GKeyFile *keyfile = getConfigFile();
|
|
char *escaped_nick = escape_nickname(nick);
|
|
|
|
/* Remove old key */
|
|
delete_nick(keyfile, escaped_nick);
|
|
|
|
/* Add new key */
|
|
password = get_keystore_password();
|
|
if (password) {
|
|
/* Encrypt the password */
|
|
encrypted = fish_encrypt(password, strlen(password), key, strlen(key), FISH_CBC_MODE);
|
|
if (!encrypted) goto end;
|
|
|
|
/* Prepend "+OK " */
|
|
wrapped = g_strconcat("+OK *", encrypted, NULL);
|
|
g_free(encrypted);
|
|
|
|
/* Store encrypted in file */
|
|
g_key_file_set_string(keyfile, escaped_nick, "key", wrapped);
|
|
g_free(wrapped);
|
|
} else {
|
|
/* Store unencrypted in file */
|
|
g_key_file_set_string(keyfile, escaped_nick, "key", key);
|
|
}
|
|
|
|
/* Store cipher mode */
|
|
g_key_file_set_integer(keyfile, escaped_nick, "mode", mode);
|
|
|
|
/* Save key store file */
|
|
ok = save_keystore(keyfile);
|
|
|
|
end:
|
|
g_key_file_free(keyfile);
|
|
g_free(escaped_nick);
|
|
return ok;
|
|
}
|
|
|
|
/**
|
|
* Deletes a nick from the key store.
|
|
*/
|
|
gboolean keystore_delete_nick(const char *nick) {
|
|
GKeyFile *keyfile = getConfigFile();
|
|
char *escaped_nick = escape_nickname(nick);
|
|
|
|
/* Delete entry */
|
|
gboolean ok = delete_nick(keyfile, escaped_nick);
|
|
|
|
/* Save */
|
|
if (ok) save_keystore(keyfile);
|
|
|
|
g_key_file_free(keyfile);
|
|
g_free(escaped_nick);
|
|
return ok;
|
|
}
|