medit/moo/plugins/usertools/mookeyfile.c
2010-11-07 01:20:45 -08:00

967 lines
24 KiB
C

/*
* mookeyfile.c
*
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@sourceforge.net>
*
* This file is part of medit. medit is free software; you can
* redistribute it and/or modify it under the terms of the
* GNU Lesser General Public License as published by the
* Free Software Foundation; either version 2.1 of the License,
* or (at your option) any later version.
*
* You should have received a copy of the GNU Lesser General Public
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
*/
#include "mookeyfile.h"
#include "mooutils/mooutils-gobject.h"
#include "mooutils/mooutils-misc.h"
#include "mooutils/mootype-macros.h"
#include "mooedit/mooedit-tests.h"
#include <string.h>
typedef enum {
MOO_KEY_FILE_ERROR_PARSE,
MOO_KEY_FILE_ERROR_FILE
} MooKeyFileError;
struct _MooKeyFile {
guint ref_count;
GQueue *items;
};
struct _MooKeyFileItem {
MooKeyFile *key_file;
char *name;
char *content;
GHashTable *keys;
};
typedef struct {
const char *ptr;
gsize len;
const char *file;
guint line_no;
GError *error;
MooKeyFile *key_file;
} Parser;
#define CHAR_IS_SPACE(c__) ((c__) == ' ' || (c__) == '\t')
#define MOO_KEY_FILE_ERROR (moo_key_file_error_quark ())
MOO_DEFINE_QUARK_STATIC (moo-key-file-error, moo_key_file_error_quark)
MOO_DEFINE_BOXED_TYPE_R (MooKeyFile, moo_key_file)
MOO_DEFINE_POINTER_TYPE (MooKeyFileItem, moo_key_file_item)
static gboolean moo_key_file_parse_file (MooKeyFile *key_file,
const char *filename,
GError **error);
static MooKeyFileItem *key_file_item_new (MooKeyFile *key_file,
const char *name);
static void key_file_item_free (MooKeyFileItem *item);
static void key_file_item_take_content (MooKeyFileItem *item,
char *content);
static void get_line (const char *line,
gsize max_len,
gsize *line_len,
gsize *next_line);
static gboolean line_is_empty_or_comment (const char *line,
gsize line_len);
static gboolean line_is_indented (const char *line,
gsize line_len);
static gboolean line_is_item (const char *line,
gsize line_len,
char **item_name);
static gboolean parse_item (Parser *parser,
MooKeyFileItem *item);
static void parser_next_line (Parser *parser,
gsize offset);
static gboolean find_and_parse_item (Parser *parser);
static gboolean parse_content (Parser *parser,
MooKeyFileItem *item);
static gboolean line_is_key_val (const char *line,
gsize line_len,
char **key,
char **val);
static guint get_indent (const char *line,
gsize line_len);
static gboolean line_is_blank (const char *line,
gsize line_len,
guint *indent);
static void
get_line (const char *line,
gsize max_len,
gsize *line_len,
gsize *next_line)
{
gsize i;
for (i = 0; i < max_len; ++i)
{
if (line[i] == '\n')
{
*line_len = i;
*next_line = i + 1;
break;
}
else if (line[i] == '\r')
{
*line_len = i;
if (i + 1 < max_len && line[i+1] == '\n')
*next_line = i + 2;
else
*next_line = i + 1;
break;
}
}
if (i == max_len)
{
*line_len = max_len;
*next_line = max_len;
}
else
{
*next_line = MIN (*next_line, max_len);
}
}
static gboolean
line_is_empty_or_comment (const char *line,
gsize line_len)
{
return line_len == 0 || line[0] == '#';
}
static guint
get_indent (const char *line,
gsize line_len)
{
guint i;
for (i = 0; i < line_len; ++i)
if (!CHAR_IS_SPACE (line[i]))
break;
return i;
}
static gboolean
line_is_indented (const char *line,
gsize line_len)
{
return line_len != 0 && CHAR_IS_SPACE (*line);
}
static gboolean
line_is_blank (const char *line,
gsize line_len,
guint *indent)
{
guint i;
*indent = line_len;
for (i = 0; i < line_len; ++i)
{
if (!CHAR_IS_SPACE (line[i]))
{
*indent = i;
return FALSE;
}
}
return TRUE;
}
static gboolean
line_is_item (const char *line,
gsize line_len,
char **item_name)
{
gsize i;
gsize bracket = line_len;
if (!line_len || line[0] != '[')
return FALSE;
for (i = 1; i < line_len && bracket == line_len; ++i)
if (line[i] == ']')
bracket = i;
if (bracket == line_len || bracket == 1)
return FALSE;
for (i = bracket + 1; i < line_len; ++i)
{
if (line[i] == '#')
break;
if (!CHAR_IS_SPACE (line[i]))
return FALSE;
}
*item_name = g_strndup (line + 1, bracket - 1);
return TRUE;
}
static gboolean
line_is_key_val (const char *line,
gsize line_len,
char **key_p,
char **val_p)
{
gsize i;
gsize equal = line_len;
char *key, *val;
for (i = 0; i < line_len && equal == line_len; ++i)
if (line[i] == '=')
equal = i;
if (equal == line_len || equal == 0)
return FALSE;
key = g_strstrip (g_strndup (line, equal));
val = g_strstrip (g_strndup (line + equal + 1, line_len - equal - 1));
if (!*key)
{
g_free (key);
g_free (val);
return FALSE;
}
*key_p = key;
*val_p = val;
return TRUE;
}
static void
parser_next_line (Parser *parser,
gsize offset)
{
g_assert (offset <= parser->len);
parser->ptr += offset;
parser->len -= offset;
parser->line_no += 1;
}
static gboolean
parse_content (Parser *parser,
MooKeyFileItem *item)
{
GString *content;
const char *line;
guint indent;
gsize line_len = 0;
gsize next_line = 0;
g_assert (parser->len > 0);
content = g_string_new (NULL);
line = parser->ptr;
get_line (line, parser->len, &line_len, &next_line);
indent = get_indent (line, line_len);
g_assert (indent > 0);
g_string_append_len (content, line + indent, line_len - indent);
parser_next_line (parser, next_line);
while (parser->len)
{
guint indent_here;
line = parser->ptr;
get_line (line, parser->len, &line_len, &next_line);
line_is_blank (line, line_len, &indent_here);
if (!indent_here)
break;
if (indent_here < indent)
{
char *text = g_strndup (line, line_len);
g_set_error (&parser->error, MOO_KEY_FILE_ERROR,
MOO_KEY_FILE_ERROR_PARSE,
"wrong indentation in file %s at line %u: %s",
parser->file, parser->line_no, text);
g_free (text);
goto error;
}
g_string_append (content, "\n");
g_string_append_len (content, line + indent, line_len - indent);
parser_next_line (parser, next_line);
}
key_file_item_take_content (item, g_string_free (content, FALSE));
return find_and_parse_item (parser);
error:
g_string_free (content, TRUE);
return FALSE;
}
static gboolean
parse_item (Parser *parser,
MooKeyFileItem *item)
{
while (parser->len)
{
gsize line_len = 0;
gsize next_line = 0;
const char *line;
char *item_name, *key, *val;
line = parser->ptr;
get_line (line, parser->len, &line_len, &next_line);
if (line_is_empty_or_comment (line, line_len))
{
parser_next_line (parser, next_line);
continue;
}
if (line_is_indented (line, line_len))
{
if (!parse_content (parser, item))
return FALSE;
else
return find_and_parse_item (parser);
}
if (line_is_item (line, line_len, &item_name))
{
item = moo_key_file_new_item (parser->key_file, item_name);
parser_next_line (parser, next_line);
g_free (item_name);
return parse_item (parser, item);
}
if (!line_is_key_val (line, line_len, &key, &val))
{
char *text = g_strndup (line, line_len);
g_set_error (&parser->error, MOO_KEY_FILE_ERROR,
MOO_KEY_FILE_ERROR_PARSE,
"unexpected text in file %s at line %u: %s",
parser->file, parser->line_no, text);
g_free (text);
return FALSE;
}
moo_key_file_item_set (item, key, val);
parser_next_line (parser, next_line);
g_free (key);
g_free (val);
}
return TRUE;
}
static gboolean
find_and_parse_item (Parser *parser)
{
while (parser->len)
{
gsize line_len = 0;
gsize next_line = 0;
const char *line;
char *item_name;
MooKeyFileItem *item;
line = parser->ptr;
get_line (line, parser->len, &line_len, &next_line);
if (line_is_empty_or_comment (line, line_len))
{
parser_next_line (parser, next_line);
continue;
}
if (line_is_indented (line, line_len))
{
g_set_error (&parser->error, MOO_KEY_FILE_ERROR,
MOO_KEY_FILE_ERROR_PARSE,
"unexpected indented block in file %s at line %u",
parser->file, parser->line_no);
return FALSE;
}
if (!line_is_item (line, line_len, &item_name))
{
char *text = g_strndup (line, line_len);
g_set_error (&parser->error, MOO_KEY_FILE_ERROR,
MOO_KEY_FILE_ERROR_PARSE,
"unexpected text in file %s at line %u: %s",
parser->file, parser->line_no, text);
g_free (text);
return FALSE;
}
item = moo_key_file_new_item (parser->key_file, item_name);
parser_next_line (parser, next_line);
g_free (item_name);
return parse_item (parser, item);
}
return TRUE;
}
static gboolean
parse_buffer (MooKeyFile *key_file,
const char *string,
gssize len,
const char *file,
GError **error)
{
Parser parser;
gboolean result;
g_return_val_if_fail (key_file != NULL, FALSE);
g_return_val_if_fail (string != NULL, FALSE);
if (len < 0)
len = strlen (string);
parser.key_file = key_file;
parser.error = NULL;
parser.len = len;
parser.ptr = string;
parser.line_no = 1;
parser.file = file ? file : "<memory>";
result = find_and_parse_item (&parser);
if (parser.error)
g_propagate_error (error, parser.error);
return result;
}
#if 0
static gboolean
moo_key_file_parse_buffer (MooKeyFile *key_file,
const char *string,
gssize len,
GError **error)
{
g_return_val_if_fail (key_file != NULL, FALSE);
g_return_val_if_fail (string != NULL, FALSE);
return parse_buffer (key_file, string, len, NULL, error);
}
#endif
MooKeyFile *
moo_key_file_ref (MooKeyFile *key_file)
{
g_return_val_if_fail (key_file != NULL, NULL);
key_file->ref_count++;
return key_file;
}
void
moo_key_file_unref (MooKeyFile *key_file)
{
g_return_if_fail (key_file != NULL);
if (--key_file->ref_count)
return;
g_queue_foreach (key_file->items, (GFunc) key_file_item_free, NULL);
g_queue_free (key_file->items);
g_free (key_file);
}
static MooKeyFileItem *
key_file_item_new (MooKeyFile *key_file,
const char *name)
{
MooKeyFileItem *item = g_new0 (MooKeyFileItem, 1);
item->key_file = key_file;
item->name = g_strdup (name);
item->content = NULL;
item->keys = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
return item;
}
static void
key_file_item_free (MooKeyFileItem *item)
{
if (item)
{
g_free (item->name);
g_free (item->content);
g_hash_table_destroy (item->keys);
g_free (item);
}
}
MooKeyFile *
moo_key_file_new (void)
{
MooKeyFile *key_file = g_new0 (MooKeyFile, 1);
key_file->ref_count = 1;
key_file->items = g_queue_new ();
return key_file;
}
MooKeyFile *
moo_key_file_new_from_file (const char *filename,
GError **error)
{
MooKeyFile *key_file;
g_return_val_if_fail (filename != NULL, NULL);
key_file = moo_key_file_new ();
if (!moo_key_file_parse_file (key_file, filename, error))
{
moo_key_file_unref (key_file);
key_file = NULL;
}
return key_file;
}
#if 0
MooKeyFile *
moo_key_file_new_from_buffer (const char *string,
gssize len,
GError **error)
{
MooKeyFile *key_file;
g_return_val_if_fail (string != NULL, NULL);
key_file = moo_key_file_new ();
if (!moo_key_file_parse_buffer (key_file, string, len, error))
{
moo_key_file_unref (key_file);
key_file = NULL;
}
return key_file;
}
#endif
static gboolean
moo_key_file_parse_file (MooKeyFile *key_file,
const char *filename,
GError **error)
{
char *contents;
gsize len;
gboolean result;
g_return_val_if_fail (key_file != NULL, FALSE);
g_return_val_if_fail (filename != NULL, FALSE);
if (!g_file_get_contents (filename, &contents, &len, error))
return FALSE;
result = parse_buffer (key_file, contents, len, filename, error);
g_free (contents);
return result;
}
guint
moo_key_file_n_items (MooKeyFile *key_file)
{
g_return_val_if_fail (key_file != NULL, 0);
return key_file->items->length;
}
MooKeyFileItem *
moo_key_file_nth_item (MooKeyFile *key_file,
guint n)
{
g_return_val_if_fail (key_file != NULL, NULL);
g_return_val_if_fail (n < key_file->items->length, NULL);
return g_queue_peek_nth (key_file->items, n);
}
MooKeyFileItem *
moo_key_file_new_item (MooKeyFile *key_file,
const char *name)
{
MooKeyFileItem *item;
g_return_val_if_fail (key_file != NULL, NULL);
item = key_file_item_new (key_file, name);
g_queue_push_nth (key_file->items, item, -1);
return item;
}
#if 0
void
moo_key_file_delete_item (MooKeyFile *key_file,
guint index)
{
MooKeyFileItem *item;
g_return_if_fail (key_file != NULL);
g_return_if_fail (index < key_file->items->length);
item = g_queue_pop_nth (key_file->items, index);
key_file_item_free (item);
}
#endif
const char *
moo_key_file_item_name (MooKeyFileItem *item)
{
g_return_val_if_fail (item != NULL, NULL);
return item->name;
}
const char *
moo_key_file_item_get (MooKeyFileItem *item,
const char *key)
{
g_return_val_if_fail (item != NULL, NULL);
g_return_val_if_fail (key != NULL, NULL);
return g_hash_table_lookup (item->keys, key);
}
char *
moo_key_file_item_steal (MooKeyFileItem *item,
const char *key)
{
gpointer orig_key, value;
g_return_val_if_fail (item != NULL, NULL);
g_return_val_if_fail (key != NULL, NULL);
if (!g_hash_table_lookup_extended (item->keys, key,
&orig_key, &value))
return NULL;
g_hash_table_steal (item->keys, key);
g_free (orig_key);
return value;
}
void
moo_key_file_item_set (MooKeyFileItem *item,
const char *key,
const char *value)
{
g_return_if_fail (item != NULL);
g_return_if_fail (key != NULL);
if (value)
g_hash_table_insert (item->keys, g_strdup (key), g_strdup (value));
else
g_hash_table_remove (item->keys, key);
}
#if 0
gboolean
moo_key_file_item_get_bool (MooKeyFileItem *item,
const char *key,
gboolean default_val)
{
const char *val;
g_return_val_if_fail (item != NULL, default_val);
g_return_val_if_fail (key != NULL, default_val);
val = moo_key_file_item_get (item, key);
if (!val)
return default_val;
else
return _moo_convert_string_to_bool (val, default_val);
}
#endif
gboolean
moo_key_file_item_steal_bool (MooKeyFileItem *item,
const char *key,
gboolean default_val)
{
char *val;
gboolean ret;
g_return_val_if_fail (item != NULL, default_val);
g_return_val_if_fail (key != NULL, default_val);
val = moo_key_file_item_steal (item, key);
if (!val)
return default_val;
ret = _moo_convert_string_to_bool (val, default_val);
g_free (val);
return ret;
}
void
moo_key_file_item_set_bool (MooKeyFileItem *item,
const char *key,
gboolean value)
{
g_return_if_fail (item != NULL);
g_return_if_fail (key != NULL);
moo_key_file_item_set (item, key, value ? "true" : "false");
}
void
moo_key_file_item_foreach (MooKeyFileItem *item,
GHFunc func,
gpointer data)
{
g_return_if_fail (item != NULL);
g_return_if_fail (func != NULL);
g_hash_table_foreach (item->keys, func, data);
}
const char *
moo_key_file_item_get_content (MooKeyFileItem *item)
{
g_return_val_if_fail (item != NULL, NULL);
return item->content;
}
char *
moo_key_file_item_steal_content (MooKeyFileItem *item)
{
char *ret;
g_return_val_if_fail (item != NULL, NULL);
ret = item->content;
item->content = NULL;
return ret;
}
static void
key_file_item_take_content (MooKeyFileItem *item,
char *content)
{
g_return_if_fail (item != NULL);
g_free (item->content);
item->content = content;
}
void
moo_key_file_item_set_content (MooKeyFileItem *item,
const char *content)
{
g_return_if_fail (item != NULL);
key_file_item_take_content (item, g_strdup (content));
}
static void
format_key (const char *key,
const char *value,
GString *string)
{
g_string_append_printf (string, "%s=%s\n", key, value);
}
static void
format_content (const char *content,
GString *string,
const char *indent)
{
const char *line;
gsize line_len;
MooLineReader lr;
for (moo_line_reader_init (&lr, content, -1);
(line = moo_line_reader_get_line (&lr, &line_len, NULL)); )
{
g_string_append (string, indent);
g_string_append_len (string, line, line_len);
g_string_append (string, "\n");
}
}
char *
moo_key_file_format (MooKeyFile *key_file,
const char *comment,
guint indent)
{
char *fill;
GList *l;
GString *string;
g_return_val_if_fail (key_file != NULL, NULL);
fill = g_strnfill (indent, ' ');
string = g_string_new (NULL);
if (comment)
{
MooLineReader lr;
const char *line;
gsize line_len;
for (moo_line_reader_init (&lr, comment, -1);
(line = moo_line_reader_get_line (&lr, &line_len, NULL)); )
{
g_string_append (string, "# ");
g_string_append_len (string, line, line_len);
g_string_append (string, "\n");
}
g_string_append (string, "\n");
}
for (l = key_file->items->head; l != NULL; l = l->next)
{
MooKeyFileItem *item = l->data;
g_string_append_printf (string, "[%s]\n", item->name);
g_hash_table_foreach (item->keys, (GHFunc) format_key, string);
if (item->content)
format_content (item->content, string, fill);
if (l->next)
g_string_append (string, "\n");
}
g_free (fill);
return g_string_free (string, FALSE);
}
#define CMT_BODY1 \
"A comment\n" \
"blah blah blah "
#define CMT1 \
"# A comment\n" \
"# blah blah blah \n" \
"\n"
#define KFS1 \
"[tool]\n" \
"id=InsertDateAndTime\n" \
"builtin=true\n" \
"\n" \
"[tool]\n" \
"id=SortLines\n" \
"type=exe\n" \
"name=Sort Lines\n" \
"input=lines\n" \
"options=need-doc\n" \
"output=insert\n" \
" sort | uniq\n" \
"\n" \
"[tool]\n" \
" code code code\n" \
" code code code \n" \
"\n" \
"[tool]\n" \
" code code code\n" \
" \n"
static void
test_load_kfs1 (gboolean load_with_comment,
gboolean format_with_comment)
{
MooKeyFile *key_file;
gboolean res;
GError *error = NULL;
const char *input = KFS1;
const char *output = KFS1;
const char *comment = NULL;
char *s;
if (load_with_comment)
input = CMT1 KFS1;
if (format_with_comment)
{
output = CMT1 KFS1;
comment = CMT_BODY1;
}
key_file = moo_key_file_new ();
res = parse_buffer (key_file, input, -1, "fake.cfg", &error);
if (!res)
TEST_FAILED_MSG ("failed to load key file content: %s",
error ? error->message : "<null>");
else
TEST_ASSERT (TRUE);
s = moo_key_file_format (key_file, comment, 2);
TEST_ASSERT_STR_EQ (s, output);
g_free (s);
moo_key_file_unref (key_file);
}
static void
test_load_kfs (void)
{
test_load_kfs1 (FALSE, FALSE);
test_load_kfs1 (FALSE, TRUE);
test_load_kfs1 (TRUE, FALSE);
test_load_kfs1 (TRUE, TRUE);
}
static void
test_key_file (void)
{
test_load_kfs ();
}
void
moo_test_key_file (void)
{
MooTestSuite *suite;
suite = moo_test_suite_new ("MooKeyFile", "MooKeyFile test suite", NULL, NULL, NULL);
moo_test_suite_add_test (suite, "MooKeyFile", "MooKeyFile test", (MooTestFunc) test_key_file, NULL);
}