mirror of
https://github.com/tstack/lnav
synced 2024-11-03 23:15:38 +00:00
199 lines
5.7 KiB
C++
199 lines
5.7 KiB
C++
/**
|
|
* Copyright (c) 2014, Timothy Stack
|
|
*
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright notice, this
|
|
* list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
* * Neither the name of Timothy Stack nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY TIMOTHY STACK AND CONTRIBUTORS ''AS IS'' AND ANY
|
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY
|
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* @file drive_json_op.cc
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "json_op.hh"
|
|
|
|
static void printer(void *ctx, const char *numberVal, size_t numberLen)
|
|
{
|
|
log_perror(write(STDOUT_FILENO, numberVal, numberLen));
|
|
}
|
|
|
|
static int handle_start_map(void *ctx)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_map_open(gen);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_map_key(void *ctx, const unsigned char * key, size_t len)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_string(gen, key, len);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_end_map(void *ctx)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_map_close(gen);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_null(void *ctx)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_null(gen);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_boolean(void *ctx, int boolVal)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_bool(gen, boolVal);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_number(void *ctx, const char *numberVal, size_t numberLen)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_number(gen, numberVal, numberLen);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_string(void *ctx, const unsigned char * stringVal, size_t len)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_string(gen, stringVal, len);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_start_array(void *ctx)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_array_open(gen);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int handle_end_array(void *ctx)
|
|
{
|
|
json_op *jo = (json_op *)ctx;
|
|
yajl_gen gen = (yajl_gen)jo->jo_ptr_data;
|
|
|
|
yajl_gen_array_close(gen);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
int retval = EXIT_SUCCESS;
|
|
|
|
log_argv(argc, argv);
|
|
|
|
if (argc != 3) {
|
|
fprintf(stderr, "error: expecting operation and json-pointer\n");
|
|
retval = EXIT_FAILURE;
|
|
}
|
|
else if (strcmp(argv[1], "get") == 0) {
|
|
unsigned char buffer[1024];
|
|
json_ptr jptr(argv[2]);
|
|
json_op jo(jptr);
|
|
yajl_handle handle;
|
|
yajl_status status;
|
|
yajl_gen gen;
|
|
ssize_t rc;
|
|
|
|
gen = yajl_gen_alloc(NULL);
|
|
yajl_gen_config(gen, yajl_gen_print_callback, printer, NULL);
|
|
yajl_gen_config(gen, yajl_gen_beautify, true);
|
|
|
|
jo.jo_ptr_callbacks.yajl_start_map = handle_start_map;
|
|
jo.jo_ptr_callbacks.yajl_map_key = handle_map_key;
|
|
jo.jo_ptr_callbacks.yajl_end_map = handle_end_map;
|
|
jo.jo_ptr_callbacks.yajl_start_array = handle_start_array;
|
|
jo.jo_ptr_callbacks.yajl_end_array = handle_end_array;
|
|
jo.jo_ptr_callbacks.yajl_null = handle_null;
|
|
jo.jo_ptr_callbacks.yajl_boolean = handle_boolean;
|
|
jo.jo_ptr_callbacks.yajl_number = handle_number;
|
|
jo.jo_ptr_callbacks.yajl_string = handle_string;
|
|
jo.jo_ptr_data = gen;
|
|
|
|
handle = yajl_alloc(&json_op::ptr_callbacks, NULL, &jo);
|
|
while ((rc = read(STDIN_FILENO, buffer, sizeof(buffer))) > 0) {
|
|
status = yajl_parse(handle, buffer, rc);
|
|
if (status == yajl_status_error) {
|
|
fprintf(stderr, "error:cannot parse JSON input -- %s\n",
|
|
yajl_get_error(handle, 1, buffer, rc));
|
|
retval = EXIT_FAILURE;
|
|
break;
|
|
}
|
|
else if (status == yajl_status_client_canceled) {
|
|
fprintf(stderr, "client cancel\n");
|
|
break;
|
|
}
|
|
}
|
|
status = yajl_complete_parse(handle);
|
|
if (status == yajl_status_error) {
|
|
fprintf(stderr, "error:cannot parse JSON input -- %s\n",
|
|
yajl_get_error(handle, 1, buffer, rc));
|
|
retval = EXIT_FAILURE;
|
|
}
|
|
else if (status == yajl_status_client_canceled) {
|
|
fprintf(stderr, "client cancel\n");
|
|
}
|
|
yajl_free(handle);
|
|
}
|
|
else {
|
|
fprintf(stderr, "error: unknown operation -- %s\n", argv[1]);
|
|
retval = EXIT_FAILURE;
|
|
}
|
|
|
|
return retval;
|
|
}
|