summary refs log tree commit diff stats
path: root/plugins/fishlim/irc.c
blob: 3586921bbc20141a8bb63dcf3194bc81c0906865 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
/*

  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 <stdlib.h>
#include <string.h>
#include "irc.h"

/**
 * Parses an IRC message. The words array should contain the message splitted
 * at spaces. The prefix and command is extracted from the message, and
 * parameters_offset is set to the index of the first parameter.
 */
bool irc_parse_message(const char *words[],
                       const char **prefix, const char **command,
                       size_t *parameters_offset) {
    size_t w = 1;
    if (prefix) *prefix = NULL;
    if (command) *command = NULL;
    
    // See if the message starts with a prefix (sender user)
    if (words[w][0] == ':') {
        if (prefix) *prefix = &words[w][1];
        w++;
    }
    
    // Check command
    if (words[w][0] == '\0') return false;
    if (command) *command = words[w];
    w++;
    
    *parameters_offset = w;
    return true;
}


/**
 * Finds the nick part of a "IRC prefix", which can have any
 * of the following forms:
 *
 *     nick
 *     nick@host
 *     nick!ident
 *     nick!ident@host
 */
char *irc_prefix_get_nick(const char *prefix) {
    const char *end;
    char *nick;
    size_t length;
    
    if (!prefix) return NULL;
    
    // Find end of nick
    end = prefix;
    while (*end != '\0' && *end != '!' && *end != '@') end++;
    
    // Allocate string
    length = end - prefix;
    nick = malloc(length+1);
    if (!nick) return NULL;
    
    // Copy to string
    memcpy(nick, prefix, length);
    nick[length] = '\0';
    return nick;
}


/**
 * Compares two nick names. Return 0 if equal. Otherwise the return value is
 * less than zero if a is less than b or greater than zero if a is greater
 * than b.
 */
