tvariant.c 28.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
/*
 * 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/>.
 */
15
#include "os.h"
16 17

#include "hash.h"
H
hjxilinx 已提交
18 19
#include "taos.h"
#include "taosdef.h"
20
#include "ttoken.h"
21
#include "ttokendef.h"
22
#include "ttype.h"
23 24
#include "tutil.h"
#include "tvariant.h"
25

W
wpan 已提交
26 27 28 29 30 31 32
#define SET_EXT_INFO(converted, res, minv, maxv, exti) do {                                                       \
                                                        if (converted == NULL || exti == NULL || *converted == false) { break; }  \
                                                        if ((res) < (minv)) { *exti = -1; break; }                                  \
                                                        if ((res) > (maxv)) { *exti = 1; break; }                                   \
                                                        assert(0);                                                                  \
                                                       } while (0)

33 34 35
void tVariantCreate(tVariant *pVar, SStrToken *token) {
  int32_t ret = 0;
  int32_t type = token->type;
36 37

  memset(pVar, 0, sizeof(tVariant));
38 39

  switch (token->type) {
40
    case TSDB_DATA_TYPE_BOOL: {
41
      if (strncasecmp(token->z, "true", 4) == 0) {
42
        pVar->i64 = TSDB_TRUE;
43
      } else if (strncasecmp(token->z, "false", 5) == 0) {
44
        pVar->i64 = TSDB_FALSE;
45 46
      } else {
        return;
47
      }
48

49 50
      break;
    }
51

52 53 54
    case TSDB_DATA_TYPE_TINYINT:
    case TSDB_DATA_TYPE_SMALLINT:
    case TSDB_DATA_TYPE_BIGINT:
55 56 57
    case TSDB_DATA_TYPE_INT:{
      ret = tStrToInteger(token->z, token->type, token->n, &pVar->i64, true);
      if (ret != 0) {
58
        SStrToken t = {0};
59
        tGetToken(token->z, &t.type);
60 61 62 63 64
        if (t.type == TK_MINUS) {  // it is a signed number which is greater than INT64_MAX or less than INT64_MIN
          pVar->nType = -1;   // -1 means error type
          return;
        }

65 66 67 68 69 70
        // data overflow, try unsigned parse the input number
        ret = tStrToInteger(token->z, token->type, token->n, &pVar->i64, false);
        if (ret != 0) {
          pVar->nType = -1;   // -1 means error type
          return;
        }
71 72
      }

73
      break;
74 75
    }

76
    case TSDB_DATA_TYPE_DOUBLE:
77 78
    case TSDB_DATA_TYPE_FLOAT: {
      pVar->dKey = strtod(token->z, NULL);
79
      break;
80 81
    }

82
    case TSDB_DATA_TYPE_BINARY: {
83
      pVar->pz = strndup(token->z, token->n);
D
dapan1121 已提交
84
      pVar->nLen = strRmquote(pVar->pz, token->n);
85 86
      break;
    }
87 88 89 90
    case TSDB_DATA_TYPE_TIMESTAMP: {
      pVar->i64 = taosGetTimestamp(TSDB_TIME_PRECISION_NANO);                           
      break;                             
    }                            
91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106
    
    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
 */
107
void tVariantCreateFromBinary(tVariant *pVar, const char *pz, size_t len, uint32_t type) {
108 109 110
  switch (type) {
    case TSDB_DATA_TYPE_BOOL:
    case TSDB_DATA_TYPE_TINYINT: {
111
      pVar->nLen = tDataTypes[type].bytes;
112 113 114 115
      pVar->i64 = GET_INT8_VAL(pz);
      break;
    }
    case TSDB_DATA_TYPE_UTINYINT: {
116
      pVar->nLen = tDataTypes[type].bytes;
117
      pVar->u64 = GET_UINT8_VAL(pz);
118 119 120
      break;
    }
    case TSDB_DATA_TYPE_SMALLINT: {
121
      pVar->nLen = tDataTypes[type].bytes;
122 123 124 125
      pVar->i64 = GET_INT16_VAL(pz);
      break;
    }
    case TSDB_DATA_TYPE_USMALLINT: {
126
      pVar->nLen = tDataTypes[type].bytes;
127
      pVar->u64 = GET_UINT16_VAL(pz);
128 129 130
      break;
    }
    case TSDB_DATA_TYPE_INT: {
131
      pVar->nLen = tDataTypes[type].bytes;
132 133 134 135
      pVar->i64 = GET_INT32_VAL(pz);
      break;
    }
    case TSDB_DATA_TYPE_UINT: {
136
      pVar->nLen = tDataTypes[type].bytes;
137
      pVar->u64 = GET_UINT32_VAL(pz);
138 139 140 141
      break;
    }
    case TSDB_DATA_TYPE_BIGINT:
    case TSDB_DATA_TYPE_TIMESTAMP: {
142
      pVar->nLen = tDataTypes[type].bytes;
143 144 145 146
      pVar->i64 = GET_INT64_VAL(pz);
      break;
    }
    case TSDB_DATA_TYPE_UBIGINT: {
147
      pVar->nLen = tDataTypes[type].bytes;
148
      pVar->u64 = GET_UINT64_VAL(pz);
149 150 151
      break;
    }
    case TSDB_DATA_TYPE_DOUBLE: {
152
      pVar->nLen = tDataTypes[type].bytes;
153 154 155 156
      pVar->dKey = GET_DOUBLE_VAL(pz);
      break;
    }
    case TSDB_DATA_TYPE_FLOAT: {
157
      pVar->nLen = tDataTypes[type].bytes;
158 159 160 161
      pVar->dKey = GET_FLOAT_VAL(pz);
      break;
    }
    case TSDB_DATA_TYPE_NCHAR: { // here we get the nchar length from raw binary bits length
162
      size_t lenInwchar = len / TSDB_NCHAR_SIZE;
163

164
      pVar->wpz = calloc(1, (lenInwchar + 1) * TSDB_NCHAR_SIZE);
165
      memcpy(pVar->wpz, pz, lenInwchar * TSDB_NCHAR_SIZE);
166
      pVar->nLen = (int32_t)len;
167 168 169
      
      break;
    }
170
    case TSDB_DATA_TYPE_BINARY: {  // todo refactor, extract a method
H
Haojun Liao 已提交
171
      pVar->pz = calloc(len + 1, sizeof(char));
172
      memcpy(pVar->pz, pz, len);
173
      pVar->nLen = (int32_t)len;
174 175 176 177
      break;
    }
    
    default:
178
      pVar->i64 = GET_INT32_VAL(pz);
179
      pVar->nLen = tDataTypes[TSDB_DATA_TYPE_INT].bytes;
180 181 182 183 184 185 186 187 188
  }
  
  pVar->nType = type;
}

void tVariantDestroy(tVariant *pVar) {
  if (pVar == NULL) return;
  
  if (pVar->nType == TSDB_DATA_TYPE_BINARY || pVar->nType == TSDB_DATA_TYPE_NCHAR) {
S
TD-1848  
Shengliang Guan 已提交
189
    tfree(pVar->pz);
190 191
    pVar->nLen = 0;
  }
weixin_48148422's avatar
weixin_48148422 已提交
192 193

  // NOTE: this is only for string array
W
wpan 已提交
194
  if (pVar->nType == TSDB_DATA_TYPE_POINTER_ARRAY) {
weixin_48148422's avatar
weixin_48148422 已提交
195 196 197 198 199 200 201
    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;
W
wpan 已提交
202 203 204
  } else if (pVar->nType == TSDB_DATA_TYPE_VALUE_ARRAY) {
    taosArrayDestroy(pVar->arr);
    pVar->arr = NULL;
weixin_48148422's avatar
weixin_48148422 已提交
205
  }
206 207
}

208 209 210 211 212
bool tVariantIsValid(tVariant *pVar) {
  assert(pVar != NULL);
  return isValidDataType(pVar->nType);
}

213 214 215
void tVariantAssign(tVariant *pDst, const tVariant *pSrc) {
  if (pSrc == NULL || pDst == NULL) return;
  
H
Haojun Liao 已提交
216
  pDst->nType = pSrc->nType;
H
Haojun Liao 已提交
217
  if (pSrc->nType == TSDB_DATA_TYPE_BINARY || pSrc->nType == TSDB_DATA_TYPE_NCHAR) {
H
Haojun Liao 已提交
218 219
    int32_t len = pSrc->nLen + TSDB_NCHAR_SIZE;
    char* p = realloc(pDst->pz, len);
H
Haojun Liao 已提交
220 221
    assert(p);

H
Haojun Liao 已提交
222
    memset(p, 0, len);
H
Haojun Liao 已提交
223 224
    pDst->pz = p;

H
Haojun Liao 已提交
225 226
    memcpy(pDst->pz, pSrc->pz, pSrc->nLen);
    pDst->nLen = pSrc->nLen;
H
Haojun Liao 已提交
227 228 229 230
    return;

  }

231
  if (IS_NUMERIC_TYPE(pSrc->nType) || (pSrc->nType == TSDB_DATA_TYPE_BOOL)) {
232
    pDst->i64 = pSrc->i64;
W
wpan 已提交
233
  } else if (pSrc->nType == TSDB_DATA_TYPE_POINTER_ARRAY) {  // this is only for string array
weixin_48148422's avatar
weixin_48148422 已提交
234 235 236 237 238 239 240
    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);
    }
W
wpan 已提交
241 242 243 244 245 246 247 248 249
  } else if (pSrc->nType == TSDB_DATA_TYPE_VALUE_ARRAY) {
      size_t num = taosArrayGetSize(pSrc->arr);
      pDst->arr = taosArrayInit(num, sizeof(int64_t));
      pDst->nLen = pSrc->nLen;
      assert(pSrc->nLen == num);
      for(size_t i = 0; i < num; i++) {
        int64_t *p = taosArrayGet(pSrc->arr, i);
        taosArrayPush(pDst->arr, p);
      }
250
  }
H
Haojun Liao 已提交
251

W
wpan 已提交
252
  if (pDst->nType != TSDB_DATA_TYPE_POINTER_ARRAY && pDst->nType != TSDB_DATA_TYPE_VALUE_ARRAY) {
H
Haojun Liao 已提交
253
    pDst->nLen = tDataTypes[pDst->nType].bytes;
H
Haojun Liao 已提交
254
  }
255 256
}

