382 lines
11 KiB
C
382 lines
11 KiB
C
/*
|
|
* Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
|
|
*
|
|
* This program is free software: you can use, redistribute, and/or modify
|
|
* it under the terms of the GNU Affero General Public License, version 3
|
|
* or later ("AGPL"), as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#define _DEFAULT_SOURCE
|
|
|
|
#include "tjson.h"
|
|
#include "cJSON.h"
|
|
#include "taoserror.h"
|
|
|
|
SJson* tjsonCreateObject() {
|
|
SJson* pJson = cJSON_CreateObject();
|
|
if (pJson == NULL) {
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
}
|
|
return pJson;
|
|
}
|
|
|
|
SJson* tjsonCreateArray() {
|
|
SJson* pJson = cJSON_CreateArray();
|
|
if (pJson == NULL) {
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
}
|
|
return pJson;
|
|
}
|
|
|
|
void tjsonDelete(SJson* pJson) {
|
|
if (pJson != NULL) {
|
|
cJSON_Delete((cJSON*)pJson);
|
|
}
|
|
}
|
|
|
|
int32_t tjsonAddIntegerToObject(SJson* pJson, const char* pName, const uint64_t number) {
|
|
char tmp[40] = {0};
|
|
snprintf(tmp, sizeof(tmp), "%" PRId64, number);
|
|
return tjsonAddStringToObject(pJson, pName, tmp);
|
|
}
|
|
|
|
int32_t tjsonAddDoubleToObject(SJson* pJson, const char* pName, const double number) {
|
|
if (NULL == cJSON_AddNumberToObject((cJSON*)pJson, pName, number)) {
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonAddBoolToObject(SJson* pJson, const char* pName, const bool boolean) {
|
|
if (NULL == cJSON_AddBoolToObject((cJSON*)pJson, pName, boolean)) {
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonAddStringToObject(SJson* pJson, const char* pName, const char* pVal) {
|
|
if (NULL == cJSON_AddStringToObject((cJSON*)pJson, pName, pVal)) {
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
SJson* tjsonAddArrayToObject(SJson* pJson, const char* pName) {
|
|
SJson* ret = (SJson*)cJSON_AddArrayToObject((cJSON*)pJson, pName);
|
|
if (ret == NULL) {
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int32_t tjsonAddItemToObject(SJson* pJson, const char* pName, SJson* pItem) {
|
|
if (cJSON_AddItemToObject((cJSON*)pJson, pName, pItem)) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
|
|
int32_t tjsonAddItemToArray(SJson* pJson, SJson* pItem) {
|
|
if (cJSON_AddItemToArray((cJSON*)pJson, pItem)) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
terrno = TSDB_CODE_OUT_OF_MEMORY;
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
|
|
int32_t tjsonAddObject(SJson* pJson, const char* pName, FToJson func, const void* pObj) {
|
|
if (NULL == pObj) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
SJson* pJobj = tjsonCreateObject();
|
|
if (NULL == pJobj || TSDB_CODE_SUCCESS != func(pObj, pJobj)) {
|
|
tjsonDelete(pJobj);
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
return tjsonAddItemToObject(pJson, pName, pJobj);
|
|
}
|
|
|
|
int32_t tjsonAddItem(SJson* pJson, FToJson func, const void* pObj) {
|
|
SJson* pJobj = tjsonCreateObject();
|
|
if (NULL == pJobj || TSDB_CODE_SUCCESS != func(pObj, pJobj)) {
|
|
tjsonDelete(pJobj);
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
return tjsonAddItemToArray(pJson, pJobj);
|
|
}
|
|
|
|
int32_t tjsonAddArray(SJson* pJson, const char* pName, FToJson func, const void* pArray, int32_t itemSize,
|
|
int32_t num) {
|
|
if (num > 0) {
|
|
SJson* pJsonArray = tjsonAddArrayToObject(pJson, pName);
|
|
if (NULL == pJsonArray) {
|
|
return TSDB_CODE_OUT_OF_MEMORY;
|
|
}
|
|
for (size_t i = 0; i < num; ++i) {
|
|
int32_t code = tjsonAddItem(pJsonArray, func, (const char*)pArray + itemSize * i);
|
|
if (TSDB_CODE_SUCCESS != code) {
|
|
return code;
|
|
}
|
|
}
|
|
}
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonAddTArray(SJson* pJson, const char* pName, FToJson func, const SArray* pArray) {
|
|
int32_t num = taosArrayGetSize(pArray);
|
|
if (num > 0) {
|
|
SJson* pJsonArray = tjsonAddArrayToObject(pJson, pName);
|
|
if (NULL == pJsonArray) {
|
|
return TSDB_CODE_OUT_OF_MEMORY;
|
|
}
|
|
for (int32_t i = 0; i < num; ++i) {
|
|
int32_t code = tjsonAddItem(pJsonArray, func, taosArrayGet(pArray, i));
|
|
if (TSDB_CODE_SUCCESS != code) {
|
|
return code;
|
|
}
|
|
}
|
|
}
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
char* tjsonToString(const SJson* pJson) { return cJSON_Print((cJSON*)pJson); }
|
|
|
|
char* tjsonToUnformattedString(const SJson* pJson) { return cJSON_PrintUnformatted((cJSON*)pJson); }
|
|
|
|
SJson* tjsonGetObjectItem(const SJson* pJson, const char* pName) { return cJSON_GetObjectItem(pJson, pName); }
|
|
|
|
int32_t tjsonGetObjectName(const SJson* pJson, char** pName) {
|
|
*pName = ((cJSON*)pJson)->string;
|
|
if (NULL == *pName) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonGetObjectValueString(const SJson* pJson, char** pValueString) {
|
|
*pValueString = ((cJSON*)pJson)->valuestring;
|
|
if (NULL == *pValueString) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonGetStringValue(const SJson* pJson, const char* pName, char* pVal) {
|
|
char* p = cJSON_GetStringValue(tjsonGetObjectItem((cJSON*)pJson, pName));
|
|
if (NULL == p) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
strcpy(pVal, p);
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonDupStringValue(const SJson* pJson, const char* pName, char** pVal) {
|
|
char* p = cJSON_GetStringValue(tjsonGetObjectItem((cJSON*)pJson, pName));
|
|
if (NULL == p) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
*pVal = taosStrdup(p);
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonGetBigIntValue(const SJson* pJson, const char* pName, int64_t* pVal) {
|
|
char* p = cJSON_GetStringValue(tjsonGetObjectItem((cJSON*)pJson, pName));
|
|
if (NULL == p) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
#ifdef WINDOWS
|
|
sscanf(p, "%" PRId64, pVal);
|
|
#else
|
|
*pVal = taosStr2Int64(p, NULL, 10);
|
|
#endif
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonGetIntValue(const SJson* pJson, const char* pName, int32_t* pVal) {
|
|
int64_t val = 0;
|
|
int32_t code = tjsonGetBigIntValue(pJson, pName, &val);
|
|
*pVal = val;
|
|
return code;
|
|
}
|
|
|
|
int32_t tjsonGetSmallIntValue(const SJson* pJson, const char* pName, int16_t* pVal) {
|
|
int64_t val = 0;
|
|
int32_t code = tjsonGetBigIntValue(pJson, pName, &val);
|
|
*pVal = val;
|
|
return code;
|
|
}
|
|
|
|
int32_t tjsonGetTinyIntValue(const SJson* pJson, const char* pName, int8_t* pVal) {
|
|
int64_t val = 0;
|
|
int32_t code = tjsonGetBigIntValue(pJson, pName, &val);
|
|
*pVal = val;
|
|
return code;
|
|
}
|
|
|
|
int32_t tjsonGetUBigIntValue(const SJson* pJson, const char* pName, uint64_t* pVal) {
|
|
char* p = cJSON_GetStringValue(tjsonGetObjectItem((cJSON*)pJson, pName));
|
|
if (NULL == p) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
#ifdef WINDOWS
|
|
sscanf(p, "%" PRIu64, pVal);
|
|
#else
|
|
*pVal = taosStr2UInt64(p, NULL, 10);
|
|
#endif
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonGetUIntValue(const SJson* pJson, const char* pName, uint32_t* pVal) {
|
|
uint64_t val = 0;
|
|
int32_t code = tjsonGetUBigIntValue(pJson, pName, &val);
|
|
*pVal = val;
|
|
return code;
|
|
}
|
|
|
|
int32_t tjsonGetUTinyIntValue(const SJson* pJson, const char* pName, uint8_t* pVal) {
|
|
uint64_t val = 0;
|
|
int32_t code = tjsonGetUBigIntValue(pJson, pName, &val);
|
|
*pVal = val;
|
|
return code;
|
|
}
|
|
|
|
int32_t tjsonGetBoolValue(const SJson* pJson, const char* pName, bool* pVal) {
|
|
const SJson* pObject = tjsonGetObjectItem(pJson, pName);
|
|
if (NULL == pObject) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
if (!cJSON_IsBool(pObject)) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
*pVal = cJSON_IsTrue(pObject) ? true : false;
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonGetDoubleValue(const SJson* pJson, const char* pName, double* pVal) {
|
|
const SJson* pObject = tjsonGetObjectItem(pJson, pName);
|
|
if (NULL == pObject) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
if (!cJSON_IsNumber(pObject)) {
|
|
return TSDB_CODE_FAILED;
|
|
}
|
|
*pVal = cJSON_GetNumberValue(pObject);
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonGetArraySize(const SJson* pJson) { return cJSON_GetArraySize(pJson); }
|
|
|
|
SJson* tjsonGetArrayItem(const SJson* pJson, int32_t index) { return cJSON_GetArrayItem(pJson, index); }
|
|
|
|
int32_t tjsonToObject(const SJson* pJson, const char* pName, FToObject func, void* pObj) {
|
|
SJson* pJsonObj = tjsonGetObjectItem(pJson, pName);
|
|
if (NULL == pJsonObj) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
return func(pJsonObj, pObj);
|
|
}
|
|
|
|
int32_t tjsonMakeObject(const SJson* pJson, const char* pName, FToObject func, void** pObj, int32_t objSize) {
|
|
if (objSize <= 0) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
SJson* pJsonObj = tjsonGetObjectItem(pJson, pName);
|
|
if (NULL == pJsonObj) {
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
*pObj = taosMemoryCalloc(1, objSize);
|
|
if (NULL == *pObj) {
|
|
return TSDB_CODE_OUT_OF_MEMORY;
|
|
}
|
|
return func(pJsonObj, *pObj);
|
|
}
|
|
|
|
int32_t tjsonToArray(const SJson* pJson, const char* pName, FToObject func, void* pArray, int32_t itemSize) {
|
|
const cJSON* jArray = tjsonGetObjectItem(pJson, pName);
|
|
int32_t size = (NULL == jArray ? 0 : tjsonGetArraySize(jArray));
|
|
for (int32_t i = 0; i < size; ++i) {
|
|
int32_t code = func(tjsonGetArrayItem(jArray, i), (char*)pArray + itemSize * i);
|
|
if (TSDB_CODE_SUCCESS != code) {
|
|
return code;
|
|
}
|
|
}
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
int32_t tjsonToTArray(const SJson* pJson, const char* pName, FToObject func, SArray** pArray, int32_t itemSize) {
|
|
const cJSON* jArray = tjsonGetObjectItem(pJson, pName);
|
|
int32_t size = tjsonGetArraySize(jArray);
|
|
if (size > 0) {
|
|
*pArray = taosArrayInit_s(itemSize, size);
|
|
if (NULL == *pArray) {
|
|
return TSDB_CODE_OUT_OF_MEMORY;
|
|
}
|
|
for (int32_t i = 0; i < size; ++i) {
|
|
int32_t code = func(tjsonGetArrayItem(jArray, i), taosArrayGet(*pArray, i));
|
|
if (TSDB_CODE_SUCCESS != code) {
|
|
return code;
|
|
}
|
|
}
|
|
}
|
|
return TSDB_CODE_SUCCESS;
|
|
}
|
|
|
|
SJson* tjsonParse(const char* pStr) { return cJSON_Parse(pStr); }
|
|
|
|
bool tjsonValidateJson(const char* jIn) {
|
|
if (!jIn) {
|
|
return false;
|
|
}
|
|
|
|
// set json real data
|
|
cJSON* root = cJSON_Parse(jIn);
|
|
if (root == NULL) {
|
|
return false;
|
|
}
|
|
|
|
if (!cJSON_IsObject(root)) {
|
|
return false;
|
|
}
|
|
int size = cJSON_GetArraySize(root);
|
|
for (int i = 0; i < size; i++) {
|
|
cJSON* item = cJSON_GetArrayItem(root, i);
|
|
if (!item) {
|
|
return false;
|
|
}
|
|
|
|
char* jsonKey = item->string;
|
|
if (!jsonKey) return false;
|
|
for (size_t j = 0; j < strlen(jsonKey); ++j) {
|
|
if (isprint(jsonKey[j]) == 0) return false;
|
|
}
|
|
|
|
if (item->type == cJSON_Object || item->type == cJSON_Array) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
const char* tjsonGetError() { return cJSON_GetErrorPtr(); }
|
|
|
|
void tjsonDeleteItemFromObject(const SJson* pJson, const char* pName) {
|
|
cJSON_DeleteItemFromObject((cJSON*)pJson, pName);
|
|
} |