int irc_nick_cmp(const char *a, const char *b) {
    char ac;
    char bc;
    char diff;
    for (;;) {
        ac = *(a++);
        bc = *(b++);
        
        // Change into IRC uppercase (see RFC 2812 section 2.2)
        if (ac >= 'a' && ac <= '~') ac &= ~0x20;
        if (bc >= 'a' && bc <= '~') bc &= ~0x20;
        
        diff = ac - bc;
        if (diff) return diff;
        if (!ac) return 0;
    }
}
ass="w"> xmlChar *code, *name; code = xmlTextReaderGetAttribute (reader, (const xmlChar *) "alpha_2_code"); name = xmlTextReaderGetAttribute (reader, (const xmlChar *) "name"); if (code != NULL && code[0] != '\0' && name != NULL && name[0] != '\0') { char *lcode; lcode = g_ascii_strdown ((char *) code, -1); xmlFree (code); g_hash_table_insert (table, lcode, name); } else { xmlFree (code); xmlFree (name); } } typedef enum { STATE_START, STATE_STOP, STATE_ENTRIES, } ParserState; static gboolean load_iso_entries (int iso, GFunc read_entry_func, gpointer user_data) { xmlTextReaderPtr reader; ParserState state = STATE_START; xmlChar iso_entries[32], iso_entry[32]; char *filename; int ret = -1; #ifdef WIN32 filename = g_strdup_printf (".\\share\\xml\\iso-codes\\iso_%d.xml", iso); #else filename = g_strdup_printf ("/usr/share/xml/iso-codes/iso_%d.xml", iso); #endif reader = xmlNewTextReaderFilename (filename); if (reader == NULL) goto out; xmlStrPrintf (iso_entries, sizeof (iso_entries), (xmlChar *)"iso_%d_entries", iso); xmlStrPrintf (iso_entry, sizeof (iso_entry), (xmlChar *)"iso_%d_entry", iso); ret = xmlTextReaderRead (reader); while (ret == 1) { const xmlChar *tag; xmlReaderTypes type; tag = xmlTextReaderConstName (reader); type = xmlTextReaderNodeType (reader); if (state == STATE_ENTRIES && type == XML_READER_TYPE_ELEMENT && xmlStrEqual (tag, iso_entry)) { read_entry_func (reader, user_data); } else if (state == STATE_START && type == XML_READER_TYPE_ELEMENT && xmlStrEqual (tag, iso_entries)) { state = STATE_ENTRIES; } else if (state == STATE_ENTRIES && type == XML_READER_TYPE_END_ELEMENT && xmlStrEqual (tag, iso_entries)) { state = STATE_STOP; } else if (type == XML_READER_TYPE_SIGNIFICANT_WHITESPACE || type == XML_READER_TYPE_WHITESPACE || type == XML_READER_TYPE_TEXT || type == XML_READER_TYPE_COMMENT) { /* eat it */ } else { /* ignore it */ } ret = xmlTextReaderRead (reader); } xmlFreeTextReader (reader); out: if (ret < 0 || state != STATE_STOP) { /* This is not critical, we will fallback to our own code */ g_free (filename); return FALSE; } g_free (filename); return TRUE; } #endif /* HAVE_ISO_CODES */ static void ensure_iso_codes_initialised (void) { static gboolean initialised = FALSE; if (initialised == TRUE) { return; } initialised = TRUE; #if defined (ENABLE_NLS) && defined (HAVE_ISO_CODES) bindtextdomain (ISO_639_DOMAIN, ISOCODESLOCALEDIR); bind_textdomain_codeset (ISO_639_DOMAIN, "UTF-8"); bindtextdomain(ISO_3166_DOMAIN, ISOCODESLOCALEDIR); bind_textdomain_codeset (ISO_3166_DOMAIN, "UTF-8"); #endif iso_639_table = g_hash_table_new_full (g_str_hash, g_str_equal, (GDestroyNotify) xmlFree, (GDestroyNotify) xmlFree); iso_3166_table = g_hash_table_new_full (g_str_hash, g_str_equal, (GDestroyNotify) g_free, (GDestroyNotify) xmlFree); #ifdef HAVE_ISO_CODES load_iso_entries (639, (GFunc) read_iso_639_entry, iso_639_table); load_iso_entries (3166, (GFunc) read_iso_3166_entry, iso_3166_table); #endif } static char * get_iso_name_for_lang_code (const char *code) { char **str; char *name = NULL; const char *langname, *localename; int len; str = g_strsplit (code, "_", -1); /* count the entries */ for (len = 0; str[len]; len++ ) /* empty */; g_return_val_if_fail (len != 0, NULL); langname = (const char *) g_hash_table_lookup (iso_639_table, str[0]); if (len == 1 && langname != NULL) { name = g_strdup (dgettext (ISO_639_DOMAIN, langname)); } else if (len == 2 && langname != NULL) { localename = (const char *) g_hash_table_lookup (iso_3166_table, str[1]); if (localename != NULL) { /* translators: the first %s is the language name, and the * second %s is the locale name. Example: * "French (France) * * Also: The text before the "|" is context to help you decide on * the correct translation. You MUST OMIT it in the translated string. */ name = g_strdup_printf (Q_("language|%s (%s)"), dgettext (ISO_639_DOMAIN, langname), dgettext (ISO_3166_DOMAIN, localename)); } else { name = g_strdup_printf (Q_("language|%s (%s)"), dgettext (ISO_639_DOMAIN, langname), str[1]); } } g_strfreev (str); return name; } /** * gtkspell_iso_codes_lookup_name_for_code: * @code: A language code, e.g. en_CA * * Looks up a name to display to the user for a language code, * this might use the iso-codes package if support was compiled * in, and it is available * * Returns: the UTF-8 string to display to the user, or NULL if * a name for the code could not be found */ char * gtkspell_iso_codes_lookup_name_for_code (const char *code) { char * lcode; char * ret; g_return_val_if_fail (code != NULL, NULL); ensure_iso_codes_initialised (); lcode = g_ascii_strdown (code, -1); ret = get_iso_name_for_lang_code (lcode); g_free (lcode); return ret; }