cJSON.c 18.4 KB
Newer Older
D
Dave Gamble 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
  Copyright (c) 2009 Dave Gamble

  Permission is hereby granted, free of charge, to any person obtaining a copy
  of this software and associated documentation files (the "Software"), to deal
  in the Software without restriction, including without limitation the rights
  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  copies of the Software, and to permit persons to whom the Software is
  furnished to do so, subject to the following conditions:

  The above copyright notice and this permission notice shall be included in
  all copies or substantial portions of the Software.

  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  THE SOFTWARE.
*/

23 24
/* cJSON */
/* JSON parser in C. */
D
Dave Gamble 已提交
25 26 27 28 29 30

#include <string.h>
#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <float.h>
31
#include <limits.h>
D
Dave Gamble 已提交
32
#include <ctype.h>
D
Dave Gamble 已提交
33 34
#include "cJSON.h"

35 36
static int cJSON_strcasecmp(const char *s1,const char *s2)
{
37
	if (!s1) return (s1==s2)?0:1;if (!s2) return 1;
38 39 40
	for(; tolower(*s1) == tolower(*s2); ++s1, ++s2)	if(*s1 == 0)	return 0;
	return tolower(*(const unsigned char *)s1) - tolower(*(const unsigned char *)s2);
}
41

42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67
static void *(*cJSON_malloc)(size_t sz) = malloc;
static void (*cJSON_free)(void *ptr) = free;

static char* cJSON_strdup(const char* str)
{
      size_t len;
      char* copy;

      len = strlen(str) + 1;
      if (!(copy = (char*)cJSON_malloc(len))) return 0;
      memcpy(copy,str,len);
      return copy;
}

void cJSON_InitHooks(cJSON_Hooks* hooks)
{
    if (!hooks) { /* Reset hooks */
        cJSON_malloc = malloc;
        cJSON_free = free;
        return;
    }

	cJSON_malloc = (hooks->malloc_fn)?hooks->malloc_fn:malloc;
	cJSON_free	 = (hooks->free_fn)?hooks->free_fn:free;
}

68
/* Internal constructor. */
69 70 71 72 73 74
static cJSON *cJSON_New_Item()
{
	cJSON* node = (cJSON*)cJSON_malloc(sizeof(cJSON));
	if (node) memset(node,0,sizeof(cJSON));
	return node;
}
D
Dave Gamble 已提交
75

76
/* Delete a cJSON structure. */
D
Dave Gamble 已提交
77 78 79 80 81 82
void cJSON_Delete(cJSON *c)
{
	cJSON *next;
	while (c)
	{
		next=c->next;
D
Dave Gamble 已提交
83 84
		if (!(c->type&cJSON_IsReference) && c->child) cJSON_Delete(c->child);
		if (!(c->type&cJSON_IsReference) && c->valuestring) cJSON_free(c->valuestring);
85 86
		if (c->string) cJSON_free(c->string);
		cJSON_free(c);
D
Dave Gamble 已提交
87 88 89 90
		c=next;
	}
}

91
/* Parse the input text to generate a number, and populate the result into item. */
D
Dave Gamble 已提交
92 93 94 95
static const char *parse_number(cJSON *item,const char *num)
{
	double n=0,sign=1,scale=0;int subscale=0,signsubscale=1;

96 97 98 99 100 101 102 103
	/* Could use sscanf for this? */
	if (*num=='-') sign=-1,num++;	/* Has sign? */
	if (*num=='0') num++;			/* is zero */
	if (*num>='1' && *num<='9')	do	n=(n*10.0)+(*num++ -'0');	while (*num>='0' && *num<='9');	/* Number? */
	if (*num=='.') {num++;		do	n=(n*10.0)+(*num++ -'0'),scale--; while (*num>='0' && *num<='9');}	/* Fractional part? */
	if (*num=='e' || *num=='E')		/* Exponent? */
	{	num++;if (*num=='+') num++;	else if (*num=='-') signsubscale=-1,num++;		/* With sign? */
		while (*num>='0' && *num<='9') subscale=(subscale*10)+(*num++ - '0');	/* Number? */
D
Dave Gamble 已提交
104 105
	}

106
	n=sign*n*pow(10.0,(scale+subscale*signsubscale));	/* number = +/- number.fraction * 10^+/- exponent */
D
Dave Gamble 已提交
107 108 109 110 111 112 113
	
	item->valuedouble=n;
	item->valueint=(int)n;
	item->type=cJSON_Number;
	return num;
}

