Browse Source

ckdb - direct code a json decode replacement for breakdown()

master
kanoi 10 years ago
parent
commit
9aa281539d
  1. 273
      src/ckdb.c
  2. 11
      src/ckdb.h

273
src/ckdb.c

@ -1233,9 +1233,10 @@ static enum cmd_values breakdown(K_TREE **trf_root, K_STORE **trf_store,
TRANSFER *transfer; TRANSFER *transfer;
K_TREE_CTX ctx[1]; K_TREE_CTX ctx[1];
K_ITEM *item; K_ITEM *item;
char *cmdptr, *idptr, *next, *eq; char *cmdptr, *idptr, *next, *eq, *end, *was;
char *data = NULL; char *data = NULL, *tmp;
bool noid = false; bool noid = false;
size_t siz;
*trf_root = NULL; *trf_root = NULL;
*trf_store = NULL; *trf_store = NULL;
@ -1284,122 +1285,192 @@ static enum cmd_values breakdown(K_TREE **trf_root, K_STORE **trf_store,
*trf_store = k_new_store(transfer_free); *trf_store = k_new_store(transfer_free);
next = data; next = data;
if (next && strncmp(next, JSON_TRANSFER, JSON_TRANSFER_LEN) == 0) { if (next && strncmp(next, JSON_TRANSFER, JSON_TRANSFER_LEN) == 0) {
json_t *json_data; // It's json
json_error_t err_val;
void *json_iter;
const char *json_key, *json_str;
json_t *json_value;
int json_typ;
size_t siz;
bool ok;
next += JSON_TRANSFER_LEN; next += JSON_TRANSFER_LEN;
json_data = json_loads(next, JSON_DISABLE_EOF_CHECK, &err_val); was = next;
if (!json_data) { while (*next == ' ')
/* This REALLY shouldn't ever get an error since the input next++;
* is a json generated string if (*next != JSON_BEGIN) {
* If that happens then dump lots of information */ LOGERR("JSON_BEGIN '%c' was: %.32s...",
char *text = safe_text(next); JSON_BEGIN, tmp = safe_text(was));
LOGERR("Json decode error from command: '%s' " free(tmp);
"json_err=(%d:%d:%d)%s:%s input='%s'", free(cmdptr);
cmd, err_val.line, err_val.column, return CMD_REPLY;
err_val.position, err_val.source, }
err_val.text, text); next++;
free(text); // while we have a new quoted name
while (*next == JSON_STR) {
was = next;
end = ++next;
// look for the end quote
while (*end && *end != JSON_STR) {
if (*(end++) == JSON_ESC)
end++;
}
if (!*end) {
LOGERR("JSON name no trailing '%c' was: %.32s...",
JSON_STR, tmp = safe_text(was));
free(tmp);
free(cmdptr);
return CMD_REPLY;
}
if (next == end) {
LOGERR("JSON zero length name was: %.32s...",
tmp = safe_text(was));
free(tmp);
free(cmdptr); free(cmdptr);
return CMD_REPLY; return CMD_REPLY;
} }
json_iter = json_object_iter(json_data); *(end++) = '\0';
K_WLOCK(transfer_free); K_WLOCK(transfer_free);
while (json_iter) {
json_key = json_object_iter_key(json_iter);
json_value = json_object_iter_value(json_iter);
item = k_unlink_head(transfer_free); item = k_unlink_head(transfer_free);
K_WUNLOCK(transfer_free);
DATA_TRANSFER(transfer, item); DATA_TRANSFER(transfer, item);
ok = true; STRNCPY(transfer->name, next);
json_typ = json_typeof(json_value); was = next = end;
switch (json_typ) { while (*next == ' ')
case JSON_STRING: next++;
json_str = json_string_value(json_value); // we have a name, now expect a value after it
siz = strlen(json_str); if (*next != JSON_VALUE) {
if (siz >= sizeof(transfer->svalue)) LOGERR("JSON_VALUE '%c' '%s' was: %.32s...",
transfer->mvalue = strdup(json_str); JSON_VALUE, transfer->name,
else { tmp = safe_text(was));
STRNCPY(transfer->svalue, json_str); free(tmp);
transfer->mvalue = transfer->svalue; free(cmdptr);
K_WLOCK(transfer_free);
k_add_head(transfer_free, item);
K_WUNLOCK(transfer_free);
return CMD_REPLY;
} }
break; was = ++next;
case JSON_REAL: while (*next == ' ')
snprintf(transfer->svalue, next++;
sizeof(transfer->svalue), if (*next == JSON_STR) {
"%f", json_real_value(json_value)); end = ++next;
transfer->mvalue = transfer->svalue; // A quoted value must have a terminating quote
break; while (*end && *end != JSON_STR) {
case JSON_INTEGER: if (*(end++) == JSON_ESC)
snprintf(transfer->svalue, end++;
sizeof(transfer->svalue), }
"%"PRId64, if (!*end) {
(int64_t)json_integer_value(json_value)); LOGERR("JSON '%s' value was: %.32s...",
transfer->mvalue = transfer->svalue; transfer->name,
break; tmp = safe_text(was));
case JSON_TRUE: free(tmp);
case JSON_FALSE: free(cmdptr);
snprintf(transfer->svalue, K_WLOCK(transfer_free);
sizeof(transfer->svalue), k_add_head(transfer_free, item);
"%s", (json_typ == JSON_TRUE) ? K_WUNLOCK(transfer_free);
TRUE_STR : FALSE_STR); return CMD_REPLY;
transfer->mvalue = transfer->svalue;
break;
case JSON_ARRAY:
{
/* only one level array of strings for now (merkletree)
* ignore other data */
size_t i, len, off, count = json_array_size(json_value);
json_t *json_element;
bool first = true;
APPEND_REALLOC_INIT(transfer->mvalue, off, len);
for (i = 0; i < count; i++) {
json_element = json_array_get(json_value, i);
if (json_is_string(json_element)) {
json_str = json_string_value(json_element);
siz = strlen(json_str);
if (first)
first = false;
else {
APPEND_REALLOC(transfer->mvalue,
off, len, " ");
} }
APPEND_REALLOC(transfer->mvalue, if (end <= next+1) {
off, len, json_str); LOGERR("JSON '%s' zero length value "
} else "was: %.32s...",
LOGERR("%s() unhandled json type %d in array %s" transfer->name,
" in cmd %s", __func__, tmp = safe_text(was));
json_typ, json_key, cmd); free(tmp);
free(cmdptr);
K_WLOCK(transfer_free);
k_add_head(transfer_free, item);
K_WUNLOCK(transfer_free);
return CMD_REPLY;
} }
siz = end - next;
end++;
} else if (*next == JSON_ARRAY) {
// Only merklehash for now
if (strcmp(transfer->name, "merklehash")) {
LOGERR("JSON '%s' can't be an array",
transfer->name);
free(cmdptr);
K_WLOCK(transfer_free);
k_add_head(transfer_free, item);
K_WUNLOCK(transfer_free);
return CMD_REPLY;
} }
break; end = ++next;
default: /* No structure testing for now since we
LOGERR("%s() unhandled json type %d in cmd %s", * don't expect merklehash to get it wrong,
__func__, json_typ, cmd); * and if it does, it will show up as some
ok = false; * other error anyway */
break; while (*end && *end != JSON_ARRAY_END)
end++;
if (end < next+1) {
LOGERR("JSON '%s' zero length value "
"was: %.32s...",
transfer->name,
tmp = safe_text(was));
free(tmp);
free(cmdptr);
K_WLOCK(transfer_free);
k_add_head(transfer_free, item);
K_WUNLOCK(transfer_free);
return CMD_REPLY;
} }
siz = end - next;
if (ok) end++;
STRNCPY(transfer->name, json_key); } else {
if (!ok || find_in_ktree(*trf_root, item, cmp_transfer, ctx)) { end = next;
if (transfer->mvalue != transfer->svalue) // A non quoted value ends on SEP, END or space
FREENULL(transfer->mvalue); while (*end && *end != JSON_SEP &&
*end != JSON_END && *end != ' ') {
end++;
}
if (!*end) {
LOGERR("JSON '%s' value was: %.32s...",
transfer->name,
tmp = safe_text(was));
free(tmp);
free(cmdptr);
K_WLOCK(transfer_free);
k_add_head(transfer_free, item); k_add_head(transfer_free, item);
K_WUNLOCK(transfer_free);
return CMD_REPLY;
}
if (next == end) {
LOGERR("JSON '%s' zero length value "
"was: %.32s...",
transfer->name,
tmp = safe_text(was));
free(tmp);
free(cmdptr);
K_WLOCK(transfer_free);
k_add_head(transfer_free, item);
K_WUNLOCK(transfer_free);
return CMD_REPLY;
}
siz = end - next;
}
if (siz >= sizeof(transfer->svalue)) {
transfer->mvalue = malloc(siz+1);
STRNCPYSIZ(transfer->mvalue, next, siz+1);
} else { } else {
STRNCPYSIZ(transfer->svalue, next, siz+1);
transfer->mvalue = transfer->svalue;
}
*trf_root = add_to_ktree(*trf_root, item, cmp_transfer); *trf_root = add_to_ktree(*trf_root, item, cmp_transfer);
k_add_head(*trf_store, item); k_add_head(*trf_store, item);
// find the separator then move to the next name
next = end;
while (*next == ' ')
next++;
if (*next == JSON_SEP) {
next++;
while (*next == ' ')
next++;
} }
json_iter = json_object_iter_next(json_data, json_iter);
} }
if (*next != JSON_END) {
LOGERR("JSON_END '%c' was: %.32s...",
JSON_END, tmp = safe_text(next));
free(tmp);
free(cmdptr);
K_WLOCK(transfer_free);
k_add_head(transfer_free, item);
K_WUNLOCK(transfer_free); K_WUNLOCK(transfer_free);
json_decref(json_data); return CMD_REPLY;
}
} else { } else {
K_WLOCK(transfer_free); K_WLOCK(transfer_free);
while (next && *next) { while (next && *next) {
@ -1433,8 +1504,10 @@ static enum cmd_values breakdown(K_TREE **trf_root, K_STORE **trf_store,
} }
if (ckdb_cmds[*which_cmds].createdate) { if (ckdb_cmds[*which_cmds].createdate) {
item = require_name(*trf_root, "createdate", 10, NULL, reply, sizeof(reply)); item = require_name(*trf_root, "createdate", 10, NULL, reply, sizeof(reply));
if (!item) if (!item) {
free(cmdptr);
return CMD_REPLY; return CMD_REPLY;
}
DATA_TRANSFER(transfer, item); DATA_TRANSFER(transfer, item);
txt_to_ctv("createdate", transfer->mvalue, cd, sizeof(*cd)); txt_to_ctv("createdate", transfer->mvalue, cd, sizeof(*cd));

11
src/ckdb.h

@ -52,7 +52,7 @@
#define DB_VLOCK "1" #define DB_VLOCK "1"
#define DB_VERSION "0.9.6" #define DB_VERSION "0.9.6"
#define CKDB_VERSION DB_VERSION"-0.841" #define CKDB_VERSION DB_VERSION"-0.900"
#define WHERE_FFL " - from %s %s() line %d" #define WHERE_FFL " - from %s %s() line %d"
#define WHERE_FFL_HERE __FILE__, __func__, __LINE__ #define WHERE_FFL_HERE __FILE__, __func__, __LINE__
@ -268,6 +268,15 @@ extern bool everyone_die;
#define JSON_TRANSFER "json=" #define JSON_TRANSFER "json="
#define JSON_TRANSFER_LEN (sizeof(JSON_TRANSFER)-1) #define JSON_TRANSFER_LEN (sizeof(JSON_TRANSFER)-1)
#define JSON_BEGIN '{'
// Arrays have limited support in breakdown()
#define JSON_ARRAY '['
#define JSON_ARRAY_END ']'
#define JSON_STR '"'
#define JSON_VALUE ':'
#define JSON_SEP ','
#define JSON_END '}'
#define JSON_ESC '\\'
// Methods for sharelog (common function for all) // Methods for sharelog (common function for all)
#define STR_WORKINFO "workinfo" #define STR_WORKINFO "workinfo"

Loading…
Cancel
Save