JSON output in controller.
This commit is contained in:
parent
6af2d1cfc6
commit
cddbab6312
3 changed files with 385 additions and 15 deletions
|
@ -14,6 +14,8 @@ typedef struct nxt_conf_json_value_s nxt_conf_json_value_t;
|
|||
|
||||
nxt_conf_json_value_t *nxt_conf_json_parse(nxt_buf_mem_t *b,
|
||||
nxt_mem_pool_t *pool);
|
||||
nxt_buf_t *nxt_conf_json_print(nxt_conf_json_value_t *value,
|
||||
nxt_mem_pool_t *pool);
|
||||
|
||||
|
||||
#endif /* _NXT_CONF_INCLUDED_ */
|
||||
|
|
|
@ -69,6 +69,20 @@ static u_char *nxt_conf_json_parse_number(u_char *pos, u_char *end,
|
|||
nxt_conf_json_value_t *value, nxt_mem_pool_t *pool);
|
||||
|
||||
|
||||
static uintptr_t nxt_conf_json_print_value(u_char *pos,
|
||||
nxt_conf_json_value_t *value);
|
||||
static uintptr_t nxt_conf_json_print_integer(u_char *pos,
|
||||
nxt_conf_json_value_t *value);
|
||||
static uintptr_t nxt_conf_json_print_string(u_char *pos,
|
||||
nxt_conf_json_value_t *value);
|
||||
static uintptr_t nxt_conf_json_print_array(u_char *pos,
|
||||
nxt_conf_json_value_t *value);
|
||||
static uintptr_t nxt_conf_json_print_object(u_char *pos,
|
||||
nxt_conf_json_value_t *value);
|
||||
|
||||
static uintptr_t nxt_conf_json_escape(u_char *dst, u_char *src, size_t size);
|
||||
|
||||
|
||||
static const nxt_lvlhsh_proto_t nxt_conf_json_object_hash_proto
|
||||
nxt_aligned(64) =
|
||||
{
|
||||
|
@ -774,3 +788,318 @@ nxt_conf_json_parse_number(u_char *pos, u_char *end,
|
|||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
nxt_buf_t *
|
||||
nxt_conf_json_print(nxt_conf_json_value_t *value, nxt_mem_pool_t *pool)
|
||||
{
|
||||
size_t size;
|
||||
nxt_buf_t *b;
|
||||
|
||||
size = nxt_conf_json_print_value(NULL, value);
|
||||
|
||||
b = nxt_buf_mem_alloc(pool, size, 0);
|
||||
if (nxt_slow_path(b == NULL)) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
b->mem.free = (u_char *) nxt_conf_json_print_value(b->mem.free, value);
|
||||
|
||||
return b;
|
||||
}
|
||||
|
||||
|
||||
static uintptr_t
|
||||
nxt_conf_json_print_value(u_char *pos, nxt_conf_json_value_t *value)
|
||||
{
|
||||
static const u_char null[4] = "null";
|
||||
static const u_char true[4] = "true";
|
||||
static const u_char false[5] = "false";
|
||||
|
||||
switch (value->type) {
|
||||
|
||||
case NXT_CONF_JSON_NULL:
|
||||
|
||||
if (pos == NULL) {
|
||||
return sizeof(null);
|
||||
}
|
||||
|
||||
return (uintptr_t) nxt_cpymem(pos, null, sizeof(null));
|
||||
|
||||
case NXT_CONF_JSON_BOOLEAN:
|
||||
|
||||
if (pos == NULL) {
|
||||
return value->u.boolean ? 4 : 5;
|
||||
}
|
||||
|
||||
if (value->u.boolean) {
|
||||
return (uintptr_t) nxt_cpymem(pos, true, sizeof(true));
|
||||
}
|
||||
|
||||
return (uintptr_t) nxt_cpymem(pos, false, sizeof(false));
|
||||
|
||||
case NXT_CONF_JSON_INTEGER:
|
||||
return nxt_conf_json_print_integer(pos, value);
|
||||
|
||||
case NXT_CONF_JSON_NUMBER:
|
||||
/* TODO */
|
||||
return (pos == NULL) ? 0 : (uintptr_t) pos;
|
||||
|
||||
case NXT_CONF_JSON_SHORT_STRING:
|
||||
case NXT_CONF_JSON_STRING:
|
||||
return nxt_conf_json_print_string(pos, value);
|
||||
|
||||
case NXT_CONF_JSON_ARRAY:
|
||||
return nxt_conf_json_print_array(pos, value);
|
||||
|
||||
case NXT_CONF_JSON_OBJECT:
|
||||
return nxt_conf_json_print_object(pos, value);
|
||||
}
|
||||
|
||||
nxt_unreachable();
|
||||
|
||||
return (pos == NULL) ? 0 : (uintptr_t) pos;
|
||||
}
|
||||
|
||||
|
||||
static uintptr_t
|
||||
nxt_conf_json_print_integer(u_char *pos, nxt_conf_json_value_t *value)
|
||||
{
|
||||
int64_t num;
|
||||
|
||||
num = value->u.integer;
|
||||
|
||||
if (pos == NULL) {
|
||||
num = llabs(num);
|
||||
|
||||
if (num <= 9999) {
|
||||
return sizeof("-9999") - 1;
|
||||
}
|
||||
|
||||
if (num <= 99999999999) {
|
||||
return sizeof("-99999999999") - 1;
|
||||
}
|
||||
|
||||
return NXT_INT64_T_LEN;
|
||||
}
|
||||
|
||||
return (uintptr_t) nxt_sprintf(pos, pos + NXT_INT64_T_LEN, "%L", num);
|
||||
}
|
||||
|
||||
|
||||
static uintptr_t
|
||||
nxt_conf_json_print_string(u_char *pos, nxt_conf_json_value_t *value)
|
||||
{
|
||||
size_t len;
|
||||
u_char *s;
|
||||
|
||||
if (value->type == NXT_CONF_JSON_SHORT_STRING) {
|
||||
len = value->u.str[0];
|
||||
s = &value->u.str[1];
|
||||
|
||||
} else {
|
||||
len = value->u.string->length;
|
||||
s = value->u.string->start;
|
||||
}
|
||||
|
||||
if (pos == NULL) {
|
||||
return 2 + len + nxt_conf_json_escape(NULL, s, len);
|
||||
}
|
||||
|
||||
*pos++ = '"';
|
||||
|
||||
pos = (u_char *) nxt_conf_json_escape(pos, s, len);
|
||||
|
||||
*pos++ = '"';
|
||||
|
||||
return (uintptr_t) pos;
|
||||
}
|
||||
|
||||
|
||||
static uintptr_t
|
||||
nxt_conf_json_print_array(u_char *pos, nxt_conf_json_value_t *value)
|
||||
{
|
||||
size_t len;
|
||||
uint32_t n;
|
||||
nxt_array_t *array;
|
||||
|
||||
array = value->u.array;
|
||||
|
||||
if (pos == NULL) {
|
||||
len = 2;
|
||||
|
||||
value = array->elts;
|
||||
|
||||
for (n = 0; n < array->nelts; n++) {
|
||||
len += nxt_conf_json_print_value(NULL, &value[n]);
|
||||
}
|
||||
|
||||
return n + len;
|
||||
}
|
||||
|
||||
*pos++ = '[';
|
||||
|
||||
if (array->nelts != 0) {
|
||||
value = array->elts;
|
||||
|
||||
pos = (u_char *) nxt_conf_json_print_value(pos, &value[0]);
|
||||
|
||||
for (n = 1; n < array->nelts; n++) {
|
||||
*pos++ = ',';
|
||||
pos = (u_char *) nxt_conf_json_print_value(pos, &value[n]);
|
||||
}
|
||||
}
|
||||
|
||||
*pos++ = ']';
|
||||
|
||||
return (uintptr_t) pos;
|
||||
}
|
||||
|
||||
|
||||
static uintptr_t
|
||||
nxt_conf_json_print_object(u_char *pos, nxt_conf_json_value_t *value)
|
||||
{
|
||||
size_t len;
|
||||
nxt_lvlhsh_t *object;
|
||||
nxt_lvlhsh_each_t lhe;
|
||||
nxt_conf_json_obj_member_t *member;
|
||||
|
||||
nxt_memzero(&lhe, sizeof(nxt_lvlhsh_each_t));
|
||||
|
||||
lhe.proto = &nxt_conf_json_object_hash_proto;
|
||||
|
||||
object = value->u.object;
|
||||
|
||||
if (pos == NULL) {
|
||||
len = 2;
|
||||
|
||||
for ( ;; ) {
|
||||
member = nxt_lvlhsh_each(object, &lhe);
|
||||
|
||||
if (member == NULL) {
|
||||
break;
|
||||
}
|
||||
|
||||
len += nxt_conf_json_print_string(NULL, &member->name) + 1
|
||||
+ nxt_conf_json_print_value(NULL, &member->value) + 1;
|
||||
}
|
||||
|
||||
return len;
|
||||
}
|
||||
|
||||
*pos++ = '{';
|
||||
|
||||
member = nxt_lvlhsh_each(object, &lhe);
|
||||
|
||||
if (member != NULL) {
|
||||
|
||||
for ( ;; ) {
|
||||
pos = (u_char *) nxt_conf_json_print_string(pos, &member->name);
|
||||
|
||||
*pos++ = ':';
|
||||
|
||||
pos = (u_char *) nxt_conf_json_print_value(pos, &member->value);
|
||||
|
||||
member = nxt_lvlhsh_each(object, &lhe);
|
||||
|
||||
if (member == NULL) {
|
||||
break;
|
||||
}
|
||||
|
||||
*pos++ = ',';
|
||||
}
|
||||
}
|
||||
|
||||
*pos++ = '}';
|
||||
|
||||
return (uintptr_t) pos;
|
||||
}
|
||||
|
||||
|
||||
static uintptr_t
|
||||
nxt_conf_json_escape(u_char *dst, u_char *src, size_t size)
|
||||
{
|
||||
u_char ch;
|
||||
size_t len;
|
||||
|
||||
if (dst == NULL) {
|
||||
len = 0;
|
||||
|
||||
while (size) {
|
||||
ch = *src++;
|
||||
|
||||
if (ch == '\\' || ch == '"') {
|
||||
len++;
|
||||
|
||||
} else if (ch <= 0x1f) {
|
||||
|
||||
switch (ch) {
|
||||
case '\n':
|
||||
case '\r':
|
||||
case '\t':
|
||||
case '\b':
|
||||
case '\f':
|
||||
len++;
|
||||
break;
|
||||
|
||||
default:
|
||||
len += sizeof("\\u001F") - 2;
|
||||
}
|
||||
}
|
||||
|
||||
size--;
|
||||
}
|
||||
|
||||
return len;
|
||||
}
|
||||
|
||||
while (size) {
|
||||
ch = *src++;
|
||||
|
||||
if (ch > 0x1f) {
|
||||
|
||||
if (ch == '\\' || ch == '"') {
|
||||
*dst++ = '\\';
|
||||
}
|
||||
|
||||
*dst++ = ch;
|
||||
|
||||
} else {
|
||||
*dst++ = '\\';
|
||||
|
||||
switch (ch) {
|
||||
case '\n':
|
||||
*dst++ = 'n';
|
||||
break;
|
||||
|
||||
case '\r':
|
||||
*dst++ = 'r';
|
||||
break;
|
||||
|
||||
case '\t':
|
||||
*dst++ = 't';
|
||||
break;
|
||||
|
||||
case '\b':
|
||||
*dst++ = 'b';
|
||||
break;
|
||||
|
||||
case '\f':
|
||||
*dst++ = 'f';
|
||||
break;
|
||||
|
||||
default:
|
||||
*dst++ = 'u'; *dst++ = '0'; *dst++ = '0';
|
||||
*dst++ = '0' + (ch >> 4);
|
||||
|
||||
ch &= 0xf;
|
||||
|
||||
*dst++ = (ch < 10) ? ('0' + ch) : ('A' + ch - 10);
|
||||
}
|
||||
}
|
||||
|
||||
size--;
|
||||
}
|
||||
|
||||
return (uintptr_t) dst;
|
||||
}
|
||||
|
|
|
@ -14,6 +14,8 @@
|
|||
typedef struct {
|
||||
nxt_http_request_parse_t parser;
|
||||
size_t length;
|
||||
|
||||
nxt_conf_json_value_t *conf;
|
||||
} nxt_controller_request_t;
|
||||
|
||||
|
||||
|
@ -41,7 +43,9 @@ static nxt_int_t nxt_controller_request_content_length(void *ctx,
|
|||
static void nxt_controller_process_request(nxt_task_t *task,
|
||||
nxt_event_conn_t *c, nxt_controller_request_t *r);
|
||||
static nxt_int_t nxt_controller_request_body_parse(nxt_task_t *task,
|
||||
nxt_event_conn_t *c);
|
||||
nxt_event_conn_t *c, nxt_controller_request_t *r);
|
||||
static void nxt_controller_conf_output(nxt_task_t *task, nxt_event_conn_t *c,
|
||||
nxt_controller_request_t *r);
|
||||
|
||||
|
||||
static nxt_http_fields_t nxt_controller_request_fields[] = {
|
||||
|
@ -513,29 +517,28 @@ static void
|
|||
nxt_controller_process_request(nxt_task_t *task, nxt_event_conn_t *c,
|
||||
nxt_controller_request_t *r)
|
||||
{
|
||||
nxt_str_t response;
|
||||
nxt_buf_t *b;
|
||||
|
||||
static const u_char response_good[]
|
||||
= "HTTP/1.0 200 OK\r\n\r\ndone\r\n";
|
||||
static const nxt_str_t resp418
|
||||
= nxt_string("HTTP/1.0 418 I'm a teapot\r\n\r\nerror\r\n");
|
||||
|
||||
static const u_char response_bad[]
|
||||
= "HTTP/1.0 418 I'm a teapot\r\n\r\nerror\r\n";
|
||||
|
||||
if (nxt_controller_request_body_parse(task, c) == NXT_OK) {
|
||||
nxt_str_set(&response, response_good);
|
||||
|
||||
} else {
|
||||
nxt_str_set(&response, response_bad);
|
||||
if (nxt_controller_request_body_parse(task, c, r) != NXT_OK) {
|
||||
goto error;
|
||||
}
|
||||
|
||||
b = nxt_buf_mem_alloc(c->mem_pool, response.length, 0);
|
||||
nxt_controller_conf_output(task, c, r);
|
||||
|
||||
return;
|
||||
|
||||
error:
|
||||
|
||||
b = nxt_buf_mem_alloc(c->mem_pool, resp418.length, 0);
|
||||
if (nxt_slow_path(b == NULL)) {
|
||||
nxt_controller_conn_close(task, c, r);
|
||||
return;
|
||||
}
|
||||
|
||||
b->mem.free = nxt_cpymem(b->mem.free, response.start, response.length);
|
||||
b->mem.free = nxt_cpymem(b->mem.free, resp418.start, resp418.length);
|
||||
|
||||
c->write = b;
|
||||
c->write_state = &nxt_controller_conn_write_state;
|
||||
|
@ -545,7 +548,8 @@ nxt_controller_process_request(nxt_task_t *task, nxt_event_conn_t *c,
|
|||
|
||||
|
||||
static nxt_int_t
|
||||
nxt_controller_request_body_parse(nxt_task_t *task, nxt_event_conn_t *c)
|
||||
nxt_controller_request_body_parse(nxt_task_t *task, nxt_event_conn_t *c,
|
||||
nxt_controller_request_t *r)
|
||||
{
|
||||
nxt_buf_t *b;
|
||||
nxt_conf_json_value_t *value;
|
||||
|
@ -558,5 +562,40 @@ nxt_controller_request_body_parse(nxt_task_t *task, nxt_event_conn_t *c)
|
|||
return NXT_ERROR;
|
||||
}
|
||||
|
||||
r->conf = value;
|
||||
|
||||
return NXT_OK;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
nxt_controller_conf_output(nxt_task_t *task, nxt_event_conn_t *c,
|
||||
nxt_controller_request_t *r)
|
||||
{
|
||||
nxt_buf_t *b;
|
||||
|
||||
static const nxt_str_t head = nxt_string("HTTP/1.0 200 OK\r\n\r\n");
|
||||
|
||||
b = nxt_buf_mem_alloc(c->mem_pool, head.length, 0);
|
||||
if (nxt_slow_path(b == NULL)) {
|
||||
nxt_controller_conn_close(task, c, r);
|
||||
return;
|
||||
}
|
||||
|
||||
b->mem.free = nxt_cpymem(b->mem.free, head.start, head.length);
|
||||
|
||||
c->write = b;
|
||||
|
||||
b = nxt_conf_json_print(r->conf, c->mem_pool);
|
||||
|
||||
if (b == NULL) {
|
||||
nxt_controller_conn_close(task, c, r);
|
||||
return;
|
||||
}
|
||||
|
||||
c->write->next = b;
|
||||
c->write_state = &nxt_controller_conn_write_state;
|
||||
|
||||
nxt_event_conn_write(task->thread->engine, c);
|
||||
return;
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue