mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-12 15:45:21 +08:00
284 lines
7.3 KiB
C
284 lines
7.3 KiB
C
|
/*
|
||
|
* Copyright 2018, University Corporation for Atmospheric Research
|
||
|
* See netcdf/COPYRIGHT file for copying and redistribution conditions.
|
||
|
*/
|
||
|
|
||
|
#include "ut_includes.h"
|
||
|
|
||
|
#define DEBUG
|
||
|
|
||
|
typedef enum Cmds {
|
||
|
cmd_none = 0,
|
||
|
cmd_build = 1,
|
||
|
cmd_parse = 2,
|
||
|
} Cmds;
|
||
|
|
||
|
/* Forward */
|
||
|
static int testbuild(void);
|
||
|
static int testparse(void);
|
||
|
|
||
|
static void dump(NCjson* json);
|
||
|
static void dumpR(NCjson* json, int depth);
|
||
|
static char* sortname(int sort);
|
||
|
|
||
|
struct Test tests[] = {
|
||
|
{"build", testbuild},
|
||
|
{"parse", testparse},
|
||
|
{NULL, NULL}
|
||
|
};
|
||
|
|
||
|
typedef struct NCJ {
|
||
|
NCjson* ncj_string;
|
||
|
NCjson* ncj_int;
|
||
|
NCjson* ncj_double;
|
||
|
NCjson* ncj_boolean;
|
||
|
NCjson* ncj_null;
|
||
|
NCjson* ncj_array1;
|
||
|
NCjson* ncj_array2;
|
||
|
NCjson* ncj_dict1;
|
||
|
NCjson* ncj_dict2;
|
||
|
} NCJ;
|
||
|
|
||
|
int
|
||
|
main(int argc, char** argv)
|
||
|
{
|
||
|
int stat = NC_NOERR;
|
||
|
|
||
|
if((stat = ut_init(argc, argv, &options))) goto done;
|
||
|
if((stat = runtests((const char**)options.cmds,tests))) goto done;
|
||
|
|
||
|
done:
|
||
|
if(stat) usage(stat);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* Build a reasonably complex json structure */
|
||
|
static int
|
||
|
build(NCJ* ncj)
|
||
|
{
|
||
|
int stat = NC_NOERR;
|
||
|
NCjson* clone;
|
||
|
|
||
|
memset(ncj,0,sizeof(NCJ));
|
||
|
|
||
|
/* Build instances of primitives */
|
||
|
if((stat = NCJnew(NCJ_STRING,&ncj->ncj_string))) goto done;
|
||
|
ncj->ncj_string->value = strdup("string");
|
||
|
if((stat = NCJnew(NCJ_INT,&ncj->ncj_int))) goto done;
|
||
|
ncj->ncj_int->value = strdup("117");
|
||
|
if((stat = NCJnew(NCJ_DOUBLE,&ncj->ncj_double))) goto done;
|
||
|
ncj->ncj_double->value = strdup("3.1415926");
|
||
|
if((stat = NCJnew(NCJ_BOOLEAN,&ncj->ncj_boolean))) goto done;
|
||
|
ncj->ncj_boolean->value = strdup("true");
|
||
|
if((stat = NCJnew(NCJ_NULL,&ncj->ncj_null))) goto done;
|
||
|
|
||
|
/* Create an empty array */
|
||
|
if((stat = NCJnew(NCJ_ARRAY,&ncj->ncj_array1))) goto done;
|
||
|
|
||
|
/* Create a filled array */
|
||
|
if((stat = NCJnew(NCJ_ARRAY,&ncj->ncj_array2))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_string,&clone))) goto done;
|
||
|
if((stat = NCJappend(ncj->ncj_array2,clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_int,&clone))) goto done;
|
||
|
if((stat = NCJappend(ncj->ncj_array2,clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_double,&clone))) goto done;
|
||
|
if((stat = NCJappend(ncj->ncj_array2,clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_boolean,&clone))) goto done;
|
||
|
if((stat = NCJappend(ncj->ncj_array2,clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_null,&clone))) goto done;
|
||
|
if((stat = NCJappend(ncj->ncj_array2,clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_array1,&clone))) goto done;
|
||
|
if((stat = NCJappend(ncj->ncj_array2,clone))) goto done;
|
||
|
|
||
|
/* Create an empty dict */
|
||
|
if((stat = NCJnew(NCJ_DICT,&ncj->ncj_dict1))) goto done;
|
||
|
|
||
|
/* Create a filled dict */
|
||
|
if((stat = NCJnew(NCJ_DICT,&ncj->ncj_dict2))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_string,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"string",clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_int,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"int",clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_double,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"double",clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_boolean,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"boolean",clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_null,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"null",clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_array1,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"array1",clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_array2,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"array2",clone))) goto done;
|
||
|
if((stat = NCJclone(ncj->ncj_dict1,&clone))) goto done;
|
||
|
if((stat = NCJinsert(ncj->ncj_dict2,"dict1",clone))) goto done;
|
||
|
|
||
|
done:
|
||
|
return THROW(stat);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
clear(NCJ* ncj)
|
||
|
{
|
||
|
NCJreclaim(ncj->ncj_array1);
|
||
|
NCJreclaim(ncj->ncj_array2);
|
||
|
NCJreclaim(ncj->ncj_dict1);
|
||
|
NCJreclaim(ncj->ncj_dict2);
|
||
|
NCJreclaim(ncj->ncj_string);
|
||
|
NCJreclaim(ncj->ncj_int);
|
||
|
NCJreclaim(ncj->ncj_double);
|
||
|
NCJreclaim(ncj->ncj_boolean);
|
||
|
NCJreclaim(ncj->ncj_null);
|
||
|
}
|
||
|
|
||
|
/* Create test netcdf4 file via netcdf.h API*/
|
||
|
static int
|
||
|
testbuild(void)
|
||
|
{
|
||
|
int stat = NC_NOERR;
|
||
|
NCJ ncj;
|
||
|
|
||
|
/* Build */
|
||
|
if((stat = build(&ncj))) goto done;
|
||
|
|
||
|
/* Primitives */
|
||
|
dump(ncj.ncj_string);
|
||
|
dump(ncj.ncj_int);
|
||
|
dump(ncj.ncj_double);
|
||
|
dump(ncj.ncj_boolean);
|
||
|
dump(ncj.ncj_null);
|
||
|
|
||
|
/* Empty array */
|
||
|
dump(ncj.ncj_array1);
|
||
|
|
||
|
/* Filled array */
|
||
|
dump(ncj.ncj_array2);
|
||
|
|
||
|
/* Empty dict */
|
||
|
dump(ncj.ncj_dict1);
|
||
|
|
||
|
/* Filled dict */
|
||
|
dump(ncj.ncj_dict2);
|
||
|
|
||
|
done:
|
||
|
clear(&ncj);
|
||
|
return THROW(stat);
|
||
|
}
|
||
|
|
||
|
/* Test the parser */
|
||
|
static int
|
||
|
testparse(void)
|
||
|
{
|
||
|
NCJ ncj;
|
||
|
int stat = NC_NOERR;
|
||
|
char* text = NULL;
|
||
|
char* result = NULL;
|
||
|
NCjson* json = NULL;
|
||
|
|
||
|
/* Build */
|
||
|
if((stat = build(&ncj))) goto done;
|
||
|
|
||
|
if((stat = NCJunparse(ncj.ncj_dict2,0,&text))) goto done;
|
||
|
|
||
|
if((stat = NCJparse(text,0,&json))) goto done;
|
||
|
|
||
|
if((stat = NCJunparse(json,0,&result))) goto done;
|
||
|
|
||
|
printf("text : |%s|\nresult: |%s|\n",text,result);
|
||
|
|
||
|
done:
|
||
|
nullfree(text);
|
||
|
nullfree(result);
|
||
|
NCJreclaim(json);
|
||
|
clear(&ncj);
|
||
|
return stat;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
dump(NCjson* json)
|
||
|
{
|
||
|
dumpR(json,0);
|
||
|
fflush(stdout);
|
||
|
fflush(stderr);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
dumpR(NCjson* json, int depth)
|
||
|
{
|
||
|
int ok, count, i;
|
||
|
long long int64v;
|
||
|
double float64v;
|
||
|
|
||
|
printf("/%s/ ",sortname(json->sort));
|
||
|
switch(json->sort) {
|
||
|
case NCJ_STRING: printf("\"%s\"",json->value); break;
|
||
|
case NCJ_INT:
|
||
|
ok = sscanf(json->value,"%lld%n",&int64v,&count);
|
||
|
if(ok != 1 || count != strlen(json->value)) goto fail;
|
||
|
printf("%lld",int64v);
|
||
|
break;
|
||
|
case NCJ_DOUBLE:
|
||
|
ok = sscanf(json->value,"%lg%n",&float64v,&count);
|
||
|
if(ok != 1 || count != strlen(json->value)) goto fail;
|
||
|
printf("%lg",float64v);
|
||
|
break;
|
||
|
case NCJ_BOOLEAN:
|
||
|
if(strcasecmp(json->value,"true") != 0
|
||
|
&& strcasecmp(json->value,"false") != 0) goto fail;
|
||
|
printf("%s",json->value);
|
||
|
break;
|
||
|
case NCJ_NULL:
|
||
|
printf("null");
|
||
|
break;
|
||
|
case NCJ_DICT:
|
||
|
if(nclistlength(json->contents) == 0) {
|
||
|
printf("{}");
|
||
|
} else {
|
||
|
printf("\n");
|
||
|
for(i=0;i<nclistlength(json->contents);i+=2) {
|
||
|
NCjson* j = NULL;
|
||
|
j = (NCjson*)nclistget(json->contents,i);
|
||
|
assert(j->sort == NCJ_STRING);
|
||
|
printf("{%d} ",depth+1);
|
||
|
printf("\"%s\" => ",j->value);
|
||
|
if(i+1 >= nclistlength(json->contents)) {/* malformed */
|
||
|
printf("<malformed>");
|
||
|
} else
|
||
|
dumpR((NCjson*)nclistget(json->contents,i+1),depth+1);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case NCJ_ARRAY:
|
||
|
if(nclistlength(json->contents) == 0) {
|
||
|
printf("[]");
|
||
|
} else {
|
||
|
printf("\n");
|
||
|
for(i=0;i<nclistlength(json->contents);i++) {
|
||
|
printf("[%d] ",depth+1);
|
||
|
dumpR((NCjson*)nclistget(json->contents,i),depth+1);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
default: printf("unknown"); break;
|
||
|
}
|
||
|
printf("\n");
|
||
|
return;
|
||
|
fail:
|
||
|
printf("????\n");
|
||
|
}
|
||
|
|
||
|
static char*
|
||
|
sortname(int sort)
|
||
|
{
|
||
|
switch (sort) {
|
||
|
case NCJ_STRING: return "String";
|
||
|
case NCJ_INT: return "Int";
|
||
|
case NCJ_DOUBLE: return "Double";
|
||
|
case NCJ_BOOLEAN: return "Boolean";
|
||
|
case NCJ_NULL: return "Null";
|
||
|
case NCJ_DICT: return "Dict";
|
||
|
case NCJ_ARRAY: return "Array";
|
||
|
default: break;
|
||
|
}
|
||
|
return "Unknown";
|
||
|
}
|