finished wring a primitive standalone json parser
This commit is contained in:
parent
c6e4aa0702
commit
407b469ec2
223
lib/hcl-json.h
223
lib/hcl-json.h
@ -29,25 +29,28 @@
|
||||
|
||||
#include <hcl.h>
|
||||
|
||||
typedef struct hcl_jsoner_t hcl_jsoner_t;
|
||||
/**
|
||||
* The hcl_json_t type defines a simple json parser.
|
||||
*/
|
||||
typedef struct hcl_json_t hcl_json_t;
|
||||
|
||||
enum hcl_jsoner_option_t
|
||||
enum hcl_json_option_t
|
||||
{
|
||||
HCL_JSON_TRAIT,
|
||||
HCL_JSON_LOG_MASK,
|
||||
};
|
||||
typedef enum hcl_jsoner_option_t hcl_jsoner_option_t;
|
||||
typedef enum hcl_json_option_t hcl_json_option_t;
|
||||
|
||||
enum hcl_jsoner_trait_t
|
||||
enum hcl_json_trait_t
|
||||
{
|
||||
/* no trait defined at this moment. XXXX is just a placeholder */
|
||||
HCL_JSON_XXXX = (1 << 0)
|
||||
};
|
||||
typedef enum hcl_jsoner_trait_t hcl_jsoner_trait_t;
|
||||
typedef enum hcl_json_trait_t hcl_json_trait_t;
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
enum hcl_jsoner_state_t
|
||||
enum hcl_json_state_t
|
||||
{
|
||||
HCL_JSON_STATE_START,
|
||||
HCL_JSON_STATE_IN_ARRAY,
|
||||
@ -57,109 +60,46 @@ enum hcl_jsoner_state_t
|
||||
HCL_JSON_STATE_IN_NUMERIC_VALUE,
|
||||
HCL_JSON_STATE_IN_QUOTED_VALUE
|
||||
};
|
||||
typedef enum hcl_jsoner_state_t hcl_jsoner_state_t;
|
||||
typedef enum hcl_json_state_t hcl_json_state_t;
|
||||
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
#if 0
|
||||
struct hcl_jsoner_root_t
|
||||
enum hcl_json_inst_t
|
||||
{
|
||||
int type;
|
||||
hcl_jsoner_value_t* value;
|
||||
};
|
||||
HCL_JSON_INST_START_ARRAY,
|
||||
HCL_JSON_INST_END_ARRAY,
|
||||
HCL_JSON_INST_START_DIC,
|
||||
HCL_JSON_INST_END_DIC,
|
||||
|
||||
struct hcl_jsoner_list_t
|
||||
{
|
||||
int type; /* array or table */
|
||||
hcl_jsoner_pair_t* cell;
|
||||
};
|
||||
HCL_JSON_INST_KEY,
|
||||
|
||||
struct hcl_jsoner_value_t
|
||||
{
|
||||
int type; /* atom or pair */
|
||||
union
|
||||
{
|
||||
hcl_jsoner_value_t* value;
|
||||
hcl_jsoner_pair_t* cell;
|
||||
} u;
|
||||
HCL_JSON_INST_STRING,
|
||||
HCL_JSON_INST_NUMBER,
|
||||
HCL_JSON_INST_NIL,
|
||||
HCL_JSON_INST_TRUE,
|
||||
HCL_JSON_INST_FALSE,
|
||||
};
|
||||
typedef enum hcl_json_inst_t hcl_json_inst_t;
|
||||
|
||||
struct hcl_jsoner_atom_t
|
||||
{
|
||||
int type; /* string, word, number */
|
||||
};
|
||||
|
||||
struct hcl_jsoner_pair_t
|
||||
{
|
||||
hcl_jsoner_atom_t* key;
|
||||
hcl_jsoner_value_t* value;
|
||||
hcl_jsoner_pair_t* next;
|
||||
};
|
||||
#endif
|
||||
/* ========================================================================= */
|
||||
enum hcl_jsoner_reply_type_t
|
||||
{
|
||||
HCL_JSON_REPLY_TYPE_OK = 0,
|
||||
HCL_JSON_REPLY_TYPE_ERROR = 1
|
||||
};
|
||||
typedef enum hcl_jsoner_reply_type_t hcl_jsoner_reply_type_t;
|
||||
|
||||
typedef void (*hcl_jsoner_log_write_t) (
|
||||
hcl_jsoner_t* json,
|
||||
typedef void (*hcl_json_log_write_t) (
|
||||
hcl_json_t* json,
|
||||
unsigned int mask,
|
||||
const hcl_ooch_t* msg,
|
||||
hcl_oow_t len
|
||||
);
|
||||
|
||||
typedef int (*hcl_jsoner_start_reply_t) (
|
||||
hcl_jsoner_t* json,
|
||||
hcl_jsoner_reply_type_t type,
|
||||
const hcl_ooch_t* dptr,
|
||||
hcl_oow_t dlen
|
||||
typedef int (*hcl_json_instcb_t) (
|
||||
hcl_json_t* json,
|
||||
hcl_json_inst_t inst,
|
||||
const hcl_oocs_t* str
|
||||
);
|
||||
|
||||
typedef int (*hcl_jsoner_feed_attr_t) (
|
||||
hcl_jsoner_t* json,
|
||||
const hcl_oocs_t* key,
|
||||
const hcl_oocs_t* val
|
||||
);
|
||||
|
||||
typedef int (*hcl_jsoner_start_data_t) (
|
||||
hcl_jsoner_t* json
|
||||
);
|
||||
|
||||
typedef int (*hcl_jsoner_feed_data_t) (
|
||||
hcl_jsoner_t* json,
|
||||
const void* ptr,
|
||||
hcl_oow_t len
|
||||
);
|
||||
|
||||
typedef int (*hcl_jsoner_end_data_t) (
|
||||
hcl_jsoner_t* json
|
||||
);
|
||||
|
||||
enum hcl_jsoner_end_reply_state_t
|
||||
struct hcl_json_prim_t
|
||||
{
|
||||
HCL_JSON_END_REPLY_STATE_OK,
|
||||
HCL_JSON_END_REPLY_STATE_REVOKED
|
||||
hcl_json_log_write_t log_write;
|
||||
hcl_json_instcb_t instcb;
|
||||
};
|
||||
typedef enum hcl_jsoner_end_reply_state_t hcl_jsoner_end_reply_state_t;
|
||||
|
||||
typedef int (*hcl_jsoner_end_reply_t) (
|
||||
hcl_jsoner_t* json,
|
||||
hcl_jsoner_end_reply_state_t state
|
||||
);
|
||||
|
||||
struct hcl_jsoner_prim_t
|
||||
{
|
||||
hcl_jsoner_log_write_t log_write;
|
||||
|
||||
hcl_jsoner_start_reply_t start_reply; /* mandatory */
|
||||
hcl_jsoner_feed_attr_t feed_attr; /* optional */
|
||||
hcl_jsoner_feed_data_t feed_data; /* optional */
|
||||
hcl_jsoner_end_reply_t end_reply; /* mandatory */
|
||||
};
|
||||
typedef struct hcl_jsoner_prim_t hcl_jsoner_prim_t;
|
||||
typedef struct hcl_json_prim_t hcl_json_prim_t;
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
@ -167,127 +107,126 @@ typedef struct hcl_jsoner_prim_t hcl_jsoner_prim_t;
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
HCL_EXPORT hcl_jsoner_t* hcl_jsoner_open (
|
||||
HCL_EXPORT hcl_json_t* hcl_json_open (
|
||||
hcl_mmgr_t* mmgr,
|
||||
hcl_oow_t xtnsize,
|
||||
hcl_jsoner_prim_t* prim,
|
||||
hcl_json_prim_t* prim,
|
||||
hcl_errnum_t* errnum
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_close (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT void hcl_json_close (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_reset (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT void hcl_json_reset (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT int hcl_jsoner_feed (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT int hcl_json_feed (
|
||||
hcl_json_t* json,
|
||||
const void* ptr,
|
||||
hcl_oow_t len,
|
||||
hcl_oow_t* xlen
|
||||
);
|
||||
|
||||
HCL_EXPORT hcl_jsoner_state_t hcl_jsoner_getstate (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT hcl_json_state_t hcl_json_getstate (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT int hcl_jsoner_setoption (
|
||||
hcl_jsoner_t* json,
|
||||
hcl_jsoner_option_t id,
|
||||
HCL_EXPORT int hcl_json_setoption (
|
||||
hcl_json_t* json,
|
||||
hcl_json_option_t id,
|
||||
const void* value
|
||||
);
|
||||
|
||||
HCL_EXPORT int hcl_jsoner_getoption (
|
||||
hcl_jsoner_t* json,
|
||||
hcl_jsoner_option_t id,
|
||||
HCL_EXPORT int hcl_json_getoption (
|
||||
hcl_json_t* json,
|
||||
hcl_json_option_t id,
|
||||
void* value
|
||||
);
|
||||
|
||||
|
||||
HCL_EXPORT void* hcl_jsoner_getxtn (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT void* hcl_json_getxtn (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT hcl_mmgr_t* hcl_jsoner_getmmgr (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT hcl_mmgr_t* hcl_json_getmmgr (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT hcl_cmgr_t* hcl_jsoner_getcmgr (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT hcl_cmgr_t* hcl_json_getcmgr (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_setcmgr (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void hcl_json_setcmgr (
|
||||
hcl_json_t* json,
|
||||
hcl_cmgr_t* cmgr
|
||||
);
|
||||
|
||||
|
||||
HCL_EXPORT hcl_errnum_t hcl_jsoner_geterrnum (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT hcl_errnum_t hcl_json_geterrnum (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT const hcl_ooch_t* hcl_jsoner_geterrstr (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT const hcl_ooch_t* hcl_json_geterrstr (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT const hcl_ooch_t* hcl_jsoner_geterrmsg (
|
||||
hcl_jsoner_t* json
|
||||
HCL_EXPORT const hcl_ooch_t* hcl_json_geterrmsg (
|
||||
hcl_json_t* json
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_seterrnum (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void hcl_json_seterrnum (
|
||||
hcl_json_t* json,
|
||||
hcl_errnum_t errnum
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_seterrbfmt (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void hcl_json_seterrbfmt (
|
||||
hcl_json_t* json,
|
||||
hcl_errnum_t errnum,
|
||||
const hcl_bch_t* fmt,
|
||||
...
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_seterrufmt (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void hcl_json_seterrufmt (
|
||||
hcl_json_t* json,
|
||||
hcl_errnum_t errnum,
|
||||
const hcl_uch_t* fmt,
|
||||
...
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_logbfmt (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void hcl_json_logbfmt (
|
||||
hcl_json_t* json,
|
||||
unsigned int mask,
|
||||
const hcl_bch_t* fmt,
|
||||
...
|
||||
);
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_logufmt (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void hcl_json_logufmt (
|
||||
hcl_json_t* json,
|
||||
unsigned int mask,
|
||||
const hcl_uch_t* fmt,
|
||||
...
|
||||
);
|
||||
|
||||
HCL_EXPORT void* hcl_jsoner_allocmem (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void* hcl_json_allocmem (
|
||||
hcl_json_t* json,
|
||||
hcl_oow_t size
|
||||
);
|
||||
|
||||
HCL_EXPORT void* hcl_jsoner_callocmem (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void* hcl_json_callocmem (
|
||||
hcl_json_t* json,
|
||||
hcl_oow_t size
|
||||
);
|
||||
|
||||
HCL_EXPORT void* hcl_jsoner_reallocmem (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void* hcl_json_reallocmem (
|
||||
hcl_json_t* json,
|
||||
void* ptr,
|
||||
hcl_oow_t size
|
||||
);
|
||||
|
||||
|
||||
HCL_EXPORT void hcl_jsoner_freemem (
|
||||
hcl_jsoner_t* json,
|
||||
HCL_EXPORT void hcl_json_freemem (
|
||||
hcl_json_t* json,
|
||||
void* ptr
|
||||
);
|
||||
|
||||
|
535
lib/json.c
535
lib/json.c
@ -34,40 +34,47 @@
|
||||
|
||||
struct dummy_hcl_xtn_t
|
||||
{
|
||||
hcl_jsoner_t* json;
|
||||
hcl_json_t* json;
|
||||
};
|
||||
typedef struct dummy_hcl_xtn_t dummy_hcl_xtn_t;
|
||||
|
||||
enum hcl_jsoner_reply_attr_type_t
|
||||
|
||||
typedef struct hcl_json_state_node_t hcl_json_state_node_t;
|
||||
struct hcl_json_state_node_t
|
||||
{
|
||||
HCL_JSON_REPLY_ATTR_TYPE_UNKNOWN,
|
||||
HCL_JSON_REPLY_ATTR_TYPE_DATA
|
||||
};
|
||||
typedef enum hcl_jsoner_reply_attr_type_t hcl_jsoner_reply_attr_type_t;
|
||||
|
||||
|
||||
|
||||
typedef struct hcl_jsoner_state_node_t hcl_jsoner_state_node_t;
|
||||
struct hcl_jsoner_state_node_t
|
||||
{
|
||||
hcl_jsoner_state_t state;
|
||||
hcl_json_state_t state;
|
||||
union
|
||||
{
|
||||
struct
|
||||
{
|
||||
int got_value;
|
||||
} ia; /* in array */
|
||||
|
||||
struct
|
||||
{
|
||||
/* 0: ready to get key (at the beginning or got comma),
|
||||
* 1: got key, 2: got colon, 3: got value */
|
||||
int state;
|
||||
} id; /* in dictionary */
|
||||
struct
|
||||
{
|
||||
int escaped;
|
||||
int digit_count;
|
||||
hcl_ooch_t acc;
|
||||
} qv;
|
||||
struct
|
||||
{
|
||||
int dotted;
|
||||
} nv;
|
||||
} u;
|
||||
hcl_jsoner_state_node_t* next;
|
||||
hcl_json_state_node_t* next;
|
||||
};
|
||||
|
||||
struct hcl_jsoner_t
|
||||
struct hcl_json_t
|
||||
{
|
||||
hcl_mmgr_t* mmgr;
|
||||
hcl_cmgr_t* cmgr;
|
||||
hcl_jsoner_prim_t prim;
|
||||
hcl_json_prim_t prim;
|
||||
hcl_t* dummy_hcl;
|
||||
|
||||
hcl_errnum_t errnum;
|
||||
@ -83,15 +90,11 @@ struct hcl_jsoner_t
|
||||
unsigned int logmask;
|
||||
} cfg;
|
||||
|
||||
hcl_jsoner_state_node_t state_top;
|
||||
hcl_jsoner_state_node_t* state_stack;
|
||||
hcl_json_state_node_t state_top;
|
||||
hcl_json_state_node_t* state_stack;
|
||||
|
||||
struct
|
||||
{
|
||||
hcl_ooch_t* ptr;
|
||||
hcl_oow_t len;
|
||||
hcl_oow_t capa;
|
||||
} tok;
|
||||
hcl_oocs_t tok;
|
||||
hcl_oow_t tok_capa;
|
||||
};
|
||||
|
||||
|
||||
@ -100,7 +103,7 @@ struct hcl_jsoner_t
|
||||
static void log_write_for_dummy (hcl_t* hcl, unsigned int mask, const hcl_ooch_t* msg, hcl_oow_t len)
|
||||
{
|
||||
dummy_hcl_xtn_t* xtn = (dummy_hcl_xtn_t*)hcl_getxtn(hcl);
|
||||
hcl_jsoner_t* json;
|
||||
hcl_json_t* json;
|
||||
|
||||
json = xtn->json;
|
||||
json->prim.log_write (json, mask, msg, len);
|
||||
@ -147,23 +150,23 @@ static HCL_INLINE int is_digitchar (hcl_ooci_t c)
|
||||
return (c >= '0' && c <= '9');
|
||||
}
|
||||
|
||||
static void clear_token (hcl_jsoner_t* json)
|
||||
static void clear_token (hcl_json_t* json)
|
||||
{
|
||||
json->tok.len = 0;
|
||||
}
|
||||
|
||||
static int add_char_to_token (hcl_jsoner_t* json, hcl_ooch_t ch)
|
||||
static int add_char_to_token (hcl_json_t* json, hcl_ooch_t ch)
|
||||
{
|
||||
if (json->tok.len >= json->tok.capa)
|
||||
if (json->tok.len >= json->tok_capa)
|
||||
{
|
||||
hcl_ooch_t* tmp;
|
||||
hcl_oow_t newcapa;
|
||||
|
||||
newcapa = HCL_ALIGN_POW2(json->tok.len + 1, HCL_JSON_TOKEN_NAME_ALIGN);
|
||||
tmp = hcl_jsoner_reallocmem(json, json->tok.ptr, newcapa * HCL_SIZEOF(*tmp));
|
||||
tmp = hcl_json_reallocmem(json, json->tok.ptr, newcapa * HCL_SIZEOF(*tmp));
|
||||
if (!tmp) return -1;
|
||||
|
||||
json->tok.capa = newcapa;
|
||||
json->tok_capa = newcapa;
|
||||
json->tok.ptr = tmp;
|
||||
}
|
||||
|
||||
@ -171,12 +174,12 @@ static int add_char_to_token (hcl_jsoner_t* json, hcl_ooch_t ch)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static HCL_INLINE int is_token (hcl_jsoner_t* json, const hcl_bch_t* str)
|
||||
static HCL_INLINE int is_token (hcl_json_t* json, const hcl_bch_t* str)
|
||||
{
|
||||
return hcl_comp_oochars_bcstr(json->tok.ptr, json->tok.len, str) == 0;
|
||||
}
|
||||
|
||||
static HCL_INLINE int is_token_integer (hcl_jsoner_t* json, hcl_oow_t* value)
|
||||
static HCL_INLINE int is_token_integer (hcl_json_t* json, hcl_oow_t* value)
|
||||
{
|
||||
hcl_oow_t i;
|
||||
hcl_oow_t v = 0;
|
||||
@ -208,11 +211,13 @@ static HCL_INLINE hcl_ooch_t unescape (hcl_ooch_t c)
|
||||
}
|
||||
}
|
||||
|
||||
static int push_state (hcl_jsoner_t* json, hcl_jsoner_state_t state)
|
||||
{
|
||||
hcl_jsoner_state_node_t* ss;
|
||||
/* ========================================================================= */
|
||||
|
||||
ss = hcl_jsoner_callocmem(json, HCL_SIZEOF(*ss));
|
||||
static int push_state (hcl_json_t* json, hcl_json_state_t state)
|
||||
{
|
||||
hcl_json_state_node_t* ss;
|
||||
|
||||
ss = hcl_json_callocmem(json, HCL_SIZEOF(*ss));
|
||||
if (!ss) return -1;
|
||||
|
||||
ss->state = state;
|
||||
@ -222,24 +227,52 @@ static int push_state (hcl_jsoner_t* json, hcl_jsoner_state_t state)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void pop_state (hcl_jsoner_t* json)
|
||||
static void pop_state (hcl_json_t* json)
|
||||
{
|
||||
hcl_jsoner_state_node_t* ss;
|
||||
hcl_json_state_node_t* ss;
|
||||
|
||||
ss = json->state_stack;
|
||||
HCL_ASSERT (json->dummy_hcl, ss != HCL_NULL && ss != &json->state_top);
|
||||
json->state_stack = ss->next;
|
||||
|
||||
if (json->state_stack->state == HCL_JSON_STATE_IN_ARRAY)
|
||||
{
|
||||
json->state_stack->u.ia.got_value = 1;
|
||||
}
|
||||
else if (json->state_stack->state == HCL_JSON_STATE_IN_DIC)
|
||||
{
|
||||
json->state_stack->u.id.state++;
|
||||
}
|
||||
|
||||
/* TODO: don't free this. move it to the free list? */
|
||||
hcl_jsoner_freemem (json, ss);
|
||||
hcl_json_freemem (json, ss);
|
||||
}
|
||||
|
||||
static void pop_all_states (hcl_jsoner_t* json)
|
||||
static void pop_all_states (hcl_json_t* json)
|
||||
{
|
||||
while (json->state_stack != &json->state_top) pop_state (json);
|
||||
}
|
||||
|
||||
static int handle_quoted_value_char (hcl_jsoner_t* json, hcl_ooci_t c)
|
||||
/* ========================================================================= */
|
||||
|
||||
static int invoke_data_inst (hcl_json_t* json, hcl_json_inst_t inst)
|
||||
{
|
||||
if (json->state_stack->state == HCL_JSON_STATE_IN_DIC && json->state_stack->u.id.state == 1)
|
||||
{
|
||||
if (inst != HCL_JSON_INST_STRING)
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "dictionary key not a string - %.*js", json->tok.len, json->tok.ptr);
|
||||
return -1;
|
||||
}
|
||||
|
||||
inst = HCL_JSON_INST_KEY;
|
||||
}
|
||||
|
||||
json->prim.instcb (json, inst, &json->tok);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int handle_quoted_value_char (hcl_json_t* json, hcl_ooci_t c)
|
||||
{
|
||||
if (json->state_stack->u.qv.escaped >= 2)
|
||||
{
|
||||
@ -301,26 +334,270 @@ static int handle_quoted_value_char (hcl_jsoner_t* json, hcl_ooci_t c)
|
||||
else if (c == '\"')
|
||||
{
|
||||
pop_state (json);
|
||||
HCL_LOG2 (json->dummy_hcl, HCL_LOG_APP | HCL_LOG_FATAL, "[%.*js]\n", json->tok.len, json->tok.ptr);
|
||||
/* TODO: call callback ARRAY_VALUE*/
|
||||
if (invoke_data_inst(json, HCL_JSON_INST_STRING) <= -1) return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
}
|
||||
|
||||
|
||||
return 0;
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int handle_numeric_value_char (hcl_jsoner_t* json, hcl_ooci_t c)
|
||||
static int handle_numeric_value_char (hcl_json_t* json, hcl_ooci_t c)
|
||||
{
|
||||
/* TODO: */
|
||||
return -1;
|
||||
if (is_digitchar(c) || (json->tok.len == 0 && (c == '+' || c == '-')))
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else if (!json->state_stack->u.nv.dotted && c == '.' &&
|
||||
json->tok.len > 0 && is_digitchar(json->tok.ptr[json->tok.len - 1]))
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
json->state_stack->u.nv.dotted = 1;
|
||||
return 1;
|
||||
}
|
||||
|
||||
pop_state (json);
|
||||
|
||||
HCL_ASSERT (json->dummy_hcl, json->tok.len > 0);
|
||||
if (!is_digitchar(json->tok.ptr[json->tok.len - 1]))
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "invalid numeric value - %.*js", json->tok.len, json->tok.ptr);
|
||||
return -1;
|
||||
}
|
||||
if (invoke_data_inst(json, HCL_JSON_INST_NUMBER) <= -1) return -1;
|
||||
return 0; /* start over */
|
||||
}
|
||||
|
||||
static int handle_char (hcl_jsoner_t* json, hcl_ooci_t c, hcl_oow_t nbytes)
|
||||
static int handle_word_value_char (hcl_json_t* json, hcl_ooci_t c)
|
||||
{
|
||||
hcl_json_inst_t inst;
|
||||
|
||||
if (is_alphachar(c))
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
|
||||
pop_state (json);
|
||||
|
||||
if (hcl_comp_oochars_bcstr(json->tok.ptr, json->tok.len, "null") == 0) inst = HCL_JSON_INST_NIL;
|
||||
else if (hcl_comp_oochars_bcstr(json->tok.ptr, json->tok.len, "true") == 0) inst = HCL_JSON_INST_TRUE;
|
||||
else if (hcl_comp_oochars_bcstr(json->tok.ptr, json->tok.len, "false") == 0) inst = HCL_JSON_INST_FALSE;
|
||||
else
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "invalue word value - %.*js", json->tok.len, json->tok.ptr);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (invoke_data_inst(json, inst) <= -1) return -1;
|
||||
return 0; /* start over */
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static int handle_start_char (hcl_json_t* json, hcl_ooci_t c)
|
||||
{
|
||||
if (c == '[')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_ARRAY) <= -1) return -1;
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
json->prim.instcb (json, HCL_JSON_INST_START_ARRAY, HCL_NULL);
|
||||
return 1;
|
||||
}
|
||||
else if (c == '{')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_DIC) <= -1) return -1;
|
||||
json->state_stack->u.id.state = 0;
|
||||
json->prim.instcb (json, HCL_JSON_INST_START_DIC, HCL_NULL);
|
||||
return 1;
|
||||
}
|
||||
else if (is_spacechar(c))
|
||||
{
|
||||
/* do nothing */
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "not starting with [ or { - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
static int handle_char_in_array (hcl_json_t* json, hcl_ooci_t c)
|
||||
{
|
||||
if (c == ']')
|
||||
{
|
||||
json->prim.instcb (json, HCL_JSON_INST_END_ARRAY, HCL_NULL);
|
||||
pop_state (json);
|
||||
return 1;
|
||||
}
|
||||
else if (c == ',')
|
||||
{
|
||||
if (!json->state_stack->u.ia.got_value)
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "redundant comma in array - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
return 1;
|
||||
}
|
||||
else if (is_spacechar(c))
|
||||
{
|
||||
/* do nothing */
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (json->state_stack->u.ia.got_value)
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "comma required in array - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (c == '\"')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_QUOTED_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 1;
|
||||
}
|
||||
/* TOOD: else if (c == '\'') HCL character
|
||||
* else if (c == '#') HCL radixed number
|
||||
*/
|
||||
else if (is_digitchar(c) || c == '+' || c == '-')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_NUMERIC_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
json->state_stack->u.nv.dotted = 0;
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (is_alphachar(c))
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_WORD_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (c == '[')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_ARRAY) <= -1) return -1;
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
json->prim.instcb (json, HCL_JSON_INST_START_ARRAY, HCL_NULL);
|
||||
return 1;
|
||||
}
|
||||
else if (c == '{')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_DIC) <= -1) return -1;
|
||||
json->state_stack->u.id.state = 0;
|
||||
json->prim.instcb (json, HCL_JSON_INST_START_DIC, HCL_NULL);
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "wrong character inside array - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int handle_char_in_dic (hcl_json_t* json, hcl_ooci_t c)
|
||||
{
|
||||
if (c == '}')
|
||||
{
|
||||
json->prim.instcb (json, HCL_JSON_INST_END_DIC, HCL_NULL);
|
||||
pop_state (json);
|
||||
return 1;
|
||||
}
|
||||
else if (c == ':')
|
||||
{
|
||||
if (json->state_stack->u.id.state != 1)
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "redundant colon in dictionary - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
json->state_stack->u.id.state++;
|
||||
return 1;
|
||||
}
|
||||
else if (c == ',')
|
||||
{
|
||||
if (json->state_stack->u.id.state != 3)
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "redundant comma in dicitonary - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
json->state_stack->u.id.state = 0;
|
||||
return 1;
|
||||
}
|
||||
else if (is_spacechar(c))
|
||||
{
|
||||
/* do nothing */
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (json->state_stack->u.id.state == 1)
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "colon required in dicitonary - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
else if (json->state_stack->u.id.state == 3)
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "comma required in dicitonary - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (c == '\"')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_QUOTED_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 1;
|
||||
}
|
||||
/* TOOD: else if (c == '\'') HCL character
|
||||
* else if (c == '#') HCL radixed number
|
||||
*/
|
||||
else if (is_digitchar(c) || c == '+' || c == '-')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_NUMERIC_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
json->state_stack->u.nv.dotted = 0;
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (is_alphachar(c))
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_WORD_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (c == '[')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_ARRAY) <= -1) return -1;
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
json->prim.instcb (json, HCL_JSON_INST_START_ARRAY, HCL_NULL);
|
||||
return 1;
|
||||
}
|
||||
else if (c == '{')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_DIC) <= -1) return -1;
|
||||
json->state_stack->u.id.state = 0;
|
||||
json->prim.instcb (json, HCL_JSON_INST_START_DIC, HCL_NULL);
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
hcl_json_seterrbfmt (json, HCL_EINVAL, "wrong character inside array - %jc", (hcl_ooch_t)c);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static int handle_char (hcl_json_t* json, hcl_ooci_t c, hcl_oow_t nbytes)
|
||||
{
|
||||
int x;
|
||||
|
||||
start_over:
|
||||
if (c == HCL_OOCI_EOF)
|
||||
{
|
||||
if (json->state_stack->state == HCL_JSON_STATE_START)
|
||||
@ -330,117 +607,51 @@ static int handle_char (hcl_jsoner_t* json, hcl_ooci_t c, hcl_oow_t nbytes)
|
||||
}
|
||||
else
|
||||
{
|
||||
hcl_jsoner_seterrbfmt (json, HCL_EFINIS, "unexpected end of data");
|
||||
hcl_json_seterrbfmt (json, HCL_EFINIS, "unexpected end of data");
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
printf ("handling [%c] %d\n", c, (int)nbytes);
|
||||
switch (json->state_stack->state)
|
||||
{
|
||||
case HCL_JSON_STATE_START:
|
||||
if (c == '[')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_ARRAY) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
else if (c == '{')
|
||||
{
|
||||
if (push_state(json, HCL_JSON_STATE_IN_DIC) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
else if (is_spacechar(c))
|
||||
{
|
||||
/* skip whitespaces at the beginning of the start line before the reply name */
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
hcl_jsoner_seterrbfmt (json, HCL_EINVAL, "not starting with [ or { - %jc", (hcl_ooch_t)c);
|
||||
goto oops;
|
||||
}
|
||||
x = handle_start_char(json, c);
|
||||
break;
|
||||
|
||||
case HCL_JSON_STATE_IN_ARRAY:
|
||||
if (c == ']')
|
||||
{
|
||||
pop_state (json);
|
||||
break;
|
||||
}
|
||||
else if (c == ',')
|
||||
{
|
||||
/* TODO: handle this */
|
||||
}
|
||||
else if (is_spacechar(c))
|
||||
{
|
||||
break;
|
||||
}
|
||||
else if (c == '\"')
|
||||
{
|
||||
|
||||
if (push_state(json, HCL_JSON_STATE_IN_QUOTED_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
break;
|
||||
}
|
||||
else if (is_alphachar(c) || is_digitchar(c))
|
||||
{
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
hcl_jsoner_seterrbfmt (json, HCL_EINVAL, "wrong character inside array - %jc", (hcl_ooch_t)c);
|
||||
goto oops;
|
||||
}
|
||||
x = handle_char_in_array(json, c);
|
||||
break;
|
||||
|
||||
case HCL_JSON_STATE_IN_DIC:
|
||||
if (c == '}')
|
||||
{
|
||||
pop_state (json);
|
||||
break;
|
||||
}
|
||||
else if (is_spacechar(c))
|
||||
{
|
||||
break;
|
||||
}
|
||||
else if (c == '\"')
|
||||
{
|
||||
break;
|
||||
}
|
||||
else if (is_alphachar(c) || is_digitchar(c))
|
||||
{
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
hcl_jsoner_seterrbfmt (json, HCL_EINVAL, "wrong character inside dictionary - %jc", (hcl_ooch_t)c);
|
||||
goto oops;
|
||||
}
|
||||
x = handle_char_in_dic(json, c);
|
||||
break;
|
||||
|
||||
case HCL_JSON_STATE_IN_WORD_VALUE:
|
||||
x = handle_word_value_char(json, c);
|
||||
break;
|
||||
|
||||
case HCL_JSON_STATE_IN_QUOTED_VALUE:
|
||||
if (handle_quoted_value_char(json, c) <= -1) goto oops;
|
||||
x = handle_quoted_value_char(json, c);
|
||||
break;
|
||||
|
||||
case HCL_JSON_STATE_IN_NUMERIC_VALUE:
|
||||
if (handle_numeric_value_char(json, c) <= -1) goto oops;
|
||||
x = handle_numeric_value_char(json, c);
|
||||
break;
|
||||
|
||||
default:
|
||||
hcl_jsoner_seterrbfmt (json, HCL_EINTERN, "internal error - must not be called for state %d", (int)json->state_stack->state);
|
||||
goto oops;
|
||||
hcl_json_seterrbfmt (json, HCL_EINTERN, "internal error - must not be called for state %d", (int)json->state_stack->state);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (x <= -1) return -1;
|
||||
if (x == 0) goto start_over;
|
||||
|
||||
return 0;
|
||||
|
||||
oops:
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int feed_json_data (hcl_jsoner_t* json, const hcl_bch_t* data, hcl_oow_t len, hcl_oow_t* xlen)
|
||||
/* ========================================================================= */
|
||||
|
||||
static int feed_json_data (hcl_json_t* json, const hcl_bch_t* data, hcl_oow_t len, hcl_oow_t* xlen)
|
||||
{
|
||||
const hcl_bch_t* ptr;
|
||||
const hcl_bch_t* end;
|
||||
@ -500,14 +711,14 @@ oops:
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
hcl_jsoner_t* hcl_jsoner_open (hcl_mmgr_t* mmgr, hcl_oow_t xtnsize, hcl_jsoner_prim_t* prim, hcl_errnum_t* errnum)
|
||||
hcl_json_t* hcl_json_open (hcl_mmgr_t* mmgr, hcl_oow_t xtnsize, hcl_json_prim_t* prim, hcl_errnum_t* errnum)
|
||||
{
|
||||
hcl_jsoner_t* json;
|
||||
hcl_json_t* json;
|
||||
hcl_t* hcl;
|
||||
hcl_vmprim_t vmprim;
|
||||
dummy_hcl_xtn_t* xtn;
|
||||
|
||||
json = (hcl_jsoner_t*)HCL_MMGR_ALLOC(mmgr, HCL_SIZEOF(*json) + xtnsize);
|
||||
json = (hcl_json_t*)HCL_MMGR_ALLOC(mmgr, HCL_SIZEOF(*json) + xtnsize);
|
||||
if (!json)
|
||||
{
|
||||
if (errnum) *errnum = HCL_ESYSMEM;
|
||||
@ -550,15 +761,15 @@ hcl_jsoner_t* hcl_jsoner_open (hcl_mmgr_t* mmgr, hcl_oow_t xtnsize, hcl_jsoner_p
|
||||
return json;
|
||||
}
|
||||
|
||||
void hcl_jsoner_close (hcl_jsoner_t* json)
|
||||
void hcl_json_close (hcl_json_t* json)
|
||||
{
|
||||
pop_all_states (json);
|
||||
if (json->tok.ptr) hcl_jsoner_freemem (json, json->tok.ptr);
|
||||
if (json->tok.ptr) hcl_json_freemem (json, json->tok.ptr);
|
||||
hcl_close (json->dummy_hcl);
|
||||
HCL_MMGR_FREE (json->mmgr, json);
|
||||
}
|
||||
|
||||
int hcl_jsoner_setoption (hcl_jsoner_t* json, hcl_jsoner_option_t id, const void* value)
|
||||
int hcl_json_setoption (hcl_json_t* json, hcl_json_option_t id, const void* value)
|
||||
{
|
||||
switch (id)
|
||||
{
|
||||
@ -579,11 +790,11 @@ int hcl_jsoner_setoption (hcl_jsoner_t* json, hcl_jsoner_option_t id, const void
|
||||
return 0;
|
||||
}
|
||||
|
||||
hcl_jsoner_seterrnum (json, HCL_EINVAL);
|
||||
hcl_json_seterrnum (json, HCL_EINVAL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
int hcl_jsoner_getoption (hcl_jsoner_t* json, hcl_jsoner_option_t id, void* value)
|
||||
int hcl_json_getoption (hcl_json_t* json, hcl_json_option_t id, void* value)
|
||||
{
|
||||
switch (id)
|
||||
{
|
||||
@ -596,55 +807,55 @@ int hcl_jsoner_getoption (hcl_jsoner_t* json, hcl_jsoner_option_t id, void* valu
|
||||
return 0;
|
||||
};
|
||||
|
||||
hcl_jsoner_seterrnum (json, HCL_EINVAL);
|
||||
hcl_json_seterrnum (json, HCL_EINVAL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
void* hcl_jsoner_getxtn (hcl_jsoner_t* json)
|
||||
void* hcl_json_getxtn (hcl_json_t* json)
|
||||
{
|
||||
return (void*)(json + 1);
|
||||
}
|
||||
|
||||
hcl_mmgr_t* hcl_jsoner_getmmgr (hcl_jsoner_t* json)
|
||||
hcl_mmgr_t* hcl_json_getmmgr (hcl_json_t* json)
|
||||
{
|
||||
return json->mmgr;
|
||||
}
|
||||
|
||||
hcl_cmgr_t* hcl_jsoner_getcmgr (hcl_jsoner_t* json)
|
||||
hcl_cmgr_t* hcl_json_getcmgr (hcl_json_t* json)
|
||||
{
|
||||
return json->cmgr;
|
||||
}
|
||||
|
||||
void hcl_jsoner_setcmgr (hcl_jsoner_t* json, hcl_cmgr_t* cmgr)
|
||||
void hcl_json_setcmgr (hcl_json_t* json, hcl_cmgr_t* cmgr)
|
||||
{
|
||||
json->cmgr = cmgr;
|
||||
}
|
||||
|
||||
hcl_errnum_t hcl_jsoner_geterrnum (hcl_jsoner_t* json)
|
||||
hcl_errnum_t hcl_json_geterrnum (hcl_json_t* json)
|
||||
{
|
||||
return json->errnum;
|
||||
}
|
||||
|
||||
const hcl_ooch_t* hcl_jsoner_geterrstr (hcl_jsoner_t* json)
|
||||
const hcl_ooch_t* hcl_json_geterrstr (hcl_json_t* json)
|
||||
{
|
||||
return hcl_errnum_to_errstr(json->errnum);
|
||||
}
|
||||
|
||||
const hcl_ooch_t* hcl_jsoner_geterrmsg (hcl_jsoner_t* json)
|
||||
const hcl_ooch_t* hcl_json_geterrmsg (hcl_json_t* json)
|
||||
{
|
||||
if (json->errmsg.len <= 0) return hcl_errnum_to_errstr(json->errnum);
|
||||
return json->errmsg.buf;
|
||||
}
|
||||
|
||||
void hcl_jsoner_seterrnum (hcl_jsoner_t* json, hcl_errnum_t errnum)
|
||||
void hcl_json_seterrnum (hcl_json_t* json, hcl_errnum_t errnum)
|
||||
{
|
||||
/*if (json->shuterr) return; */
|
||||
json->errnum = errnum;
|
||||
json->errmsg.len = 0;
|
||||
}
|
||||
|
||||
void hcl_jsoner_seterrbfmt (hcl_jsoner_t* json, hcl_errnum_t errnum, const hcl_bch_t* fmt, ...)
|
||||
void hcl_json_seterrbfmt (hcl_json_t* json, hcl_errnum_t errnum, const hcl_bch_t* fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
@ -658,7 +869,7 @@ void hcl_jsoner_seterrbfmt (hcl_jsoner_t* json, hcl_errnum_t errnum, const hcl_b
|
||||
json->errmsg.len = json->dummy_hcl->errmsg.len;
|
||||
}
|
||||
|
||||
void hcl_jsoner_seterrufmt (hcl_jsoner_t* json, hcl_errnum_t errnum, const hcl_uch_t* fmt, ...)
|
||||
void hcl_json_seterrufmt (hcl_json_t* json, hcl_errnum_t errnum, const hcl_uch_t* fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
@ -674,7 +885,7 @@ void hcl_jsoner_seterrufmt (hcl_jsoner_t* json, hcl_errnum_t errnum, const hcl_u
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
void hcl_jsoner_logbfmt (hcl_jsoner_t* json, unsigned int mask, const hcl_bch_t* fmt, ...)
|
||||
void hcl_json_logbfmt (hcl_json_t* json, unsigned int mask, const hcl_bch_t* fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
va_start (ap, fmt);
|
||||
@ -682,7 +893,7 @@ void hcl_jsoner_logbfmt (hcl_jsoner_t* json, unsigned int mask, const hcl_bch_t*
|
||||
va_end (ap);
|
||||
}
|
||||
|
||||
void hcl_jsoner_logufmt (hcl_jsoner_t* json, unsigned int mask, const hcl_uch_t* fmt, ...)
|
||||
void hcl_json_logufmt (hcl_json_t* json, unsigned int mask, const hcl_uch_t* fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
va_start (ap, fmt);
|
||||
@ -692,45 +903,45 @@ void hcl_jsoner_logufmt (hcl_jsoner_t* json, unsigned int mask, const hcl_uch_t*
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
void* hcl_jsoner_allocmem (hcl_jsoner_t* json, hcl_oow_t size)
|
||||
void* hcl_json_allocmem (hcl_json_t* json, hcl_oow_t size)
|
||||
{
|
||||
void* ptr;
|
||||
|
||||
ptr = HCL_MMGR_ALLOC(json->mmgr, size);
|
||||
if (!ptr) hcl_jsoner_seterrnum (json, HCL_ESYSMEM);
|
||||
if (!ptr) hcl_json_seterrnum (json, HCL_ESYSMEM);
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void* hcl_jsoner_callocmem (hcl_jsoner_t* json, hcl_oow_t size)
|
||||
void* hcl_json_callocmem (hcl_json_t* json, hcl_oow_t size)
|
||||
{
|
||||
void* ptr;
|
||||
|
||||
ptr = HCL_MMGR_ALLOC(json->mmgr, size);
|
||||
if (!ptr) hcl_jsoner_seterrnum (json, HCL_ESYSMEM);
|
||||
if (!ptr) hcl_json_seterrnum (json, HCL_ESYSMEM);
|
||||
else HCL_MEMSET (ptr, 0, size);
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void* hcl_jsoner_reallocmem (hcl_jsoner_t* json, void* ptr, hcl_oow_t size)
|
||||
void* hcl_json_reallocmem (hcl_json_t* json, void* ptr, hcl_oow_t size)
|
||||
{
|
||||
ptr = HCL_MMGR_REALLOC(json->mmgr, ptr, size);
|
||||
if (!ptr) hcl_jsoner_seterrnum (json, HCL_ESYSMEM);
|
||||
if (!ptr) hcl_json_seterrnum (json, HCL_ESYSMEM);
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void hcl_jsoner_freemem (hcl_jsoner_t* json, void* ptr)
|
||||
void hcl_json_freemem (hcl_json_t* json, void* ptr)
|
||||
{
|
||||
HCL_MMGR_FREE (json->mmgr, ptr);
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
hcl_jsoner_state_t hcl_jsoner_getstate (hcl_jsoner_t* json)
|
||||
hcl_json_state_t hcl_json_getstate (hcl_json_t* json)
|
||||
{
|
||||
return json->state_stack->state;
|
||||
}
|
||||
|
||||
void hcl_jsoner_reset (hcl_jsoner_t* json)
|
||||
void hcl_json_reset (hcl_json_t* json)
|
||||
{
|
||||
/* TODO: reset XXXXXXXXXXXXXXXXXXXXXXXXXXXxxxxx */
|
||||
pop_all_states (json);
|
||||
@ -738,7 +949,7 @@ void hcl_jsoner_reset (hcl_jsoner_t* json)
|
||||
json->state_stack->state = HCL_JSON_STATE_START;
|
||||
}
|
||||
|
||||
int hcl_jsoner_feed (hcl_jsoner_t* json, const void* ptr, hcl_oow_t len, hcl_oow_t* xlen)
|
||||
int hcl_json_feed (hcl_json_t* json, const void* ptr, hcl_oow_t len, hcl_oow_t* xlen)
|
||||
{
|
||||
int x;
|
||||
hcl_oow_t total, ylen;
|
||||
|
97
lib/main-j.c
97
lib/main-j.c
@ -9,8 +9,8 @@
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
typedef struct jsoner_xtn_t jsoner_xtn_t;
|
||||
struct jsoner_xtn_t
|
||||
typedef struct json_xtn_t json_xtn_t;
|
||||
struct json_xtn_t
|
||||
{
|
||||
int logfd;
|
||||
unsigned int logmask;
|
||||
@ -84,12 +84,12 @@ static int write_all (int fd, const hcl_bch_t* ptr, hcl_oow_t len)
|
||||
}
|
||||
|
||||
|
||||
static int write_log (hcl_jsoner_t* jsoner, int fd, const hcl_bch_t* ptr, hcl_oow_t len)
|
||||
static int write_log (hcl_json_t* json, int fd, const hcl_bch_t* ptr, hcl_oow_t len)
|
||||
{
|
||||
jsoner_xtn_t* xtn;
|
||||
json_xtn_t* xtn;
|
||||
|
||||
|
||||
xtn = hcl_jsoner_getxtn(jsoner);
|
||||
xtn = hcl_json_getxtn(json);
|
||||
|
||||
while (len > 0)
|
||||
{
|
||||
@ -136,10 +136,10 @@ static int write_log (hcl_jsoner_t* jsoner, int fd, const hcl_bch_t* ptr, hcl_oo
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void flush_log (hcl_jsoner_t* jsoner, int fd)
|
||||
static void flush_log (hcl_json_t* json, int fd)
|
||||
{
|
||||
jsoner_xtn_t* xtn;
|
||||
xtn = hcl_jsoner_getxtn(jsoner);
|
||||
json_xtn_t* xtn;
|
||||
xtn = hcl_json_getxtn(json);
|
||||
if (xtn->logbuf.len > 0)
|
||||
{
|
||||
write_all (fd, xtn->logbuf.buf, xtn->logbuf.len);
|
||||
@ -147,15 +147,15 @@ static void flush_log (hcl_jsoner_t* jsoner, int fd)
|
||||
}
|
||||
}
|
||||
|
||||
static void log_write (hcl_jsoner_t* jsoner, unsigned int mask, const hcl_ooch_t* msg, hcl_oow_t len)
|
||||
static void log_write (hcl_json_t* json, unsigned int mask, const hcl_ooch_t* msg, hcl_oow_t len)
|
||||
{
|
||||
hcl_bch_t buf[256];
|
||||
hcl_oow_t ucslen, bcslen;
|
||||
jsoner_xtn_t* xtn;
|
||||
json_xtn_t* xtn;
|
||||
hcl_oow_t msgidx;
|
||||
int n, logfd;
|
||||
|
||||
xtn = hcl_jsoner_getxtn(jsoner);
|
||||
xtn = hcl_json_getxtn(json);
|
||||
|
||||
if (mask & HCL_LOG_STDERR)
|
||||
{
|
||||
@ -198,14 +198,14 @@ static void log_write (hcl_jsoner_t* jsoner, unsigned int mask, const hcl_ooch_t
|
||||
tslen = 25;
|
||||
}
|
||||
|
||||
write_log (jsoner, logfd, ts, tslen);
|
||||
write_log (json, logfd, ts, tslen);
|
||||
}
|
||||
|
||||
if (logfd == xtn->logfd && xtn->logfd_istty)
|
||||
{
|
||||
if (mask & HCL_LOG_FATAL) write_log (jsoner, logfd, "\x1B[1;31m", 7);
|
||||
else if (mask & HCL_LOG_ERROR) write_log (jsoner, logfd, "\x1B[1;32m", 7);
|
||||
else if (mask & HCL_LOG_WARN) write_log (jsoner, logfd, "\x1B[1;33m", 7);
|
||||
if (mask & HCL_LOG_FATAL) write_log (json, logfd, "\x1B[1;31m", 7);
|
||||
else if (mask & HCL_LOG_ERROR) write_log (json, logfd, "\x1B[1;32m", 7);
|
||||
else if (mask & HCL_LOG_WARN) write_log (json, logfd, "\x1B[1;33m", 7);
|
||||
}
|
||||
|
||||
#if defined(HCL_OOCH_IS_UCH)
|
||||
@ -228,7 +228,7 @@ static void log_write (hcl_jsoner_t* jsoner, unsigned int mask, const hcl_ooch_t
|
||||
/*assert (ucslen > 0);*/
|
||||
|
||||
/* attempt to write all converted characters */
|
||||
if (write_log(jsoner, logfd, buf, bcslen) <= -1) break;
|
||||
if (write_log(json, logfd, buf, bcslen) <= -1) break;
|
||||
|
||||
if (n == 0) break;
|
||||
else
|
||||
@ -244,41 +244,76 @@ static void log_write (hcl_jsoner_t* jsoner, unsigned int mask, const hcl_ooch_t
|
||||
}
|
||||
}
|
||||
#else
|
||||
write_log (jsoner, logfd, msg, len);
|
||||
write_log (json, logfd, msg, len);
|
||||
#endif
|
||||
|
||||
if (logfd == xtn->logfd && xtn->logfd_istty)
|
||||
{
|
||||
if (mask & (HCL_LOG_FATAL | HCL_LOG_ERROR | HCL_LOG_WARN)) write_log (jsoner, logfd, "\x1B[0m", 4);
|
||||
if (mask & (HCL_LOG_FATAL | HCL_LOG_ERROR | HCL_LOG_WARN)) write_log (json, logfd, "\x1B[0m", 4);
|
||||
}
|
||||
|
||||
flush_log (jsoner, logfd);
|
||||
flush_log (json, logfd);
|
||||
}
|
||||
|
||||
static int instcb (hcl_json_t* json, hcl_json_inst_t it, const hcl_oocs_t* str)
|
||||
{
|
||||
switch (it)
|
||||
{
|
||||
case HCL_JSON_INST_START_ARRAY:
|
||||
hcl_json_logbfmt (json, HCL_LOG_INFO | HCL_LOG_APP, "[\n");
|
||||
break;
|
||||
case HCL_JSON_INST_END_ARRAY:
|
||||
hcl_json_logbfmt (json, HCL_LOG_INFO | HCL_LOG_APP, "]\n");
|
||||
break;
|
||||
case HCL_JSON_INST_START_DIC:
|
||||
hcl_json_logbfmt (json, HCL_LOG_INFO | HCL_LOG_APP, "{\n");
|
||||
break;
|
||||
case HCL_JSON_INST_END_DIC:
|
||||
hcl_json_logbfmt (json, HCL_LOG_INFO | HCL_LOG_APP, "}\n");
|
||||
break;
|
||||
|
||||
case HCL_JSON_INST_KEY:
|
||||
hcl_json_logbfmt (json, HCL_LOG_INFO | HCL_LOG_APP, "%.*js: ", str->len, str->ptr);
|
||||
break;
|
||||
|
||||
case HCL_JSON_INST_STRING:
|
||||
case HCL_JSON_INST_NUMBER:
|
||||
case HCL_JSON_INST_TRUE:
|
||||
case HCL_JSON_INST_FALSE:
|
||||
case HCL_JSON_INST_NIL:
|
||||
hcl_json_logbfmt (json, HCL_LOG_INFO | HCL_LOG_APP, "%.*js\n", str->len, str->ptr);
|
||||
break;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
/* ========================================================================= */
|
||||
|
||||
int main (int argc, char* argv[])
|
||||
{
|
||||
|
||||
hcl_jsoner_t* jsoner;
|
||||
hcl_jsoner_prim_t json_prim;
|
||||
jsoner_xtn_t* jsoner_xtn;
|
||||
hcl_json_t* json;
|
||||
hcl_json_prim_t json_prim;
|
||||
json_xtn_t* json_xtn;
|
||||
hcl_oow_t xlen;
|
||||
const char* p;
|
||||
|
||||
|
||||
memset (&json_prim, 0, HCL_SIZEOF(json_prim));
|
||||
json_prim.log_write = log_write;
|
||||
|
||||
jsoner = hcl_jsoner_open (&sys_mmgr, HCL_SIZEOF(jsoner_xtn_t), &json_prim, NULL);
|
||||
json_prim.instcb = instcb;
|
||||
|
||||
json = hcl_json_open (&sys_mmgr, HCL_SIZEOF(json_xtn_t), &json_prim, NULL);
|
||||
|
||||
jsoner_xtn = hcl_jsoner_getxtn(jsoner);
|
||||
jsoner_xtn->logmask = HCL_LOG_ALL_LEVELS | HCL_LOG_ALL_TYPES;
|
||||
json_xtn = hcl_json_getxtn(json);
|
||||
json_xtn->logmask = HCL_LOG_ALL_LEVELS | HCL_LOG_ALL_TYPES;
|
||||
|
||||
p = "[ \"ab\\xab\\uC88B\\uC544\\uC6A9c\", \"kaden\", \"iron\" ]";
|
||||
|
||||
p = "[ \"ab\\xab\\uC88B\\uC544\\uC6A9c\", \"kaden\", \"iron\", true, { \"null\": \"abc\", \"123\": \"AA20AA\", \"10\": -0.123 } ]";
|
||||
|
||||
hcl_jsoner_feed (jsoner, p, strlen(p), &xlen);
|
||||
hcl_jsoner_close (jsoner);
|
||||
if (hcl_json_feed(json, p, strlen(p), &xlen) <= -1)
|
||||
{
|
||||
hcl_json_logbfmt (json, HCL_LOG_FATAL | HCL_LOG_APP, "ERROR: %js\n", hcl_json_geterrmsg(json));
|
||||
}
|
||||
|
||||
hcl_json_close (json);
|
||||
return 0;
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user