301 lines
7.4 KiB
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;
|
|
}
|