123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836 |
- #include <ctype.h>
- #include <string.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include "cJSON_Utils.h"
- static char* cJSONUtils_strdup(const char* str)
- {
- size_t len = 0;
- char *copy = NULL;
- len = strlen(str) + 1;
- if (!(copy = (char*)malloc(len)))
- {
- return NULL;
- }
- memcpy(copy, str, len);
- return copy;
- }
- static int cJSONUtils_strcasecmp(const char *s1, const char *s2)
- {
- if (!s1)
- {
- return (s1 == s2) ? 0 : 1; /* both NULL? */
- }
- if (!s2)
- {
- return 1;
- }
- for (; tolower(*s1) == tolower(*s2); ++s1, ++s2)
- {
- if (*s1 == 0)
- {
- return 0;
- }
- }
- return tolower(*(const unsigned char *)s1) - tolower(*(const unsigned char *)s2);
- }
- /* JSON Pointer implementation: */
- static int cJSONUtils_Pstrcasecmp(const char *a, const char *e)
- {
- if (!a || !e)
- {
- return (a == e) ? 0 : 1; /* both NULL? */
- }
- for (; *a && *e && (*e != '/'); a++, e++) /* compare until next '/' */
- {
- if (*e == '~')
- {
- /* check for escaped '~' (~0) and '/' (~1) */
- if (!((e[1] == '0') && (*a == '~')) && !((e[1] == '1') && (*a == '/')))
- {
- /* invalid escape sequence or wrong character in *a */
- return 1;
- }
- else
- {
- e++;
- }
- }
- else if (tolower(*a) != tolower(*e))
- {
- return 1;
- }
- }
- if (((*e != 0) && (*e != '/')) != (*a != 0))
- {
- /* one string has ended, the other not */
- return 1;
- }
- return 0;
- }
- static int cJSONUtils_PointerEncodedstrlen(const char *s)
- {
- int l = 0;
- for (; *s; s++, l++)
- {
- if ((*s == '~') || (*s == '/'))
- {
- l++;
- }
- }
- return l;
- }
- static void cJSONUtils_PointerEncodedstrcpy(char *d, const char *s)
- {
- for (; *s; s++)
- {
- if (*s == '/')
- {
- *d++ = '~';
- *d++ = '1';
- }
- else if (*s == '~')
- {
- *d++ = '~';
- *d++ = '0';
- }
- else
- {
- *d++ = *s;
- }
- }
- *d = '\0';
- }
- char *cJSONUtils_FindPointerFromObjectTo(cJSON *object, cJSON *target)
- {
- int type = object->type;
- int c = 0;
- cJSON *obj = 0;
- if (object == target)
- {
- /* found */
- return cJSONUtils_strdup("");
- }
- int nlen = 0;
- /* recursively search all children of the object */
- for (obj = object->child; obj; obj = obj->next, c++)
- {
- char *found = cJSONUtils_FindPointerFromObjectTo(obj, target);
- if (found)
- {
- if ((type & 0xFF) == cJSON_Array)
- {
- /* reserve enough memory for a 64 bit integer + '/' and '\0' */
- nlen = strlen(found) + 23;
- char *ret = (char*)malloc(nlen);
- sprintf_s(ret, nlen, "/%d%s", c, found); /* /<array_index><path> */
- free(found);
- return ret;
- }
- else if ((type & 0xFF) == cJSON_Object)
- {
- nlen = strlen(found) + cJSONUtils_PointerEncodedstrlen(obj->string) + 2;
- char *ret = (char*)malloc(nlen);
- *ret = '/';
- cJSONUtils_PointerEncodedstrcpy(ret + 1, obj->string);
- strcat_s(ret, nlen, found);
- free(found);
- return ret;
- }
- /* reached leaf of the tree, found nothing */
- free(found);
- return NULL;
- }
- }
- /* not found */
- return NULL;
- }
- cJSON *cJSONUtils_GetPointer(cJSON *object, const char *pointer)
- {
- /* follow path of the pointer */
- while ((*pointer++ == '/') && object)
- {
- if ((object->type & 0xFF) == cJSON_Array)
- {
- int which = 0;
- /* parse array index */
- while ((*pointer >= '0') && (*pointer <= '9'))
- {
- which = (10 * which) + (*pointer++ - '0');
- }
- if (*pointer && (*pointer != '/'))
- {
- /* not end of string or new path token */
- return NULL;
- }
- object = cJSON_GetArrayItem(object, which);
- }
- else if ((object->type & 0xFF) == cJSON_Object)
- {
- object = object->child;
- /* GetObjectItem. */
- while (object && cJSONUtils_Pstrcasecmp(object->string, pointer))
- {
- object = object->next;
- }
- /* skip to the next path token or end of string */
- while (*pointer && (*pointer != '/'))
- {
- pointer++;
- }
- }
- else
- {
- return NULL;
- }
- }
- return object;
- }
- /* JSON Patch implementation. */
- static void cJSONUtils_InplaceDecodePointerString(char *string)
- {
- char *s2 = string;
- for (; *string; s2++, string++)
- {
- *s2 = (*string != '~')
- ? (*string)
- : ((*(++string) == '0')
- ? '~'
- : '/');
- }
- *s2 = '\0';
- }
- static cJSON *cJSONUtils_PatchDetach(cJSON *object, const char *path)
- {
- char *parentptr = NULL;
- char *childptr = NULL;
- cJSON *parent = NULL;
- cJSON *ret = NULL;
- /* copy path and split it in parent and child */
- parentptr = cJSONUtils_strdup(path);
- childptr = strrchr(parentptr, '/'); /* last '/' */
- if (childptr)
- {
- /* split strings */
- *childptr++ = '\0';
- }
- parent = cJSONUtils_GetPointer(object, parentptr);
- cJSONUtils_InplaceDecodePointerString(childptr);
- if (!parent)
- {
- /* Couldn't find object to remove child from. */
- ret = NULL;
- }
- else if ((parent->type & 0xFF) == cJSON_Array)
- {
- ret = cJSON_DetachItemFromArray(parent, atoi(childptr));
- }
- else if ((parent->type & 0xFF) == cJSON_Object)
- {
- ret = cJSON_DetachItemFromObject(parent, childptr);
- }
- free(parentptr);
- /* return the detachted item */
- return ret;
- }
- static int cJSONUtils_Compare(cJSON *a, cJSON *b)
- {
- if ((a->type & 0xFF) != (b->type & 0xFF))
- {
- /* mismatched type. */
- return -1;
- }
- switch (a->type & 0xFF)
- {
- case cJSON_Number:
- /* numeric mismatch. */
- return ((a->valueint != b->valueint) || (a->valuedouble != b->valuedouble)) ? -2 : 0;
- case cJSON_String:
- /* string mismatch. */
- return (strcmp(a->valuestring, b->valuestring) != 0) ? -3 : 0;
- case cJSON_Array:
- for (a = a->child, b = b->child; a && b; a = a->next, b = b->next)
- {
- int err = cJSONUtils_Compare(a, b);
- if (err)
- {
- return err;
- }
- }
- /* array size mismatch? (one of both children is not NULL) */
- return (a || b) ? -4 : 0;
- case cJSON_Object:
- cJSONUtils_SortObject(a);
- cJSONUtils_SortObject(b);
- a = a->child;
- b = b->child;
- while (a && b)
- {
- int err = 0;
- /* compare object keys */
- if (cJSONUtils_strcasecmp(a->string, b->string))
- {
- /* missing member */
- return -6;
- }
- err = cJSONUtils_Compare(a, b);
- if (err)
- {
- return err;
- }
- a = a->next;
- b = b->next;
- }
- /* object length mismatch (one of both children is not null) */
- return (a || b) ? -5 : 0;
- default:
- break;
- }
- /* null, true or false */
- return 0;
- }
- static int cJSONUtils_ApplyPatch(cJSON *object, cJSON *patch)
- {
- cJSON *op = NULL;
- cJSON *path = NULL;
- cJSON *value = NULL;
- cJSON *parent = NULL;
- int opcode = 0;
- char *parentptr = NULL;
- char *childptr = NULL;
- op = cJSON_GetObjectItem(patch, "op");
- path = cJSON_GetObjectItem(patch, "path");
- if (!op || !path)
- {
- /* malformed patch. */
- return 2;
- }
- /* decode operation */
- if (!strcmp(op->valuestring, "add"))
- {
- opcode = 0;
- }
- else if (!strcmp(op->valuestring, "remove"))
- {
- opcode = 1;
- }
- else if (!strcmp(op->valuestring, "replace"))
- {
- opcode = 2;
- }
- else if (!strcmp(op->valuestring, "move"))
- {
- opcode = 3;
- }
- else if (!strcmp(op->valuestring, "copy"))
- {
- opcode = 4;
- }
- else if (!strcmp(op->valuestring, "test"))
- {
- /* compare value: {...} with the given path */
- return cJSONUtils_Compare(cJSONUtils_GetPointer(object, path->valuestring), cJSON_GetObjectItem(patch, "value"));
- }
- else
- {
- /* unknown opcode. */
- return 3;
- }
- /* Remove/Replace */
- if ((opcode == 1) || (opcode == 2))
- {
- /* Get rid of old. */
- cJSON_Delete(cJSONUtils_PatchDetach(object, path->valuestring));
- if (opcode == 1)
- {
- /* For Remove, this is job done. */
- return 0;
- }
- }
- /* Copy/Move uses "from". */
- if ((opcode == 3) || (opcode == 4))
- {
- cJSON *from = cJSON_GetObjectItem(patch, "from");
- if (!from)
- {
- /* missing "from" for copy/move. */
- return 4;
- }
- if (opcode == 3)
- {
- /* move */
- value = cJSONUtils_PatchDetach(object, from->valuestring);
- }
- if (opcode == 4)
- {
- /* copy */
- value = cJSONUtils_GetPointer(object, from->valuestring);
- }
- if (!value)
- {
- /* missing "from" for copy/move. */
- return 5;
- }
- if (opcode == 4)
- {
- value = cJSON_Duplicate(value, 1);
- }
- if (!value)
- {
- /* out of memory for copy/move. */
- return 6;
- }
- }
- else /* Add/Replace uses "value". */
- {
- value = cJSON_GetObjectItem(patch, "value");
- if (!value)
- {
- /* missing "value" for add/replace. */
- return 7;
- }
- value = cJSON_Duplicate(value, 1);
- if (!value)
- {
- /* out of memory for add/replace. */
- return 8;
- }
- }
- /* Now, just add "value" to "path". */
- /* split pointer in parent and child */
- parentptr = cJSONUtils_strdup(path->valuestring);
- childptr = strrchr(parentptr, '/');
- if (childptr)
- {
- *childptr++ = '\0';
- }
- parent = cJSONUtils_GetPointer(object, parentptr);
- cJSONUtils_InplaceDecodePointerString(childptr);
- /* add, remove, replace, move, copy, test. */
- if (!parent)
- {
- /* Couldn't find object to add to. */
- free(parentptr);
- cJSON_Delete(value);
- return 9;
- }
- else if ((parent->type & 0xFF) == cJSON_Array)
- {
- if (!strcmp(childptr, "-"))
- {
- cJSON_AddItemToArray(parent, value);
- }
- else
- {
- cJSON_InsertItemInArray(parent, atoi(childptr), value);
- }
- }
- else if ((parent->type & 0xFF) == cJSON_Object)
- {
- cJSON_DeleteItemFromObject(parent, childptr);
- cJSON_AddItemToObject(parent, childptr, value);
- }
- else
- {
- cJSON_Delete(value);
- }
- free(parentptr);
- return 0;
- }
- int cJSONUtils_ApplyPatches(cJSON *object, cJSON *patches)
- {
- int err = 0;
- if ((patches->type & 0xFF) != cJSON_Array)
- {
- /* malformed patches. */
- return 1;
- }
- if (patches)
- {
- patches = patches->child;
- }
- while (patches)
- {
- if ((err = cJSONUtils_ApplyPatch(object, patches)))
- {
- return err;
- }
- patches = patches->next;
- }
- return 0;
- }
- static void cJSONUtils_GeneratePatch(cJSON *patches, const char *op, const char *path, const char *suffix, cJSON *val)
- {
- int nlen = 0;
- cJSON *patch = cJSON_CreateObject();
- cJSON_AddItemToObject(patch, "op", cJSON_CreateString(op));
- if (suffix)
- {
- nlen = strlen(path) + cJSONUtils_PointerEncodedstrlen(suffix) + 2;
- char *newpath = (char*)malloc(nlen);
- cJSONUtils_PointerEncodedstrcpy(newpath + sprintf_s(newpath, nlen, "%s/", path), suffix);
- cJSON_AddItemToObject(patch, "path", cJSON_CreateString(newpath));
- free(newpath);
- }
- else
- {
- cJSON_AddItemToObject(patch, "path", cJSON_CreateString(path));
- }
- if (val)
- {
- cJSON_AddItemToObject(patch, "value", cJSON_Duplicate(val, 1));
- }
- cJSON_AddItemToArray(patches, patch);
- }
- void cJSONUtils_AddPatchToArray(cJSON *array, const char *op, const char *path, cJSON *val)
- {
- cJSONUtils_GeneratePatch(array, op, path, 0, val);
- }
- static void cJSONUtils_CompareToPatch(cJSON *patches, const char *path, cJSON *from, cJSON *to)
- {
- if ((from->type & 0xFF) != (to->type & 0xFF))
- {
- cJSONUtils_GeneratePatch(patches, "replace", path, 0, to);
- return;
- }
- int nlen = 0;
- switch ((from->type & 0xFF))
- {
- case cJSON_Number:
- if ((from->valueint != to->valueint) || (from->valuedouble != to->valuedouble))
- {
- cJSONUtils_GeneratePatch(patches, "replace", path, 0, to);
- }
- return;
- case cJSON_String:
- if (strcmp(from->valuestring, to->valuestring) != 0)
- {
- cJSONUtils_GeneratePatch(patches, "replace", path, 0, to);
- }
- return;
- case cJSON_Array:
- {
- int c = 0;
- nlen = strlen(path) + 23;
- char *newpath = (char*)malloc(nlen); /* Allow space for 64bit int. */
- /* generate patches for all array elements that exist in "from" and "to" */
- for (c = 0, from = from->child, to = to->child; from && to; from = from->next, to = to->next, c++)
- {
- sprintf_s(newpath, nlen, "%s/%d", path, c); /* path of the current array element */
- cJSONUtils_CompareToPatch(patches, newpath, from, to);
- }
- /* remove leftover elements from 'from' that are not in 'to' */
- for (; from; from = from->next, c++)
- {
- sprintf_s(newpath, nlen, "%d", c);
- cJSONUtils_GeneratePatch(patches, "remove", path, newpath, 0);
- }
- /* add new elements in 'to' that were not in 'from' */
- for (; to; to = to->next, c++)
- {
- cJSONUtils_GeneratePatch(patches, "add", path, "-", to);
- }
- free(newpath);
- return;
- }
- case cJSON_Object:
- {
- cJSON *a = NULL;
- cJSON *b = NULL;
- cJSONUtils_SortObject(from);
- cJSONUtils_SortObject(to);
- a = from->child;
- b = to->child;
- /* for all object values in the object with more of them */
- while (a || b)
- {
- int diff = (!a) ? 1 : ((!b) ? -1 : cJSONUtils_strcasecmp(a->string, b->string));
- if (!diff)
- {
- /* both object keys are the same */
- nlen = strlen(path) + cJSONUtils_PointerEncodedstrlen(a->string) + 2;
- char *newpath = (char*)malloc(nlen);
- cJSONUtils_PointerEncodedstrcpy(newpath + sprintf_s(newpath, nlen, "%s/", path), a->string);
- /* create a patch for the element */
- cJSONUtils_CompareToPatch(patches, newpath, a, b);
- free(newpath);
- a = a->next;
- b = b->next;
- }
- else if (diff < 0)
- {
- /* object element doesn't exist in 'to' --> remove it */
- cJSONUtils_GeneratePatch(patches, "remove", path, a->string, 0);
- a = a->next;
- }
- else
- {
- /* object element doesn't exist in 'from' --> add it */
- cJSONUtils_GeneratePatch(patches, "add", path, b->string, b);
- b = b->next;
- }
- }
- return;
- }
- default:
- break;
- }
- }
- cJSON* cJSONUtils_GeneratePatches(cJSON *from, cJSON *to)
- {
- cJSON *patches = cJSON_CreateArray();
- cJSONUtils_CompareToPatch(patches, "", from, to);
- return patches;
- }
- /* sort lists using mergesort */
- static cJSON *cJSONUtils_SortList(cJSON *list)
- {
- cJSON *first = list;
- cJSON *second = list;
- cJSON *ptr = list;
- if (!list || !list->next)
- {
- /* One entry is sorted already. */
- return list;
- }
- while (ptr && ptr->next && (cJSONUtils_strcasecmp(ptr->string, ptr->next->string) < 0))
- {
- /* Test for list sorted. */
- ptr = ptr->next;
- }
- if (!ptr || !ptr->next)
- {
- /* Leave sorted lists unmodified. */
- return list;
- }
- /* reset ptr to the beginning */
- ptr = list;
- while (ptr)
- {
- /* Walk two pointers to find the middle. */
- second = second->next;
- ptr = ptr->next;
- /* advances ptr two steps at a time */
- if (ptr)
- {
- ptr = ptr->next;
- }
- }
- if (second && second->prev)
- {
- /* Split the lists */
- second->prev->next = NULL;
- }
- /* Recursively sort the sub-lists. */
- first = cJSONUtils_SortList(first);
- second = cJSONUtils_SortList(second);
- list = ptr = NULL;
- while (first && second) /* Merge the sub-lists */
- {
- if (cJSONUtils_strcasecmp(first->string, second->string) < 0)
- {
- if (!list)
- {
- /* start merged list with the first element of the first list */
- list = ptr = first;
- }
- else
- {
- /* add first element of first list to merged list */
- ptr->next = first;
- first->prev = ptr;
- ptr = first;
- }
- first = first->next;
- }
- else
- {
- if (!list)
- {
- /* start merged list with the first element of the second list */
- list = ptr = second;
- }
- else
- {
- /* add first element of second list to merged list */
- ptr->next = second;
- second->prev = ptr;
- ptr = second;
- }
- second = second->next;
- }
- }
- if (first)
- {
- /* Append rest of first list. */
- if (!list)
- {
- return first;
- }
- ptr->next = first;
- first->prev = ptr;
- }
- if (second)
- {
- /* Append rest of second list */
- if (!list)
- {
- return second;
- }
- ptr->next = second;
- second->prev = ptr;
- }
- return list;
- }
- void cJSONUtils_SortObject(cJSON *object)
- {
- object->child = cJSONUtils_SortList(object->child);
- }
- cJSON* cJSONUtils_MergePatch(cJSON *target, cJSON *patch)
- {
- if (!patch || ((patch->type & 0xFF) != cJSON_Object))
- {
- /* scalar value, array or NULL, just duplicate */
- cJSON_Delete(target);
- return cJSON_Duplicate(patch, 1);
- }
- if (!target || ((target->type & 0xFF) != cJSON_Object))
- {
- cJSON_Delete(target);
- target = cJSON_CreateObject();
- }
- patch = patch->child;
- while (patch)
- {
- if ((patch->type & 0xFF) == cJSON_NULL)
- {
- /* NULL is the indicator to remove a value, see RFC7396 */
- cJSON_DeleteItemFromObject(target, patch->string);
- }
- else
- {
- cJSON *replaceme = cJSON_DetachItemFromObject(target, patch->string);
- cJSON_AddItemToObject(target, patch->string, cJSONUtils_MergePatch(replaceme, patch));
- }
- patch = patch->next;
- }
- return target;
- }
- cJSON *cJSONUtils_GenerateMergePatch(cJSON *from, cJSON *to)
- {
- cJSON *patch = NULL;
- if (!to)
- {
- /* patch to delete everything */
- return cJSON_CreateNull();
- }
- if (((to->type & 0xFF) != cJSON_Object) || !from || ((from->type & 0xFF) != cJSON_Object))
- {
- return cJSON_Duplicate(to, 1);
- }
- cJSONUtils_SortObject(from);
- cJSONUtils_SortObject(to);
- from = from->child;
- to = to->child;
- patch = cJSON_CreateObject();
- while (from || to)
- {
- int compare = from ? (to ? strcmp(from->string, to->string) : -1) : 1;
- if (compare < 0)
- {
- /* from has a value that to doesn't have -> remove */
- cJSON_AddItemToObject(patch, from->string, cJSON_CreateNull());
- from = from->next;
- }
- else if (compare > 0)
- {
- /* to has a value that from doesn't have -> add to patch */
- cJSON_AddItemToObject(patch, to->string, cJSON_Duplicate(to, 1));
- to = to->next;
- }
- else
- {
- /* object key exists in both objects */
- if (cJSONUtils_Compare(from, to))
- {
- /* not identical --> generate a patch */
- cJSON_AddItemToObject(patch, to->string, cJSONUtils_GenerateMergePatch(from, to));
- }
- /* next key in the object */
- from = from->next;
- to = to->next;
- }
- }
- if (!patch->child)
- {
- cJSON_Delete(patch);
- return NULL;
- }
- return patch;
- }
|