257
int32_t tVariantCompare(const tVariant* p1, const tVariant* p2) {
H
Haojun Liao 已提交
258 259 260 261 262 263 264 265 266 267 268
  if (p1->nType == TSDB_DATA_TYPE_NULL && p2->nType == TSDB_DATA_TYPE_NULL) {
    return 0;
  }

  if (p1->nType == TSDB_DATA_TYPE_NULL) {
    return -1;
  }

  if (p2->nType == TSDB_DATA_TYPE_NULL) {
    return 1;
  }
269

270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293
  if (p1->nType == TSDB_DATA_TYPE_BINARY || p1->nType == TSDB_DATA_TYPE_NCHAR) {
    if (p1->nLen == p2->nLen) {
      return memcmp(p1->pz, p2->pz, p1->nLen);
    } else {
      return p1->nLen > p2->nLen? 1:-1;
    }
  } else if (p1->nType == TSDB_DATA_TYPE_FLOAT || p1->nType == TSDB_DATA_TYPE_DOUBLE) {
    if (p1->dKey == p2->dKey) {
      return 0;
    } else {
      return p1->dKey > p2->dKey? 1:-1;
    }
  } else if (IS_UNSIGNED_NUMERIC_TYPE(p1->nType)) {
    if (p1->u64 == p2->u64) {
      return 0;
    } else {
      return p1->u64 > p2->u64? 1:-1;
    }
  } else {
    if (p1->i64 == p2->i64) {
      return 0;
    } else {
      return p1->i64 > p2->i64? 1:-1;
    }
294 295 296
  }
}

297 298 299 300 301 302 303 304 305 306 307 308
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] = '\'';
H
[TD-92]  
Hui Li 已提交
309
      taosUcs4ToMbs(pVar->wpz, (twcslen(pVar->wpz) + 1) * TSDB_NCHAR_SIZE, dst + 1);
310
      int32_t len = (int32_t)strlen(dst);
311 312 313 314 315 316 317 318 319
      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:
320 321 322 323
    case TSDB_DATA_TYPE_UTINYINT:
    case TSDB_DATA_TYPE_USMALLINT:
    case TSDB_DATA_TYPE_UINT:
      return sprintf(dst, "%d", (int32_t)pVar->i64);
324 325
    
    case TSDB_DATA_TYPE_BIGINT:
326 327 328
      return sprintf(dst, "%" PRId64, pVar->i64);
    case TSDB_DATA_TYPE_UBIGINT:
      return sprintf(dst, "%" PRIu64, pVar->u64);
329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
    case TSDB_DATA_TYPE_FLOAT:
    case TSDB_DATA_TYPE_DOUBLE:
      return sprintf(dst, "%.9lf", pVar->dKey);
    
    default:
      return 0;
  }
}

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;
355 356
  } else if (memcmp(pstr, L"null", wcslen(L"null")) == 0) {
    return TSDB_DATA_BOOL_NULL;
357 358 359 360 361 362
  } else {
    return -1;
  }
}

static int32_t toBinary(tVariant *pVariant, char **pDest, int32_t *pDestSize) {
363
  const int32_t INITIAL_ALLOC_SIZE = 40;
364
  char *        pBuf = NULL;
365 366

  // it is a in-place convert type for tVariant, local buffer is needed
367 368 369 370 371 372 373
  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) {
dengyihao's avatar
dengyihao 已提交
374
      if (newSize >= INITIAL_ALLOC_SIZE) {
375 376 377
        pBuf = realloc(pBuf, newSize + 1);
      }
      
378
      taosUcs4ToMbs(pVariant->wpz, (int32_t)newSize, pBuf);
379 380 381
      free(pVariant->wpz);
      pBuf[newSize] = 0;
    } else {
382
      taosUcs4ToMbs(pVariant->wpz, (int32_t)newSize, *pDest);
383 384 385
    }
    
  } else {
386 387
    if (IS_SIGNED_NUMERIC_TYPE(pVariant->nType)) {
      sprintf(pBuf == NULL ? *pDest : pBuf, "%" PRId64, pVariant->i64);
388 389 390
    } 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) {
391
      sprintf(pBuf == NULL ? *pDest : pBuf, "%s", (pVariant->i64 == TSDB_TRUE) ? "TRUE" : "FALSE");
392 393 394 395 396 397 398 399 400
    } else if (pVariant->nType == 0) {  // null data
      setNull(pBuf == NULL ? *pDest : pBuf, TSDB_DATA_TYPE_BINARY, 0);
    }
  }
  
  if (pBuf != NULL) {
    *pDest = pBuf;
  }
  
401
  *pDestSize = (int32_t)strlen(*pDest);
402 403 404 405 406 407 408 409
  return 0;
}

static int32_t toNchar(tVariant *pVariant, char **pDest, int32_t *pDestSize) {
  char tmpBuf[40] = {0};
  
  char *  pDst = tmpBuf;
  int32_t nLen = 0;
410 411 412 413 414 415

  // convert the number to string, than convert it to wchar string.
  if (IS_SIGNED_NUMERIC_TYPE(pVariant->nType)) {
    nLen = sprintf(pDst, "%" PRId64, pVariant->i64);
  } else if (IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
    nLen = sprintf(pDst, "%"PRIu64, pVariant->u64);
416 417 418 419 420 421
  } 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) {
422
    nLen = sprintf(pDst, "%s", (pVariant->i64 == TSDB_TRUE) ? "TRUE" : "FALSE");
423 424 425 426
  }
  
  if (*pDest == pVariant->pz) {
    wchar_t *pWStr = calloc(1, (nLen + 1) * TSDB_NCHAR_SIZE);
427 428
    bool ret = taosMbsToUcs4(pDst, nLen, (char *)pWStr, (nLen + 1) * TSDB_NCHAR_SIZE, NULL);
    if (!ret) {
H
Haojun Liao 已提交
429
      tfree(pWStr);
430 431 432
      return -1;
    }

433 434 435 436 437 438
    // free the binary buffer in the first place
    if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
      free(pVariant->wpz);
    }
    
    pVariant->wpz = pWStr;
H
[TD-92]  
Hui Li 已提交
439
    *pDestSize = twcslen(pVariant->wpz);
440 441 442 443 444 445 446
    
    // 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 {
447
    int32_t output = 0;
448

449 450 451 452 453
    bool ret = taosMbsToUcs4(pDst, nLen, *pDest, (nLen + 1) * TSDB_NCHAR_SIZE, &output);
    if (!ret) {
      return -1;
    }

H
hjxilinx 已提交
454
    if (pDestSize != NULL) {
455
      *pDestSize = output;
H
hjxilinx 已提交
456
    }
457 458 459 460 461 462
  }
  
  return 0;
}

static FORCE_INLINE int32_t convertToDouble(char *pStr, int32_t len, double *value) {
H
Haojun Liao 已提交
463
  SStrToken stoken = {.z = pStr, .n = len};
464
  if (TK_ILLEGAL == tGetNumericStringType(&stoken)) {
465 466 467 468 469 470 471
    return -1;
  }
  
  *value = strtod(pStr, NULL);
  return 0;
}

W
wpan 已提交
472
static FORCE_INLINE int32_t convertToInteger(tVariant *pVariant, int64_t *result, int32_t type, bool issigned, bool releaseVariantPtr, bool *converted) {
473
  if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
H
Haojun Liao 已提交
474
    setNull((char *)result, type, tDataTypes[type].bytes);
475 476
    return 0;
  }
477 478

  errno = 0;
D
fix bug  
dapan1121 已提交
479
  if (IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || (pVariant->nType == TSDB_DATA_TYPE_BOOL)) {
480 481 482 483 484
    *result = pVariant->i64;
  } else if (IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
    *result = pVariant->u64;
  } else if (IS_FLOAT_TYPE(pVariant->nType)) {
    *result = (int64_t) pVariant->dKey;
485
  } else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
486
    SStrToken token = {.z = pVariant->pz, .n = pVariant->nLen};
487
    /*int32_t n = */tGetToken(pVariant->pz, &token.type);
488

489 490 491 492 493
    if (token.type == TK_NULL) {
      if (releaseVariantPtr) {
        free(pVariant->pz);
        pVariant->nLen = 0;
      }
494

H
Haojun Liao 已提交
495
      setNull((char *)result, type, tDataTypes[type].bytes);
496 497
      return 0;
    }
498 499 500 501

    // decide if it is a valid number
    token.type = tGetNumericStringType(&token);
    if (token.type == TK_ILLEGAL) {
502 503
      return -1;
    }
504 505

    int64_t res = 0;
506 507
    int32_t t = tStrToInteger(token.z, token.type, token.n, &res, issigned);
    if (t != 0) {
508 509
      return -1;
    }
510 511 512 513 514 515 516

    if (releaseVariantPtr) {
      free(pVariant->pz);
      pVariant->nLen = 0;
    }

    *result = res;
517 518 519 520
  } else if (pVariant->nType == TSDB_DATA_TYPE_NCHAR) {
    errno = 0;
    wchar_t *endPtr = NULL;
    
H
Haojun Liao 已提交
521
    SStrToken token = {0};
522
    token.n = tGetToken(pVariant->pz, &token.type);
523 524
    
    if (token.type == TK_MINUS || token.type == TK_PLUS) {
525
      token.n = tGetToken(pVariant->pz + token.n, &token.type);
526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544
    }
    
    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;
      }
H
Haojun Liao 已提交
545
      setNull((char *)result, type, tDataTypes[type].bytes);
546 547 548 549 550 551 552 553 554 555 556 557 558 559 560
      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;
    }
  }
561

W
wpan 已提交
562 563 564 565
  if (converted) {
    *converted = true;
  }

566
  bool code = false;
567 568

  uint64_t ui = 0;
569 570 571 572 573 574 575 576 577 578
  switch(type) {
    case TSDB_DATA_TYPE_TINYINT:
      code = IS_VALID_TINYINT(*result); break;
    case TSDB_DATA_TYPE_SMALLINT:
      code = IS_VALID_SMALLINT(*result); break;
    case TSDB_DATA_TYPE_INT:
      code = IS_VALID_INT(*result); break;
    case TSDB_DATA_TYPE_BIGINT:
      code = IS_VALID_BIGINT(*result); break;
    case TSDB_DATA_TYPE_UTINYINT:
579 580
      ui = *result;
      code = IS_VALID_UTINYINT(ui); break;
581
    case TSDB_DATA_TYPE_USMALLINT:
582 583
      ui = *result;
      code = IS_VALID_USMALLINT(ui); break;
584
    case TSDB_DATA_TYPE_UINT:
585 586
      ui = *result;
      code = IS_VALID_UINT(ui); break;
587
    case TSDB_DATA_TYPE_UBIGINT:
588 589
      ui = *result;
      code = IS_VALID_UBIGINT(ui); break;
590
  }
591 592

  return code? 0:-1;
593 594 595 596
}

static int32_t convertToBool(tVariant *pVariant, int64_t *pDest) {
  if (pVariant->nType == TSDB_DATA_TYPE_BOOL) {
597 598 599
    *pDest = pVariant->i64;  // in order to be compatible to null of bool
  } else if (IS_NUMERIC_TYPE(pVariant->nType)) {
    *pDest = ((pVariant->i64 != 0) ? TSDB_TRUE : TSDB_FALSE);
600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626
  } 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
 */
627
int32_t tVariantDump(tVariant *pVariant, char *payload, int16_t type, bool includeLengthPrefix) {
W
wpan 已提交
628 629 630 631 632 633 634 635 636 637 638 639
  return tVariantDumpEx(pVariant, payload, type, includeLengthPrefix, NULL, NULL);
}

/*
 * transfer data from variant serve as the implicit data conversion: from input sql string pVariant->nType
 * to column type defined in schema
 */
int32_t tVariantDumpEx(tVariant *pVariant, char *payload, int16_t type, bool includeLengthPrefix, bool *converted, char *extInfo) {
  if (converted) {
    *converted = false;
  }
  
640
  if (pVariant == NULL || (pVariant->nType != 0 && !isValidDataType(pVariant->nType))) {
641 642
    return -1;
  }
643

644
  errno = 0;  // reset global error code
H
Haojun Liao 已提交
645
  int64_t result = 0;
646

647 648
  switch (type) {
    case TSDB_DATA_TYPE_BOOL: {
649
      if (convertToBool(pVariant, &result) < 0) {
650 651
        return -1;
      }
652 653

      *(int8_t *)payload = (int8_t)result;
654 655 656 657
      break;
    }
    
    case TSDB_DATA_TYPE_TINYINT: {
W
wpan 已提交
658 659
      if (convertToInteger(pVariant, &result, type, true, false, converted) < 0) {
        SET_EXT_INFO(converted, result, INT8_MIN + 1, INT8_MAX, extInfo);
660 661
        return -1;
      }
662 663 664 665 666
      *((int8_t *)payload) = (int8_t) result;
      break;
    }

    case TSDB_DATA_TYPE_UTINYINT: {
W
wpan 已提交
667 668
      if (convertToInteger(pVariant, &result, type, false, false, converted) < 0) {        
        SET_EXT_INFO(converted, result, 0, UINT8_MAX - 1, extInfo);
669 670 671
        return -1;
      }
      *((uint8_t *)payload) = (uint8_t) result;
672 673 674 675
      break;
    }
    
    case TSDB_DATA_TYPE_SMALLINT: {
W
wpan 已提交
676 677
      if (convertToInteger(pVariant, &result, type, true, false, converted) < 0) {
        SET_EXT_INFO(converted, result, INT16_MIN + 1, INT16_MAX, extInfo);
678 679 680 681 682
        return -1;
      }
      *((int16_t *)payload) = (int16_t)result;
      break;
    }
683 684

    case TSDB_DATA_TYPE_USMALLINT: {
W
wpan 已提交
685 686
      if (convertToInteger(pVariant, &result, type, false, false, converted) < 0) {
        SET_EXT_INFO(converted, result, 0, UINT16_MAX - 1, extInfo);
687 688 689 690 691
        return -1;
      }
      *((uint16_t *)payload) = (uint16_t)result;
      break;
    }
692 693
    
    case TSDB_DATA_TYPE_INT: {
W
wpan 已提交
694 695
      if (convertToInteger(pVariant, &result, type, true, false, converted) < 0) {
        SET_EXT_INFO(converted, result, INT32_MIN + 1, INT32_MAX, extInfo);
696 697 698 699 700
        return -1;
      }
      *((int32_t *)payload) = (int32_t)result;
      break;
    }
701 702

    case TSDB_DATA_TYPE_UINT: {
W
wpan 已提交
703 704
      if (convertToInteger(pVariant, &result, type, false, false, converted) < 0) {
        SET_EXT_INFO(converted, result, 0, UINT32_MAX - 1, extInfo);
705 706 707 708 709
        return -1;
      }
      *((uint32_t *)payload) = (uint32_t)result;
      break;
    }
710 711
    
    case TSDB_DATA_TYPE_BIGINT: {
W
wpan 已提交
712 713
      if (convertToInteger(pVariant, &result, type, true, false, converted) < 0) {
        SET_EXT_INFO(converted, (int64_t)result, INT64_MIN + 1, INT64_MAX, extInfo);
714 715 716 717 718
        return -1;
      }
      *((int64_t *)payload) = (int64_t)result;
      break;
    }
719 720

    case TSDB_DATA_TYPE_UBIGINT: {
W
wpan 已提交
721 722
      if (convertToInteger(pVariant, &result, type, false, false, converted) < 0) {
        SET_EXT_INFO(converted, (uint64_t)result, 0, UINT64_MAX - 1, extInfo);
723 724 725 726 727 728
        return -1;
      }
      *((uint64_t *)payload) = (uint64_t)result;
      break;
    }

729 730 731 732 733 734 735
    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 {
S
Shengliang Guan 已提交
736
          double  value = -1;
737
          int32_t ret = convertToDouble(pVariant->pz, pVariant->nLen, &value);
738 739 740 741
          if ((errno == ERANGE && (float)value == -1) || (ret != 0)) {
            return -1;
          }

W
wpan 已提交
742 743 744 745 746 747 748 749
          if (converted) {
            *converted = true;
          }
          
          if (value > FLT_MAX || value < -FLT_MAX) {
            SET_EXT_INFO(converted, value, -FLT_MAX, FLT_MAX, extInfo);
            return -1;
          }
S
TD-1530  
Shengliang Guan 已提交
750
          SET_FLOAT_VAL(payload, value);
751
        }
752
      } else if (pVariant->nType == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
W
wpan 已提交
753 754 755 756 757 758 759 760 761
        if (converted) {
          *converted = true;
        }
        
        if (pVariant->i64 > FLT_MAX || pVariant->i64 < -FLT_MAX) {          
          SET_EXT_INFO(converted, pVariant->i64, -FLT_MAX, FLT_MAX, extInfo);
          return -1;
        }

762
        SET_FLOAT_VAL(payload, pVariant->i64);
763
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
W
wpan 已提交
764 765 766 767 768 769 770 771 772
        if (converted) {
          *converted = true;
        }
        
        if (pVariant->dKey > FLT_MAX || pVariant->dKey < -FLT_MAX) {          
          SET_EXT_INFO(converted, pVariant->dKey, -FLT_MAX, FLT_MAX, extInfo);
          return -1;
        }
      
S
TD-1530  
Shengliang Guan 已提交
773
        SET_FLOAT_VAL(payload, pVariant->dKey);
774
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
775
        *((uint32_t *)payload) = TSDB_DATA_FLOAT_NULL;
776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798
        return 0;
      }

      float fv = GET_FLOAT_VAL(payload);
      if (isinf(fv) || isnan(fv) || fv > FLT_MAX || fv < -FLT_MAX) {
        return -1;
      }
      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;
          }

S
TD-1530  
Shengliang Guan 已提交
799
          SET_DOUBLE_VAL(payload, value);
800
        }