114
/* Render the number nicely from the given item into a string. */
D
Dave Gamble 已提交
115 116 117 118
static char *print_number(cJSON *item)
{
	char *str;
	double d=item->valuedouble;
119
	if (fabs(((double)item->valueint)-d)<=DBL_EPSILON && d<=INT_MAX && d>=INT_MIN)
D
Dave Gamble 已提交
120
	{
121
		str=(char*)cJSON_malloc(21);	/* 2^64+1 can be represented in 21 chars. */
122
		if (str) sprintf(str,"%d",item->valueint);
D
Dave Gamble 已提交
123 124 125
	}
	else
	{
126
		str=(char*)cJSON_malloc(64);	/* This is a nice tradeoff. */
127 128 129 130 131 132
		if (str)
		{
			if (fabs(floor(d)-d)<=DBL_EPSILON)			sprintf(str,"%.0f",d);
			else if (fabs(d)<1.0e-6 || fabs(d)>1.0e9)	sprintf(str,"%e",d);
			else										sprintf(str,"%f",d);
		}
D
Dave Gamble 已提交
133 134 135 136
	}
	return str;
}

137
/* Parse the input text into an unescaped cstring, and populate item. */
138
static const unsigned char firstByteMark[7] = { 0x00, 0x00, 0xC0, 0xE0, 0xF0, 0xF8, 0xFC };
D
Dave Gamble 已提交
139 140 141
static const char *parse_string(cJSON *item,const char *str)
{
	const char *ptr=str+1;char *ptr2;char *out;int len=0;unsigned uc;
142
	if (*str!='\"') return 0;	/* not a string! */
D
Dave Gamble 已提交
143
	
144
	while (*ptr!='\"' && (unsigned char)*ptr>31 && ++len) if (*ptr++ == '\\') ptr++;	/* Skip escaped quotes. */
D
Dave Gamble 已提交
145
	
146
	out=(char*)cJSON_malloc(len+1);	/* This is how long we need for the string, roughly. */
147
	if (!out) return 0;
D
Dave Gamble 已提交
148 149
	
	ptr=str+1;ptr2=out;
150
	while (*ptr!='\"' && (unsigned char)*ptr>31)
D
Dave Gamble 已提交
151 152 153 154 155 156 157 158 159 160 161 162
	{
		if (*ptr!='\\') *ptr2++=*ptr++;
		else
		{
			ptr++;
			switch (*ptr)
			{
				case 'b': *ptr2++='\b';	break;
				case 'f': *ptr2++='\f';	break;
				case 'n': *ptr2++='\n';	break;
				case 'r': *ptr2++='\r';	break;
				case 't': *ptr2++='\t';	break;
163 164
				case 'u':	 /* transcode utf16 to utf8. DOES NOT SUPPORT SURROGATE PAIRS CORRECTLY. */
					sscanf(ptr+1,"%4x",&uc);	/* get the unicode char. */
D
Dave Gamble 已提交
165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185
					len=3;if (uc<0x80) len=1;else if (uc<0x800) len=2;ptr2+=len;
					
					switch (len) {
						case 3: *--ptr2 =((uc | 0x80) & 0xBF); uc >>= 6;
						case 2: *--ptr2 =((uc | 0x80) & 0xBF); uc >>= 6;
						case 1: *--ptr2 =(uc | firstByteMark[len]);
					}
					ptr2+=len;ptr+=4;
					break;
				default:  *ptr2++=*ptr; break;
			}
			ptr++;
		}
	}
	*ptr2=0;
	if (*ptr=='\"') ptr++;
	item->valuestring=out;
	item->type=cJSON_String;
	return ptr;
}

