2
0
Fork 0
lighttpd2/src/main/value.c

301 lines
7.4 KiB
C

#include <lighttpd/base.h>
liValue* li_value_new_none() {
liValue *v = g_slice_new0(liValue);
v->type = LI_VALUE_NONE;
return v;
}
liValue* li_value_new_bool(gboolean val) {
liValue *v = g_slice_new0(liValue);
v->data.boolean = val;
v->type = LI_VALUE_BOOLEAN;
return v;
}
liValue* li_value_new_number(gint64 val) {
liValue *v = g_slice_new0(liValue);
v->data.number = val;
v->type = LI_VALUE_NUMBER;
return v;
}
liValue* li_value_new_string(GString *val) {
liValue *v = g_slice_new0(liValue);
v->data.string = val;
v->type = LI_VALUE_STRING;
return v;
}
liValue* li_value_new_list() {
liValue *v = g_slice_new0(liValue);
v->data.list = g_array_new(FALSE, TRUE, sizeof(liValue*));
v->type = LI_VALUE_LIST;
return v;
}
static void _value_hash_free_key(gpointer data) {
g_string_free((GString*) data, TRUE);
}
static void _value_hash_free_value(gpointer data) {
li_value_free((liValue*) data);
}
liValue* li_value_new_hash() {
liValue *v = g_slice_new0(liValue);
v->data.hash = g_hash_table_new_full(
(GHashFunc) g_string_hash, (GEqualFunc) g_string_equal,
_value_hash_free_key, _value_hash_free_value);
v->type = LI_VALUE_HASH;
return v;
}
liValue* li_value_new_action(liServer *srv, liAction *a) {
liValue *v = g_slice_new0(liValue);
v->data.val_action.srv = srv;
v->data.val_action.action = a;
v->type = LI_VALUE_ACTION;
return v;
}
liValue* li_value_new_condition(liServer *srv, liCondition *c) {
liValue *v = g_slice_new0(liValue);
v->data.val_cond.srv = srv;
v->data.val_cond.cond = c;
v->type = LI_VALUE_CONDITION;
return v;
}
liValue* li_value_copy(liValue* val) {
liValue *n;
switch (val->type) {
case LI_VALUE_NONE: n = li_value_new_bool(FALSE); n->type = LI_VALUE_NONE; return n; /* hack */
case LI_VALUE_BOOLEAN: return li_value_new_bool(val->data.boolean);
case LI_VALUE_NUMBER: return li_value_new_number(val->data.number);
case LI_VALUE_STRING: return li_value_new_string(g_string_new_len(GSTR_LEN(val->data.string)));
/* list: we have to copy every value in the list! */
case LI_VALUE_LIST:
n = li_value_new_list();
g_array_set_size(n->data.list, val->data.list->len);
for (guint i = 0; i < val->data.list->len; i++) {
g_array_index(n->data.list, liValue*, i) = li_value_copy(g_array_index(val->data.list, liValue*, i));
}
return n;
/* hash: iterate over hashtable, clone each value */
case LI_VALUE_HASH:
n = li_value_new_hash();
{
GHashTableIter iter;
gpointer k, v;
g_hash_table_iter_init(&iter, val->data.hash);
while (g_hash_table_iter_next(&iter, &k, &v))
g_hash_table_insert(n->data.hash, g_string_new_len(GSTR_LEN((GString*)k)), li_value_copy((liValue*)v));
}
return n;
case LI_VALUE_ACTION:
li_action_acquire(val->data.val_action.action);
n = li_value_new_action(val->data.val_action.srv, val->data.val_action.action);
return n;
case LI_VALUE_CONDITION:
li_condition_acquire(val->data.val_cond.cond);
n = li_value_new_condition(val->data.val_cond.srv, val->data.val_cond.cond);
return n;
}
return NULL;
}
void li_value_free(liValue* val) {
if (!val) return;
switch (val->type) {
case LI_VALUE_NONE:
case LI_VALUE_BOOLEAN:
case LI_VALUE_NUMBER:
/* Nothing to free */
break;
case LI_VALUE_STRING:
g_string_free(val->data.string, TRUE);
break;
case LI_VALUE_LIST:
li_value_list_free(val->data.list);
break;
case LI_VALUE_HASH:
g_hash_table_destroy(val->data.hash);
break;
case LI_VALUE_ACTION:
li_action_release(val->data.val_action.srv, val->data.val_action.action);
break;
case LI_VALUE_CONDITION:
li_condition_release(val->data.val_cond.srv, val->data.val_cond.cond);
break;
}
val->type = LI_VALUE_NONE;
g_slice_free(liValue, val);
}
const char* li_value_type_string(liValueType type) {
switch(type) {
case LI_VALUE_NONE:
return "none";
case LI_VALUE_BOOLEAN:
return "boolean";
case LI_VALUE_NUMBER:
return "number";
case LI_VALUE_STRING:
return "string";
case LI_VALUE_LIST:
return "list";
case LI_VALUE_HASH:
return "hash";
case LI_VALUE_ACTION:
return "action";
case LI_VALUE_CONDITION:
return "condition";
}
return "<unknown>";
}
void li_value_list_free(GArray *vallist) {
if (!vallist) return;
for (gsize i = 0; i < vallist->len; i++) {
li_value_free(g_array_index(vallist, liValue*, i));
}
g_array_free(vallist, TRUE);
}
GString *li_value_to_string(liValue *val) {
GString *str;
switch (val->type) {
case LI_VALUE_NONE:
return NULL;
case LI_VALUE_BOOLEAN:
str = g_string_new(val->data.boolean ? "true" : "false");
break;
case LI_VALUE_NUMBER:
str = g_string_sized_new(0);
g_string_printf(str, "%" G_GINT64_FORMAT, val->data.number);
break;
case LI_VALUE_STRING:
str = g_string_new_len(CONST_STR_LEN("\""));
g_string_append_len(str, GSTR_LEN(val->data.string));
g_string_append_c(str, '"');
break;
case LI_VALUE_LIST:
str = g_string_new_len(CONST_STR_LEN("("));
if (val->data.list->len) {
GString *tmp = li_value_to_string(g_array_index(val->data.list, liValue*, 0));
g_string_append(str, tmp->str);
g_string_free(tmp, TRUE);
for (guint i = 1; i < val->data.list->len; i++) {
tmp = li_value_to_string(g_array_index(val->data.list, liValue*, i));
g_string_append_len(str, CONST_STR_LEN(", "));
g_string_append(str, tmp->str);
g_string_free(tmp, TRUE);
}
}
g_string_append_c(str, ')');
break;
case LI_VALUE_HASH:
{
GHashTableIter iter;
gpointer k, v;
GString *tmp;
guint i = 0;
str = g_string_new_len(CONST_STR_LEN("["));
g_hash_table_iter_init(&iter, val->data.hash);
while (g_hash_table_iter_next(&iter, &k, &v)) {
if (i)
g_string_append_len(str, CONST_STR_LEN(", "));
tmp = li_value_to_string((liValue*)v);
g_string_append_len(str, GSTR_LEN((GString*)k));
g_string_append_len(str, CONST_STR_LEN(": "));
g_string_append_len(str, GSTR_LEN(tmp));
g_string_free(tmp, TRUE);
i++;
}
g_string_append_c(str, ']');
break;
}
case LI_VALUE_ACTION:
str = g_string_new_len(CONST_STR_LEN("<action>"));
break;
case LI_VALUE_CONDITION:
str = g_string_new_len(CONST_STR_LEN("<condition>"));
break;
default:
return NULL;
}
return str;
}
gpointer li_value_extract_ptr(liValue *val) {
gpointer ptr = NULL;
if (!val) return NULL;
switch (val->type) {
case LI_VALUE_NONE:
break;
case LI_VALUE_BOOLEAN:
break;
case LI_VALUE_NUMBER:
break;
case LI_VALUE_STRING:
ptr = val->data.string;
break;
case LI_VALUE_LIST:
ptr = val->data.list;
break;
case LI_VALUE_HASH:
ptr = val->data.hash;
break;
case LI_VALUE_ACTION:
ptr = val->data.val_action.action;
break;
case LI_VALUE_CONDITION:
ptr = val->data.val_action.action;
break;
}
val->type = LI_VALUE_NONE;
return ptr;
}
GString* li_value_extract_string(liValue *val) {
if (val->type != LI_VALUE_STRING) return NULL;
val->type = LI_VALUE_NONE;
return val->data.string;
}
GArray* li_value_extract_list(liValue *val) {
if (val->type != LI_VALUE_LIST) return NULL;
val->type = LI_VALUE_NONE;
return val->data.list;
}
GHashTable* li_value_extract_hash(liValue *val) {
if (val->type != LI_VALUE_HASH) return NULL;
val->type = LI_VALUE_NONE;
return val->data.hash;
}
liAction* li_value_extract_action(liValue *val) {
if (val->type != LI_VALUE_ACTION) return NULL;
val->type = LI_VALUE_NONE;
return val->data.val_action.action;
}
liCondition* li_value_extract_condition(liValue *val) {
if (val->type != LI_VALUE_CONDITION) return NULL;
val->type = LI_VALUE_NONE;
return val->data.val_cond.cond;
}