801
      } else if (pVariant->nType == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
802
        SET_DOUBLE_VAL(payload, pVariant->i64);
803
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
S
TD-1530  
Shengliang Guan 已提交
804
        SET_DOUBLE_VAL(payload, pVariant->dKey);
805 806 807 808 809 810
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
        *((int64_t *)payload) = TSDB_DATA_DOUBLE_NULL;
        return 0;
      }

      double dv = GET_DOUBLE_VAL(payload);
811
      if (errno == ERANGE || isinf(dv) || isnan(dv)) {
812 813
        return -1;
      }
814

815 816 817 818
      break;
    }
    
    case TSDB_DATA_TYPE_BINARY: {
819 820 821 822 823 824 825 826 827 828
      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);
          }
        }
829
      } else {
830 831
        if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
          setVardataNull(payload, TSDB_DATA_TYPE_BINARY);
832
        } else {
833 834 835 836 837 838 839 840 841 842
          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));
843 844 845 846 847 848 849 850
        }
      }
      break;
    }
    case TSDB_DATA_TYPE_TIMESTAMP: {
      if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
        *((int64_t *)payload) = TSDB_DATA_BIGINT_NULL;
      } else {
851
        *((int64_t *)payload) = pVariant->i64;
852 853 854 855
      }
      break;
    }
    case TSDB_DATA_TYPE_NCHAR: {
L
liu0x54 已提交
856
      int32_t newlen = 0;
857 858 859 860 861
      if (!includeLengthPrefix) {
        if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
          *(uint32_t *)payload = TSDB_DATA_NCHAR_NULL;
        } else {
          if (pVariant->nType != TSDB_DATA_TYPE_NCHAR) {
862 863 864
            if (toNchar(pVariant, &payload, &newlen) != 0) {
              return -1;
            }
865 866 867 868
          } else {
            wcsncpy((wchar_t *)payload, pVariant->wpz, pVariant->nLen);
          }
        }
869
      } else {
870 871
        if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
          setVardataNull(payload, TSDB_DATA_TYPE_NCHAR);
872
        } else {
873 874 875
          char *p = varDataVal(payload);

          if (pVariant->nType != TSDB_DATA_TYPE_NCHAR) {
876 877 878
            if (toNchar(pVariant, &p, &newlen) != 0) {
              return -1;
            }
879
          } else {
D
fix bug  
dapan1121 已提交
880
            memcpy(p, pVariant->wpz, pVariant->nLen);
L
liu0x54 已提交
881
            newlen = pVariant->nLen;
882 883
          }

L
liu0x54 已提交
884
          varDataSetLen(payload, newlen);  // the length may be changed after toNchar function called
885
          assert(p == varDataVal(payload));
886 887
        }
      }
888
      
889 890 891 892 893 894 895
      break;
    }
  }
  
  return 0;
}

W
wpan 已提交
896

897 898 899 900 901 902 903 904 905 906 907 908 909
/*
 * 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
910
      if (convertToBool(pVariant, &pVariant->i64) < 0) {
911 912 913 914 915 916 917 918 919 920
        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: {
W
wpan 已提交
921
      convertToInteger(pVariant, &(pVariant->i64), type, true, true, NULL);
922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947
      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) {
H
Haojun Liao 已提交
948 949
        double tmp = (double) pVariant->i64;
        pVariant->dKey = tmp;
950 951 952 953 954 955 956 957 958 959 960 961 962 963
      }
      
      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) {
964
        if (toNchar(pVariant, &pVariant->pz, &pVariant->nLen) != 0) {
965 966
          return -1;
        }
967 968 969 970 971 972 973
      }
      pVariant->nType = type;
      break;
    }
  }
  
  return 0;
D
fix bug  
dapan1121 已提交
974
}