186
/* Render the cstring provided to an escaped version that can be printed. */
D
Dave Gamble 已提交
187 188 189 190
static char *print_string_ptr(const char *str)
{
	const char *ptr;char *ptr2,*out;int len=0;
	
191
	if (!str) return cJSON_strdup("");
192
	ptr=str;while (*ptr && ++len) {if ((unsigned char)*ptr<32 || *ptr=='\"' || *ptr=='\\') len++;ptr++;}
D
Dave Gamble 已提交
193
	
194
	out=(char*)cJSON_malloc(len+3);
195 196
	if (!out) return 0;

D
Dave Gamble 已提交
197 198 199 200
	ptr2=out;ptr=str;
	*ptr2++='\"';
	while (*ptr)
	{
201
		if ((unsigned char)*ptr>31 && *ptr!='\"' && *ptr!='\\') *ptr2++=*ptr++;
D
Dave Gamble 已提交
202 203 204 205 206 207 208 209 210 211 212 213
		else
		{
			*ptr2++='\\';
			switch (*ptr++)
			{
				case '\\':	*ptr2++='\\';	break;
				case '\"':	*ptr2++='\"';	break;
				case '\b':	*ptr2++='b';	break;
				case '\f':	*ptr2++='f';	break;
				case '\n':	*ptr2++='n';	break;
				case '\r':	*ptr2++='r';	break;
				case '\t':	*ptr2++='t';	break;
214
				default: ptr2--;	break;	/* eviscerate with prejudice. */
D
Dave Gamble 已提交
215 216 217 218 219 220
			}
		}
	}
	*ptr2++='\"';*ptr2++=0;
	return out;
}
221
/* Invote print_string_ptr (which is useful) on an item. */
D
Dave Gamble 已提交
222 223
static char *print_string(cJSON *item)	{return print_string_ptr(item->valuestring);}

224
/* Predeclare these prototypes. */
D
Dave Gamble 已提交
225
static const char *parse_value(cJSON *item,const char *value);
D
Dave Gamble 已提交
226
static char *print_value(cJSON *item,int depth,int fmt);
D
Dave Gamble 已提交
227
static const char *parse_array(cJSON *item,const char *value);
D
Dave Gamble 已提交
228
static char *print_array(cJSON *item,int depth,int fmt);
D
Dave Gamble 已提交
229
static const char *parse_object(cJSON *item,const char *value);
D
Dave Gamble 已提交
230
static char *print_object(cJSON *item,int depth,int fmt);
D
Dave Gamble 已提交
231

232
/* Utility to jump whitespace and cr/lf */
D
Dave Gamble 已提交
233
static const char *skip(const char *in) {while (in && *in && (unsigned char)*in<=32) in++; return in;}
234

235
/* Parse an object - create a new root, and populate. */
236 237 238 239 240 241 242 243 244
cJSON *cJSON_Parse(const char *value)
{
	cJSON *c=cJSON_New_Item();
	if (!c) return 0;       /* memory fail */

	if (!parse_value(c,skip(value))) {cJSON_Delete(c);return 0;}
	return c;
}

245
/* Render a cJSON item/entity/structure to text. */
D
Dave Gamble 已提交
246 247
char *cJSON_Print(cJSON *item)				{return print_value(item,0,1);}
char *cJSON_PrintUnformatted(cJSON *item)	{return print_value(item,0,0);}
D
Dave Gamble 已提交
248

