917 lines
26 KiB
C
917 lines
26 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/>.
|
|
*/
|
|
|
|
#include "tvariant.h"
|
|
#include "hash.h"
|
|
#include "hashfunc.h"
|
|
#include "os.h"
|
|
#include "hash.h"
|
|
#include "taos.h"
|
|
#include "taosdef.h"
|
|
#include "tstoken.h"
|
|
#include "ttokendef.h"
|
|
#include "tutil.h"
|
|
|
|
// todo support scientific expression number and oct number
|
|
void tVariantCreate(tVariant *pVar, SSQLToken *token) { tVariantCreateFromString(pVar, token->z, token->n, token->type); }
|
|
|
|
void tVariantCreateFromString(tVariant *pVar, char *pz, uint32_t len, uint32_t type) {
|
|
memset(pVar, 0, sizeof(tVariant));
|
|
|
|
switch (type) {
|
|
case TSDB_DATA_TYPE_BOOL: {
|
|
int32_t k = strncasecmp(pz, "true", 4);
|
|
if (k == 0) {
|
|
pVar->i64Key = TSDB_TRUE;
|
|
} else {
|
|
assert(strncasecmp(pz, "false", 5) == 0);
|
|
pVar->i64Key = TSDB_FALSE;
|
|
}
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_TINYINT:
|
|
case TSDB_DATA_TYPE_SMALLINT:
|
|
case TSDB_DATA_TYPE_BIGINT:
|
|
case TSDB_DATA_TYPE_INT:
|
|
pVar->i64Key = strtoll(pz, NULL, 10);
|
|
break;
|
|
case TSDB_DATA_TYPE_DOUBLE:
|
|
case TSDB_DATA_TYPE_FLOAT:
|
|
pVar->dKey = strtod(pz, NULL);
|
|
break;
|
|
case TSDB_DATA_TYPE_BINARY: {
|
|
pVar->pz = strndup(pz, len);
|
|
pVar->nLen = strdequote(pVar->pz);
|
|
break;
|
|
}
|
|
|
|
default: { // nType == 0 means the null value
|
|
type = TSDB_DATA_TYPE_NULL;
|
|
}
|
|
}
|
|
|
|
pVar->nType = type;
|
|
}
|
|
|
|
/**
|
|
* create tVariant from binary string, not ascii data
|
|
* @param pVar
|
|
* @param pz
|
|
* @param len
|
|
* @param type
|
|
*/
|
|
void tVariantCreateFromBinary(tVariant *pVar, const char *pz, size_t len, uint32_t type) {
|
|
switch (type) {
|
|
case TSDB_DATA_TYPE_BOOL:
|
|
case TSDB_DATA_TYPE_TINYINT: {
|
|
pVar->i64Key = GET_INT8_VAL(pz);
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_SMALLINT: {
|
|
pVar->i64Key = GET_INT16_VAL(pz);
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_INT: {
|
|
pVar->i64Key = GET_INT32_VAL(pz);
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_BIGINT:
|
|
case TSDB_DATA_TYPE_TIMESTAMP: {
|
|
pVar->i64Key = GET_INT64_VAL(pz);
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_DOUBLE: {
|
|
pVar->dKey = GET_DOUBLE_VAL(pz);
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_FLOAT: {
|
|
pVar->dKey = GET_FLOAT_VAL(pz);
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_NCHAR: { // here we get the nchar length from raw binary bits length
|
|
int32_t lenInwchar = len / TSDB_NCHAR_SIZE;
|
|
pVar->wpz = calloc(1, (lenInwchar + 1) * TSDB_NCHAR_SIZE);
|
|
|
|
wcsncpy(pVar->wpz, (wchar_t *)pz, lenInwchar);
|
|
pVar->wpz[lenInwchar] = 0;
|
|
pVar->nLen = len;
|
|
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_BINARY: { // todo refactor, extract a method
|
|
pVar->pz = calloc(len, sizeof(char));
|
|
memcpy(pVar->pz, pz, len);
|
|
pVar->nLen = len;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
pVar->i64Key = GET_INT32_VAL(pVar);
|
|
}
|
|
|
|
pVar->nType = type;
|
|
}
|
|
|
|
void tVariantDestroy(tVariant *pVar) {
|
|
if (pVar == NULL) return;
|
|
|
|
if (pVar->nType == TSDB_DATA_TYPE_BINARY || pVar->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
tfree(pVar->pz);
|
|
pVar->nLen = 0;
|
|
}
|
|
|
|
// NOTE: this is only for string array
|
|
if (pVar->nType == TSDB_DATA_TYPE_ARRAY) {
|
|
size_t num = taosArrayGetSize(pVar->arr);
|
|
for(size_t i = 0; i < num; i++) {
|
|
void* p = taosArrayGetP(pVar->arr, i);
|
|
free(p);
|
|
}
|
|
taosArrayDestroy(pVar->arr);
|
|
pVar->arr = NULL;
|
|
}
|
|
}
|
|
|
|
void tVariantAssign(tVariant *pDst, const tVariant *pSrc) {
|
|
if (pSrc == NULL || pDst == NULL) return;
|
|
|
|
*pDst = *pSrc;
|
|
|
|
if (pSrc->nType == TSDB_DATA_TYPE_BINARY || pSrc->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
int32_t len = pSrc->nLen + 1;
|
|
if (pSrc->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
len = len * TSDB_NCHAR_SIZE;
|
|
}
|
|
|
|
pDst->pz = calloc(1, len);
|
|
memcpy(pDst->pz, pSrc->pz, len);
|
|
return;
|
|
}
|
|
|
|
// this is only for string array
|
|
if (pSrc->nType == TSDB_DATA_TYPE_ARRAY) {
|
|
size_t num = taosArrayGetSize(pSrc->arr);
|
|
pDst->arr = taosArrayInit(num, sizeof(char*));
|
|
for(size_t i = 0; i < num; i++) {
|
|
char* p = (char*)taosArrayGetP(pSrc->arr, i);
|
|
char* n = strdup(p);
|
|
taosArrayPush(pDst->arr, &n);
|
|
}
|
|
}
|
|
}
|
|
|
|
int32_t tVariantToString(tVariant *pVar, char *dst) {
|
|
if (pVar == NULL || dst == NULL) return 0;
|
|
|
|
switch (pVar->nType) {
|
|
case TSDB_DATA_TYPE_BINARY: {
|
|
int32_t len = sprintf(dst, "\'%s\'", pVar->pz);
|
|
assert(len <= pVar->nLen + sizeof("\'") * 2); // two more chars
|
|
return len;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_NCHAR: {
|
|
dst[0] = '\'';
|
|
taosUcs4ToMbs(pVar->wpz, (twcslen(pVar->wpz) + 1) * TSDB_NCHAR_SIZE, dst + 1);
|
|
int32_t len = strlen(dst);
|
|
dst[len] = '\'';
|
|
dst[len + 1] = 0;
|
|
return len + 1;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_BOOL:
|
|
case TSDB_DATA_TYPE_TINYINT:
|
|
case TSDB_DATA_TYPE_SMALLINT:
|
|
case TSDB_DATA_TYPE_INT:
|
|
return sprintf(dst, "%d", (int32_t)pVar->i64Key);
|
|
|
|
case TSDB_DATA_TYPE_BIGINT:
|
|
return sprintf(dst, "%" PRId64, pVar->i64Key);
|
|
|
|
case TSDB_DATA_TYPE_FLOAT:
|
|
case TSDB_DATA_TYPE_DOUBLE:
|
|
return sprintf(dst, "%.9lf", pVar->dKey);
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
static int32_t doConvertToInteger(tVariant *pVariant, char *pDest, int32_t type, bool releaseVariantPtr) {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
setNull(pDest, type, tDataTypeDesc[type].nSize);
|
|
return 0;
|
|
}
|
|
|
|
if (pVariant->nType >= TSDB_DATA_TYPE_BOOL && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
*((int64_t *)pDest) = pVariant->i64Key;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
|
|
if ((pVariant->dKey < INT64_MIN) || (pVariant->dKey > INT64_MAX)) {
|
|
return -1;
|
|
}
|
|
|
|
*((int64_t *)pDest) = (int64_t)pVariant->dKey;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
errno = 0;
|
|
char *endPtr = NULL;
|
|
|
|
SSQLToken token = {0};
|
|
token.n = tSQLGetToken(pVariant->pz, &token.type);
|
|
|
|
if (token.type == TK_MINUS || token.type == TK_PLUS) {
|
|
token.n = tSQLGetToken(pVariant->pz + token.n, &token.type);
|
|
}
|
|
|
|
if (token.type == TK_FLOAT) {
|
|
double v = strtod(pVariant->pz, &endPtr);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if ((errno == ERANGE && v == -1) || (isinf(v) || isnan(v))) {
|
|
return -1;
|
|
}
|
|
|
|
if ((v < INT64_MIN) || (v > INT64_MAX)) {
|
|
return -1;
|
|
}
|
|
|
|
*((int64_t *)pDest) = (int64_t)v;
|
|
} else if (token.type == TK_INTEGER) {
|
|
int64_t val = strtoll(pVariant->pz, &endPtr, 10);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if (errno == ERANGE) {
|
|
return -1; // data overflow
|
|
}
|
|
|
|
*((int64_t *)pDest) = val;
|
|
} else if (token.type == TK_NULL) {
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
setNull(pDest, type, tDataTypeDesc[type].nSize);
|
|
} else {
|
|
return -1;
|
|
}
|
|
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
errno = 0;
|
|
wchar_t *endPtr = NULL;
|
|
|
|
SSQLToken token = {0};
|
|
token.n = tSQLGetToken(pVariant->pz, &token.type);
|
|
|
|
if (token.type == TK_MINUS || token.type == TK_PLUS) {
|
|
token.n = tSQLGetToken(pVariant->pz + token.n, &token.type);
|
|
}
|
|
|
|
if (token.type == TK_FLOAT) {
|
|
double v = wcstod(pVariant->wpz, &endPtr);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if ((errno == ERANGE && v == -1) || (isinf(v) || isnan(v))) {
|
|
return -1;
|
|
}
|
|
|
|
if ((v < INT64_MIN) || (v > INT64_MAX)) {
|
|
return -1;
|
|
}
|
|
|
|
*((int64_t *)pDest) = (int64_t)v;
|
|
} else if (token.type == TK_NULL) {
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
setNull(pDest, type, tDataTypeDesc[type].nSize);
|
|
} else {
|
|
int64_t val = wcstoll(pVariant->wpz, &endPtr, 10);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if (errno == ERANGE) {
|
|
return -1; // data overflow
|
|
}
|
|
|
|
*((int64_t *)pDest) = val;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
static FORCE_INLINE int32_t convertToBoolImpl(char *pStr, int32_t len) {
|
|
if ((strncasecmp(pStr, "true", len) == 0) && (len == 4)) {
|
|
return TSDB_TRUE;
|
|
} else if ((strncasecmp(pStr, "false", len) == 0) && (len == 5)) {
|
|
return TSDB_FALSE;
|
|
} else if (strcasecmp(pStr, TSDB_DATA_NULL_STR_L) == 0) {
|
|
return TSDB_DATA_BOOL_NULL;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static FORCE_INLINE int32_t wcsconvertToBoolImpl(wchar_t *pstr, int32_t len) {
|
|
if ((wcsncasecmp(pstr, L"true", len) == 0) && (len == 4)) {
|
|
return TSDB_TRUE;
|
|
} else if (wcsncasecmp(pstr, L"false", len) == 0 && (len == 5)) {
|
|
return TSDB_FALSE;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static int32_t toBinary(tVariant *pVariant, char **pDest, int32_t *pDestSize) {
|
|
const int32_t INITIAL_ALLOC_SIZE = 20;
|
|
char * pBuf = NULL;
|
|
|
|
if (*pDest == pVariant->pz) {
|
|
pBuf = calloc(1, INITIAL_ALLOC_SIZE);
|
|
}
|
|
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
size_t newSize = pVariant->nLen * TSDB_NCHAR_SIZE;
|
|
if (pBuf != NULL) {
|
|
if (newSize >= INITIAL_ALLOC_SIZE) {
|
|
pBuf = realloc(pBuf, newSize + 1);
|
|
}
|
|
|
|
taosUcs4ToMbs(pVariant->wpz, newSize, pBuf);
|
|
free(pVariant->wpz);
|
|
pBuf[newSize] = 0;
|
|
} else {
|
|
taosUcs4ToMbs(pVariant->wpz, newSize, *pDest);
|
|
}
|
|
|
|
} else {
|
|
if (pVariant->nType >= TSDB_DATA_TYPE_TINYINT && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
sprintf(pBuf == NULL ? *pDest : pBuf, "%" PRId64, pVariant->i64Key);
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
|
|
sprintf(pBuf == NULL ? *pDest : pBuf, "%lf", pVariant->dKey);
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_BOOL) {
|
|
sprintf(pBuf == NULL ? *pDest : pBuf, "%s", (pVariant->i64Key == TSDB_TRUE) ? "TRUE" : "FALSE");
|
|
} else if (pVariant->nType == 0) { // null data
|
|
setNull(pBuf == NULL ? *pDest : pBuf, TSDB_DATA_TYPE_BINARY, 0);
|
|
}
|
|
}
|
|
|
|
if (pBuf != NULL) {
|
|
*pDest = pBuf;
|
|
}
|
|
|
|
*pDestSize = strlen(*pDest);
|
|
return 0;
|
|
}
|
|
|
|
// todo handle the error
|
|
static int32_t toNchar(tVariant *pVariant, char **pDest, int32_t *pDestSize) {
|
|
char tmpBuf[40] = {0};
|
|
|
|
char * pDst = tmpBuf;
|
|
int32_t nLen = 0;
|
|
|
|
if (pVariant->nType >= TSDB_DATA_TYPE_TINYINT && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
nLen = sprintf(pDst, "%" PRId64, pVariant->i64Key);
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
|
|
nLen = sprintf(pDst, "%lf", pVariant->dKey);
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
pDst = pVariant->pz;
|
|
nLen = pVariant->nLen;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_BOOL) {
|
|
nLen = sprintf(pDst, "%s", (pVariant->i64Key == TSDB_TRUE) ? "TRUE" : "FALSE");
|
|
}
|
|
|
|
if (*pDest == pVariant->pz) {
|
|
wchar_t *pWStr = calloc(1, (nLen + 1) * TSDB_NCHAR_SIZE);
|
|
taosMbsToUcs4(pDst, nLen, (char *)pWStr, (nLen + 1) * TSDB_NCHAR_SIZE, NULL);
|
|
|
|
// free the binary buffer in the first place
|
|
if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
free(pVariant->wpz);
|
|
}
|
|
|
|
pVariant->wpz = pWStr;
|
|
*pDestSize = twcslen(pVariant->wpz);
|
|
|
|
// shrink the allocate memory, no need to check here.
|
|
char* tmp = realloc(pVariant->wpz, (*pDestSize + 1)*TSDB_NCHAR_SIZE);
|
|
assert(tmp != NULL);
|
|
|
|
pVariant->wpz = (wchar_t *)tmp;
|
|
} else {
|
|
size_t output = -1;
|
|
taosMbsToUcs4(pDst, nLen, *pDest, (nLen + 1) * TSDB_NCHAR_SIZE, &output);
|
|
|
|
if (pDestSize != NULL) {
|
|
*pDestSize = output;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static FORCE_INLINE int32_t convertToDouble(char *pStr, int32_t len, double *value) {
|
|
SSQLToken stoken = {.z = pStr, .n = len};
|
|
|
|
if (TK_ILLEGAL == isValidNumber(&stoken)) {
|
|
return -1;
|
|
}
|
|
|
|
*value = strtod(pStr, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static FORCE_INLINE int32_t convertToInteger(tVariant *pVariant, int64_t *result, int32_t type, int64_t lowBnd,
|
|
int64_t upperBnd, bool releaseVariantPtr) {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
setNull((char *)result, type, tDataTypeDesc[type].nSize);
|
|
return 0;
|
|
}
|
|
|
|
if (pVariant->nType >= TSDB_DATA_TYPE_BOOL && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
*result = pVariant->i64Key;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
|
|
*result = (int64_t)pVariant->dKey;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
errno = 0;
|
|
char *endPtr = NULL;
|
|
|
|
SSQLToken token = {0};
|
|
token.n = tSQLGetToken(pVariant->pz, &token.type);
|
|
|
|
if (token.type == TK_MINUS || token.type == TK_PLUS) {
|
|
token.n = tSQLGetToken(pVariant->pz + token.n, &token.type);
|
|
}
|
|
|
|
if (token.type == TK_NULL) {
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
setNull((char *)result, type, tDataTypeDesc[type].nSize);
|
|
return 0;
|
|
}
|
|
|
|
SSQLToken sToken = {.z = pVariant->pz, .n = pVariant->nLen};
|
|
if (TK_ILLEGAL == isValidNumber(&sToken)) {
|
|
return -1;
|
|
}
|
|
|
|
if (token.type == TK_FLOAT) {
|
|
double v = strtod(pVariant->pz, &endPtr);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if ((errno == ERANGE && v == -1) || (isinf(v) || isnan(v))) {
|
|
return -1;
|
|
}
|
|
|
|
*result = (int64_t)v;
|
|
} else if (token.type == TK_INTEGER) {
|
|
int64_t val = strtoll(pVariant->pz, &endPtr, 10);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if (errno == ERANGE) {
|
|
return -1; // data overflow
|
|
}
|
|
|
|
*result = val;
|
|
} else {
|
|
return -1;
|
|
}
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
errno = 0;
|
|
wchar_t *endPtr = NULL;
|
|
|
|
SSQLToken token = {0};
|
|
token.n = tSQLGetToken(pVariant->pz, &token.type);
|
|
|
|
if (token.type == TK_MINUS || token.type == TK_PLUS) {
|
|
token.n = tSQLGetToken(pVariant->pz + token.n, &token.type);
|
|
}
|
|
|
|
if (token.type == TK_FLOAT) {
|
|
double v = wcstod(pVariant->wpz, &endPtr);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if ((errno == ERANGE && v == -1) || (isinf(v) || isnan(v))) {
|
|
return -1;
|
|
}
|
|
|
|
*result = (int64_t)v;
|
|
} else if (token.type == TK_NULL) {
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
setNull((char *)result, type, tDataTypeDesc[type].nSize);
|
|
return 0;
|
|
} else {
|
|
int64_t val = wcstoll(pVariant->wpz, &endPtr, 10);
|
|
if (releaseVariantPtr) {
|
|
free(pVariant->pz);
|
|
pVariant->nLen = 0;
|
|
}
|
|
|
|
if (errno == ERANGE) {
|
|
return -1; // data overflow
|
|
}
|
|
|
|
*result = val;
|
|
}
|
|
}
|
|
|
|
if ((*result <= lowBnd) || (*result > upperBnd)) {
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int32_t convertToBool(tVariant *pVariant, int64_t *pDest) {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_BOOL) {
|
|
*pDest = pVariant->i64Key; // in order to be compatible to null of bool
|
|
} else if (pVariant->nType >= TSDB_DATA_TYPE_TINYINT && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
*pDest = ((pVariant->i64Key != 0) ? TSDB_TRUE : TSDB_FALSE);
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_FLOAT || pVariant->nType == TSDB_DATA_TYPE_DOUBLE) {
|
|
*pDest = ((pVariant->dKey != 0) ? TSDB_TRUE : TSDB_FALSE);
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
int32_t ret = 0;
|
|
if ((ret = convertToBoolImpl(pVariant->pz, pVariant->nLen)) < 0) {
|
|
return ret;
|
|
}
|
|
|
|
*pDest = ret;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
int32_t ret = 0;
|
|
if ((ret = wcsconvertToBoolImpl(pVariant->wpz, pVariant->nLen)) < 0) {
|
|
return ret;
|
|
}
|
|
*pDest = ret;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
*pDest = TSDB_DATA_BOOL_NULL;
|
|
}
|
|
|
|
assert(*pDest == TSDB_TRUE || *pDest == TSDB_FALSE || *pDest == TSDB_DATA_BOOL_NULL);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* transfer data from variant serve as the implicit data conversion: from input sql string pVariant->nType
|
|
* to column type defined in schema
|
|
*
|
|
* todo handle the return value
|
|
*/
|
|
int32_t tVariantDump(tVariant *pVariant, char *payload, int16_t type, bool includeLengthPrefix) {
|
|
if (pVariant == NULL || (pVariant->nType != 0 && !isValidDataType(pVariant->nType))) {
|
|
return -1;
|
|
}
|
|
|
|
errno = 0; // reset global error code
|
|
|
|
switch (type) {
|
|
case TSDB_DATA_TYPE_BOOL: {
|
|
int64_t dst = 0;
|
|
if (convertToBool(pVariant, &dst) < 0) {
|
|
return -1;
|
|
}
|
|
*(int8_t *)payload = (int8_t)dst;
|
|
break;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_TINYINT: {
|
|
int64_t result = 0;
|
|
if (convertToInteger(pVariant, &result, type, INT8_MIN, INT8_MAX, false) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
*((int8_t *)payload) = (int8_t)result;
|
|
break;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_SMALLINT: {
|
|
int64_t result = 0;
|
|
if (convertToInteger(pVariant, &result, type, INT16_MIN, INT16_MAX, false) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
*((int16_t *)payload) = (int16_t)result;
|
|
break;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_INT: {
|
|
int64_t result = 0;
|
|
if (convertToInteger(pVariant, &result, type, INT32_MIN, INT32_MAX, false) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
*((int32_t *)payload) = (int32_t)result;
|
|
break;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_BIGINT: {
|
|
int64_t result = 0;
|
|
if (convertToInteger(pVariant, &result, type, INT64_MIN, INT64_MAX, false) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
*((int64_t *)payload) = (int64_t)result;
|
|
break;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_FLOAT: {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
if (strncasecmp(TSDB_DATA_NULL_STR_L, pVariant->pz, pVariant->nLen) == 0 &&
|
|
strlen(TSDB_DATA_NULL_STR_L) == pVariant->nLen) {
|
|
*((int32_t *)payload) = TSDB_DATA_FLOAT_NULL;
|
|
return 0;
|
|
} else {
|
|
double value;
|
|
int32_t ret;
|
|
ret = convertToDouble(pVariant->pz, pVariant->nLen, &value);
|
|
if ((errno == ERANGE && (float)value == -1) || (ret != 0)) {
|
|
return -1;
|
|
}
|
|
|
|
#ifdef _TD_ARM_32_
|
|
//memcpy(&payload, &value, sizeof(float));
|
|
float fv = (float)value;
|
|
SET_FLOAT_VAL_ALIGN(payload, &fv);
|
|
#else
|
|
*((float *)payload) = (float)value;
|
|
#endif
|
|
}
|
|
} else if (pVariant->nType >= TSDB_DATA_TYPE_BOOL && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
#ifdef _TD_ARM_32_
|
|
//memcpy(&payload, &pVariant->i64Key, sizeof(float));
|
|
float fv = (float)pVariant->i64Key;
|
|
SET_FLOAT_VAL_ALIGN(payload, &fv);
|
|
#else
|
|
*((float *)payload) = pVariant->i64Key;
|
|
#endif
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
|
|
#ifdef _TD_ARM_32_
|
|
//memcpy(&payload, &pVariant->dKey, sizeof(float));
|
|
float fv = (float)pVariant->dKey;
|
|
SET_FLOAT_VAL_ALIGN(payload, &fv);
|
|
#else
|
|
*((float *)payload) = (float)pVariant->dKey;
|
|
#endif
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
*((int32_t *)payload) = TSDB_DATA_FLOAT_NULL;
|
|
return 0;
|
|
}
|
|
|
|
#ifdef _TD_ARM_32_
|
|
float fv = GET_FLOAT_VAL(payload);
|
|
if (isinf(fv) || isnan(fv) || fv > FLT_MAX || fv < -FLT_MAX) {
|
|
return -1;
|
|
}
|
|
#else
|
|
if (isinf(*((float *)payload)) || isnan(*((float *)payload)) || *((float *)payload) > FLT_MAX ||
|
|
*((float *)payload) < -FLT_MAX) {
|
|
return -1;
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_DOUBLE: {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
if (strncasecmp(TSDB_DATA_NULL_STR_L, pVariant->pz, pVariant->nLen) == 0 &&
|
|
strlen(TSDB_DATA_NULL_STR_L) == pVariant->nLen) {
|
|
*((int64_t *)payload) = TSDB_DATA_DOUBLE_NULL;
|
|
return 0;
|
|
} else {
|
|
double value = 0;
|
|
int32_t ret;
|
|
ret = convertToDouble(pVariant->pz, pVariant->nLen, &value);
|
|
if ((errno == ERANGE && value == -1) || (ret != 0)) {
|
|
return -1;
|
|
}
|
|
|
|
#ifdef _TD_ARM_32_
|
|
SET_DOUBLE_VAL_ALIGN(payload, &value);
|
|
#else
|
|
*((double *)payload) = value;
|
|
#endif
|
|
}
|
|
} else if (pVariant->nType >= TSDB_DATA_TYPE_BOOL && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
#ifdef _TD_ARM_32_
|
|
double dv = (double)(pVariant->i64Key);
|
|
SET_DOUBLE_VAL_ALIGN(payload, &dv);
|
|
#else
|
|
*((double *)payload) = pVariant->i64Key;
|
|
#endif
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
|
|
#ifdef _TD_ARM_32_
|
|
double dv = (double)(pVariant->dKey);
|
|
SET_DOUBLE_VAL_ALIGN(payload, &dv);
|
|
#else
|
|
*((double *)payload) = pVariant->dKey;
|
|
#endif
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
*((int64_t *)payload) = TSDB_DATA_DOUBLE_NULL;
|
|
return 0;
|
|
}
|
|
|
|
#ifdef _TD_ARM_32_
|
|
double dv = GET_DOUBLE_VAL(payload);
|
|
if (isinf(dv) || isnan(dv) || dv > DBL_MAX || dv < -DBL_MAX) {
|
|
return -1;
|
|
}
|
|
#else
|
|
if (isinf(*((double *)payload)) || isnan(*((double *)payload)) || *((double *)payload) > DBL_MAX ||
|
|
*((double *)payload) < -DBL_MAX) {
|
|
return -1;
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
case TSDB_DATA_TYPE_BINARY: {
|
|
if (!includeLengthPrefix) {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
*(uint8_t*) payload = TSDB_DATA_BINARY_NULL;
|
|
} else {
|
|
if (pVariant->nType != TSDB_DATA_TYPE_BINARY) {
|
|
toBinary(pVariant, &payload, &pVariant->nLen);
|
|
} else {
|
|
strncpy(payload, pVariant->pz, pVariant->nLen);
|
|
}
|
|
}
|
|
} else {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
setVardataNull(payload, TSDB_DATA_TYPE_BINARY);
|
|
} else {
|
|
char *p = varDataVal(payload);
|
|
|
|
if (pVariant->nType != TSDB_DATA_TYPE_BINARY) {
|
|
toBinary(pVariant, &p, &pVariant->nLen);
|
|
} else {
|
|
strncpy(p, pVariant->pz, pVariant->nLen);
|
|
}
|
|
|
|
varDataSetLen(payload, pVariant->nLen);
|
|
assert(p == varDataVal(payload));
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_TIMESTAMP: {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
*((int64_t *)payload) = TSDB_DATA_BIGINT_NULL;
|
|
} else {
|
|
*((int64_t *)payload) = pVariant->i64Key;
|
|
}
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_NCHAR: {
|
|
int32_t newlen = 0;
|
|
if (!includeLengthPrefix) {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
*(uint32_t *)payload = TSDB_DATA_NCHAR_NULL;
|
|
} else {
|
|
if (pVariant->nType != TSDB_DATA_TYPE_NCHAR) {
|
|
toNchar(pVariant, &payload, &newlen);
|
|
} else {
|
|
wcsncpy((wchar_t *)payload, pVariant->wpz, pVariant->nLen);
|
|
}
|
|
}
|
|
} else {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
|
|
setVardataNull(payload, TSDB_DATA_TYPE_NCHAR);
|
|
} else {
|
|
char *p = varDataVal(payload);
|
|
|
|
if (pVariant->nType != TSDB_DATA_TYPE_NCHAR) {
|
|
toNchar(pVariant, &p, &newlen);
|
|
} else {
|
|
wcsncpy((wchar_t *)p, pVariant->wpz, pVariant->nLen);
|
|
newlen = pVariant->nLen;
|
|
}
|
|
|
|
varDataSetLen(payload, newlen); // the length may be changed after toNchar function called
|
|
assert(p == varDataVal(payload));
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* In variant, bool/smallint/tinyint/int/bigint share the same attribution of
|
|
* structure, also ignore the convert the type required
|
|
*
|
|
* It is actually the bigint/binary/bool/nchar type transfer
|
|
*/
|
|
int32_t tVariantTypeSetType(tVariant *pVariant, char type) {
|
|
if (pVariant == NULL || pVariant->nType == 0) { // value is not set
|
|
return 0;
|
|
}
|
|
|
|
switch (type) {
|
|
case TSDB_DATA_TYPE_BOOL: { // bool
|
|
if (convertToBool(pVariant, &pVariant->i64Key) < 0) {
|
|
return -1;
|
|
}
|
|
|
|
pVariant->nType = type;
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_INT:
|
|
case TSDB_DATA_TYPE_BIGINT:
|
|
case TSDB_DATA_TYPE_TINYINT:
|
|
case TSDB_DATA_TYPE_SMALLINT: {
|
|
convertToInteger(pVariant, &(pVariant->i64Key), type, INT64_MIN, INT64_MAX, true);
|
|
pVariant->nType = TSDB_DATA_TYPE_BIGINT;
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_FLOAT:
|
|
case TSDB_DATA_TYPE_DOUBLE: {
|
|
if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
|
|
errno = 0;
|
|
double v = strtod(pVariant->pz, NULL);
|
|
if ((errno == ERANGE && v == -1) || (isinf(v) || isnan(v))) {
|
|
free(pVariant->pz);
|
|
return -1;
|
|
}
|
|
|
|
free(pVariant->pz);
|
|
pVariant->dKey = v;
|
|
} else if (pVariant->nType == TSDB_DATA_TYPE_NCHAR) {
|
|
errno = 0;
|
|
double v = wcstod(pVariant->wpz, NULL);
|
|
if ((errno == ERANGE && v == -1) || (isinf(v) || isnan(v))) {
|
|
free(pVariant->pz);
|
|
return -1;
|
|
}
|
|
|
|
free(pVariant->pz);
|
|
pVariant->dKey = v;
|
|
} else if (pVariant->nType >= TSDB_DATA_TYPE_BOOL && pVariant->nType <= TSDB_DATA_TYPE_BIGINT) {
|
|
pVariant->dKey = (double)(pVariant->i64Key);
|
|
}
|
|
|
|
pVariant->nType = TSDB_DATA_TYPE_DOUBLE;
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_BINARY: {
|
|
if (pVariant->nType != TSDB_DATA_TYPE_BINARY) {
|
|
toBinary(pVariant, &pVariant->pz, &pVariant->nLen);
|
|
}
|
|
pVariant->nType = type;
|
|
break;
|
|
}
|
|
case TSDB_DATA_TYPE_NCHAR: {
|
|
if (pVariant->nType != TSDB_DATA_TYPE_NCHAR) {
|
|
toNchar(pVariant, &pVariant->pz, &pVariant->nLen);
|
|
}
|
|
pVariant->nType = type;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|