249
/* Parser core - when encountering text, process appropriately. */
D
Dave Gamble 已提交
250 251
static const char *parse_value(cJSON *item,const char *value)
{
252
	if (!value)						return 0;	/* Fail on null. */
D
Dave Gamble 已提交
253 254 255 256 257 258 259 260
	if (!strncmp(value,"null",4))	{ item->type=cJSON_NULL;  return value+4; }
	if (!strncmp(value,"false",5))	{ item->type=cJSON_False; return value+5; }
	if (!strncmp(value,"true",4))	{ item->type=cJSON_True; item->valueint=1;	return value+4; }
	if (*value=='\"')				{ return parse_string(item,value); }
	if (*value=='-' || (*value>='0' && *value<='9'))	{ return parse_number(item,value); }
	if (*value=='[')				{ return parse_array(item,value); }
	if (*value=='{')				{ return parse_object(item,value); }

261
	return 0;	/* failure. */
D
Dave Gamble 已提交
262 263
}

264
/* Render a value to text. */
D
Dave Gamble 已提交
265
static char *print_value(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
266 267
{
	char *out=0;
D
Dave Gamble 已提交
268
	if (!item) return 0;
D
Dave Gamble 已提交
269
	switch ((item->type)&255)
D
Dave Gamble 已提交
270
	{
271 272 273
		case cJSON_NULL:	out=cJSON_strdup("null");	break;
		case cJSON_False:	out=cJSON_strdup("false");break;
		case cJSON_True:	out=cJSON_strdup("true"); break;
D
Dave Gamble 已提交
274 275
		case cJSON_Number:	out=print_number(item);break;
		case cJSON_String:	out=print_string(item);break;
D
Dave Gamble 已提交
276 277
		case cJSON_Array:	out=print_array(item,depth,fmt);break;
		case cJSON_Object:	out=print_object(item,depth,fmt);break;
D
Dave Gamble 已提交
278 279 280 281
	}
	return out;
}

282
/* Build an array from input text. */
D
Dave Gamble 已提交
283 284 285
static const char *parse_array(cJSON *item,const char *value)
{
	cJSON *child;
286
	if (*value!='[')	return 0;	/* not an array! */
D
Dave Gamble 已提交
287 288 289

	item->type=cJSON_Array;
	value=skip(value+1);
290
	if (*value==']') return value+1;	/* empty array. */
D
Dave Gamble 已提交
291 292

	item->child=child=cJSON_New_Item();
293 294
	if (!item->child) return 0;		 /* memory fail */
	value=skip(parse_value(child,skip(value)));	/* skip any spacing, get the value. */
295
	if (!value) return 0;
D
Dave Gamble 已提交
296 297 298 299

	while (*value==',')
	{
		cJSON *new_item;
300
		if (!(new_item=cJSON_New_Item())) return 0; 	/* memory fail */
D
Dave Gamble 已提交
301 302
		child->next=new_item;new_item->prev=child;child=new_item;
		value=skip(parse_value(child,skip(value+1)));
303
		if (!value) return 0;	/* memory fail */
D
Dave Gamble 已提交
304 305
	}

306 307
	if (*value==']') return value+1;	/* end of array */
	return 0;	/* malformed. */
D
Dave Gamble 已提交
308 309
}

310
/* Render an array to text */
D
Dave Gamble 已提交
311
static char *print_array(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
312
{
313 314
	char **entries;
	char *out=0,*ptr,*ret;int len=5;
D
Dave Gamble 已提交
315
	cJSON *child=item->child;
316
	int numentries=0,i=0,fail=0;
D
Dave Gamble 已提交
317
	
318
	/* How many entries in the array? */
319
	while (child) numentries++,child=child->next;
320
	/* Allocate an array to hold the values for each */
321 322 323
	entries=(char**)cJSON_malloc(numentries*sizeof(char*));
	if (!entries) return 0;
	memset(entries,0,numentries*sizeof(char*));
324
	/* Retrieve all the results: */
325 326
	child=item->child;
	while (child && !fail)
D
Dave Gamble 已提交
327
	{
D
Dave Gamble 已提交
328
		ret=print_value(child,depth+1,fmt);
329
		entries[i++]=ret;
D
Dave Gamble 已提交
330
		if (ret) len+=strlen(ret)+2+(fmt?1:0); else fail=1;
D
Dave Gamble 已提交
331 332
		child=child->next;
	}
333
	
334
	/* If we didn't fail, try to malloc the output string */
335
	if (!fail) out=cJSON_malloc(len);
336
	/* If that fails, we fail. */
337 338
	if (!out) fail=1;

339
	/* Handle failure. */
340 341 342 343 344 345 346
	if (fail)
	{
		for (i=0;i<numentries;i++) if (entries[i]) cJSON_free(entries[i]);
		cJSON_free(entries);
		return 0;
	}
	
347
	/* Compose the output array. */
348 349 350 351 352
	*out='[';
	ptr=out+1;*ptr=0;
	for (i=0;i<numentries;i++)
	{
		strcpy(ptr,entries[i]);ptr+=strlen(entries[i]);
D
Dave Gamble 已提交
353
		if (i!=numentries-1) {*ptr++=',';if(fmt)*ptr++=' ';*ptr=0;}
354 355 356
		cJSON_free(entries[i]);
	}
	cJSON_free(entries);
D
Dave Gamble 已提交
357 358 359 360
	*ptr++=']';*ptr++=0;
	return out;	
}

361
/* Build an object from the text. */
D
Dave Gamble 已提交
362 363 364
static const char *parse_object(cJSON *item,const char *value)
{
	cJSON *child;
365
	if (*value!='{')	return 0;	/* not an object! */
D
Dave Gamble 已提交
366 367 368
	
	item->type=cJSON_Object;
	value=skip(value+1);
369
	if (*value=='}') return value+1;	/* empty array. */
D
Dave Gamble 已提交
370 371
	
	item->child=child=cJSON_New_Item();
372
	if (!item->child) return 0;
D
Dave Gamble 已提交
373
	value=skip(parse_string(child,skip(value)));
374
	if (!value) return 0;
D
Dave Gamble 已提交
375
	child->string=child->valuestring;child->valuestring=0;
376 377
	if (*value!=':') return 0;	/* fail! */
	value=skip(parse_value(child,skip(value+1)));	/* skip any spacing, get the value. */
378
	if (!value) return 0;
D
Dave Gamble 已提交
379 380 381 382
	
	while (*value==',')
	{
		cJSON *new_item;
383
		if (!(new_item=cJSON_New_Item()))	return 0; /* memory fail */
D
Dave Gamble 已提交
384 385
		child->next=new_item;new_item->prev=child;child=new_item;
		value=skip(parse_string(child,skip(value+1)));
386
		if (!value) return 0;
D
Dave Gamble 已提交
387
		child->string=child->valuestring;child->valuestring=0;
388 389
		if (*value!=':') return 0;	/* fail! */
		value=skip(parse_value(child,skip(value+1)));	/* skip any spacing, get the value. */
390
		if (!value) return 0;
D
Dave Gamble 已提交
391 392
	}
	
393 394
	if (*value=='}') return value+1;	/* end of array */
	return 0;	/* malformed. */
D
Dave Gamble 已提交
395 396
}

397
/* Render an object to text. */
D
Dave Gamble 已提交
398
static char *print_object(cJSON *item,int depth,int fmt)
D
Dave Gamble 已提交
399
{
400 401
	char **entries=0,**names=0;
	char *out=0,*ptr,*ret,*str;int len=7,i=0,j;
D
Dave Gamble 已提交
402
	cJSON *child=item->child;
403
	int numentries=0,fail=0;
404
	/* Count the number of entries. */
405
	while (child) numentries++,child=child->next;
406
	/* Allocate space for the names and the objects */
407 408 409 410 411 412 413
	entries=(char**)cJSON_malloc(numentries*sizeof(char*));
	if (!entries) return 0;
	names=(char**)cJSON_malloc(numentries*sizeof(char*));
	if (!names) {cJSON_free(entries);return 0;}
	memset(entries,0,sizeof(char*)*numentries);
	memset(names,0,sizeof(char*)*numentries);

414
	/* Collect all the results into our arrays: */
D
Dave Gamble 已提交
415
	child=item->child;depth++;if (fmt) len+=depth;
D
Dave Gamble 已提交
416 417
	while (child)
	{
418
		names[i]=str=print_string_ptr(child->string);
D
Dave Gamble 已提交
419 420
		entries[i++]=ret=print_value(child,depth,fmt);
		if (str && ret) len+=strlen(ret)+strlen(str)+2+(fmt?2+depth:0); else fail=1;
421 422 423
		child=child->next;
	}
	
424
	/* Try to allocate the output string */
425 426 427
	if (!fail) out=(char*)cJSON_malloc(len);
	if (!out) fail=1;

428
	/* Handle failure */
429 430
	if (fail)
	{
431 432
		for (i=0;i<numentries;i++) {if (names[i]) cJSON_free(names[i]);if (entries[i]) cJSON_free(entries[i]);}
		cJSON_free(names);cJSON_free(entries);
433 434 435
		return 0;
	}
	
436
	/* Compose the output: */
D
Dave Gamble 已提交
437
	*out='{';ptr=out+1;if (fmt)*ptr++='\n';*ptr=0;
438 439
	for (i=0;i<numentries;i++)
	{
D
Dave Gamble 已提交
440
		if (fmt) for (j=0;j<depth;j++) *ptr++='\t';
441
		strcpy(ptr,names[i]);ptr+=strlen(names[i]);
D
Dave Gamble 已提交
442
		*ptr++=':';if (fmt) *ptr++='\t';
443 444
		strcpy(ptr,entries[i]);ptr+=strlen(entries[i]);
		if (i!=numentries-1) *ptr++=',';
D
Dave Gamble 已提交
445
		if (fmt) *ptr++='\n';*ptr=0;
446
		cJSON_free(names[i]);cJSON_free(entries[i]);
D
Dave Gamble 已提交
447
	}
448 449
	
	cJSON_free(names);cJSON_free(entries);
D
Dave Gamble 已提交
450
	if (fmt) for (i=0;i<depth-1;i++) *ptr++='\t';
D
Dave Gamble 已提交
451 452 453 454
	*ptr++='}';*ptr++=0;
	return out;	
}

455
/* Get Array size/item / object item. */
D
Dave Gamble 已提交
456
int    cJSON_GetArraySize(cJSON *array)							{cJSON *c=array->child;int i=0;while(c)i++,c=c->next;return i;}
457
cJSON *cJSON_GetArrayItem(cJSON *array,int item)				{cJSON *c=array->child;  while (c && item>0) item--,c=c->next; return c;}
458
cJSON *cJSON_GetObjectItem(cJSON *object,const char *string)	{cJSON *c=object->child; while (c && cJSON_strcasecmp(c->string,string)) c=c->next; return c;}
D
Dave Gamble 已提交
459

460
/* Utility for array list handling. */
D
Dave Gamble 已提交
461
static void suffix_object(cJSON *prev,cJSON *item) {prev->next=item;item->prev=prev;}
462
/* Utility for handling references. */
463
static cJSON *create_reference(cJSON *item) {cJSON *ref=cJSON_New_Item();if (!ref) return 0;memcpy(ref,item,sizeof(cJSON));ref->string=0;ref->type|=cJSON_IsReference;ref->next=ref->prev=0;return ref;}
D
Dave Gamble 已提交
464

465
/* Add item to array/object. */
466
void   cJSON_AddItemToArray(cJSON *array, cJSON *item)						{cJSON *c=array->child;if (!item) return; if (!c) {array->child=item;} else {while (c && c->next) c=c->next; suffix_object(c,item);}}
467
void   cJSON_AddItemToObject(cJSON *object,const char *string,cJSON *item)	{if (!item) return; if (item->string) cJSON_free(item->string);item->string=cJSON_strdup(string);cJSON_AddItemToArray(object,item);}
D
Dave Gamble 已提交
468 469 470
void	cJSON_AddItemReferenceToArray(cJSON *array, cJSON *item)						{cJSON_AddItemToArray(array,create_reference(item));}
void	cJSON_AddItemReferenceToObject(cJSON *object,const char *string,cJSON *item)	{cJSON_AddItemToObject(object,string,create_reference(item));}

D
Dave Gamble 已提交
471
cJSON *cJSON_DetachItemFromArray(cJSON *array,int which)			{cJSON *c=array->child;while (c && which>0) c=c->next,which--;if (!c) return 0;
472
	if (c->prev) c->prev->next=c->next;if (c->next) c->next->prev=c->prev;if (c==array->child) array->child=c->next;c->prev=c->next=0;return c;}
473
void   cJSON_DeleteItemFromArray(cJSON *array,int which)			{cJSON_Delete(cJSON_DetachItemFromArray(array,which));}
474
cJSON *cJSON_DetachItemFromObject(cJSON *object,const char *string) {int i=0;cJSON *c=object->child;while (c && cJSON_strcasecmp(c->string,string)) i++,c=c->next;if (c) return cJSON_DetachItemFromArray(object,i);return 0;}
475
void   cJSON_DeleteItemFromObject(cJSON *object,const char *string) {cJSON_Delete(cJSON_DetachItemFromObject(object,string));}
D
Dave Gamble 已提交
476

477
/* Replace array/object items with new ones. */
D
Dave Gamble 已提交
478 479
void   cJSON_ReplaceItemInArray(cJSON *array,int which,cJSON *newitem)		{cJSON *c=array->child;while (c && which>0) c=c->next,which--;if (!c) return;
	newitem->next=c->next;newitem->prev=c->prev;if (newitem->next) newitem->next->prev=newitem;
480
	if (c==array->child) array->child=newitem; else newitem->prev->next=newitem;c->next=c->prev=0;cJSON_Delete(c);}
481
void   cJSON_ReplaceItemInObject(cJSON *object,const char *string,cJSON *newitem){int i=0;cJSON *c=object->child;while(c && cJSON_strcasecmp(c->string,string))i++,c=c->next;if(c){newitem->string=cJSON_strdup(string);cJSON_ReplaceItemInArray(object,i,newitem);}}
D
Dave Gamble 已提交
482

483
/* Create basic types: */
484 485 486
cJSON *cJSON_CreateNull()						{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_NULL;return item;}
cJSON *cJSON_CreateTrue()						{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_True;return item;}
cJSON *cJSON_CreateFalse()						{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_False;return item;}
D
Dave Gamble 已提交
487
cJSON *cJSON_CreateBool(int b)					{cJSON *item=cJSON_New_Item();if(item)item->type=b?cJSON_True:cJSON_False;return item;}
488 489 490 491
cJSON *cJSON_CreateNumber(double num)			{cJSON *item=cJSON_New_Item();if(item){item->type=cJSON_Number;item->valuedouble=num;item->valueint=(int)num;}return item;}
cJSON *cJSON_CreateString(const char *string)	{cJSON *item=cJSON_New_Item();if(item){item->type=cJSON_String;item->valuestring=cJSON_strdup(string);}return item;}
cJSON *cJSON_CreateArray()						{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_Array;return item;}
cJSON *cJSON_CreateObject()						{cJSON *item=cJSON_New_Item();if(item)item->type=cJSON_Object;return item;}
D
Dave Gamble 已提交
492

493
/* Create Arrays: */
494 495 496 497
cJSON *cJSON_CreateIntArray(int *numbers,int count)				{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateNumber(numbers[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}
cJSON *cJSON_CreateFloatArray(float *numbers,int count)			{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateNumber(numbers[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}
cJSON *cJSON_CreateDoubleArray(double *numbers,int count)		{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateNumber(numbers[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}
cJSON *cJSON_CreateStringArray(const char **strings,int count)	{int i;cJSON *n=0,*p=0,*a=cJSON_CreateArray();for(i=0;a && i<count;i++){n=cJSON_CreateString(strings[i]);if(!i)a->child=n;else suffix_object(p,n);p=n;}return a;}