tvariant.c 25.5 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

26 27 28
void tVariantCreate(tVariant *pVar, SStrToken *token) {
  int32_t ret = 0;
  int32_t type = token->type;
29 30

  memset(pVar, 0, sizeof(tVariant));
31 32

  switch (token->type) {
33
    case TSDB_DATA_TYPE_BOOL: {
34
      int32_t k = strncasecmp(token->z, "true", 4);
35
      if (k == 0) {
36
        pVar->i64 = TSDB_TRUE;
37
      } else {
38
        assert(strncasecmp(token->z, "false", 5) == 0);
39
        pVar->i64 = TSDB_FALSE;
40
      }
41

42 43
      break;
    }
44

45 46 47
    case TSDB_DATA_TYPE_TINYINT:
    case TSDB_DATA_TYPE_SMALLINT:
    case TSDB_DATA_TYPE_BIGINT:
48 49 50
    case TSDB_DATA_TYPE_INT:{
      ret = tStrToInteger(token->z, token->type, token->n, &pVar->i64, true);
      if (ret != 0) {
51
        SStrToken t = {0};
52
        tGetToken(token->z, &t.type);
53 54 55 56 57
        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;
        }

58 59 60 61 62 63
        // 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;
        }
64 65
      }

66
      break;
67 68
    }

69
    case TSDB_DATA_TYPE_DOUBLE:
70 71
    case TSDB_DATA_TYPE_FLOAT: {
      pVar->dKey = strtod(token->z, NULL);
72
      break;
73 74
    }

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

153
      pVar->wpz = calloc(1, (lenInwchar + 1) * TSDB_NCHAR_SIZE);
154
      memcpy(pVar->wpz, pz, lenInwchar * TSDB_NCHAR_SIZE);
155
      pVar->nLen = (int32_t)len;
156 157 158
      
      break;
    }
159
    case TSDB_DATA_TYPE_BINARY: {  // todo refactor, extract a method
H
Haojun Liao 已提交
160
      pVar->pz = calloc(len + 1, sizeof(char));
161
      memcpy(pVar->pz, pz, len);
162
      pVar->nLen = (int32_t)len;
163 164 165 166
      break;
    }
    
    default:
167
      pVar->i64 = GET_INT32_VAL(pz);
168
      pVar->nLen = tDataTypes[TSDB_DATA_TYPE_INT].bytes;
169 170 171 172 173 174 175 176 177
  }
  
  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 已提交
178
    tfree(pVar->pz);
179 180
    pVar->nLen = 0;
  }
weixin_48148422's avatar
weixin_48148422 已提交
181 182 183 184 185 186 187 188 189 190 191

  // 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;
  }
192 193
}

194 195 196 197 198
bool tVariantIsValid(tVariant *pVar) {
  assert(pVar != NULL);
  return isValidDataType(pVar->nType);
}

199 200 201
void tVariantAssign(tVariant *pDst, const tVariant *pSrc) {
  if (pSrc == NULL || pDst == NULL) return;
  
H
Haojun Liao 已提交
202
  pDst->nType = pSrc->nType;
H
Haojun Liao 已提交
203
  if (pSrc->nType == TSDB_DATA_TYPE_BINARY || pSrc->nType == TSDB_DATA_TYPE_NCHAR) {
H
Haojun Liao 已提交
204 205
    int32_t len = pSrc->nLen + TSDB_NCHAR_SIZE;
    char* p = realloc(pDst->pz, len);
H
Haojun Liao 已提交
206 207
    assert(p);

H
Haojun Liao 已提交
208
    memset(p, 0, len);
H
Haojun Liao 已提交
209 210
    pDst->pz = p;

H
Haojun Liao 已提交
211 212
    memcpy(pDst->pz, pSrc->pz, pSrc->nLen);
    pDst->nLen = pSrc->nLen;
H
Haojun Liao 已提交
213 214 215 216
    return;

  }

217
  if (IS_NUMERIC_TYPE(pSrc->nType) || (pSrc->nType == TSDB_DATA_TYPE_BOOL)) {
218
    pDst->i64 = pSrc->i64;
H
Haojun Liao 已提交
219
  } else if (pSrc->nType == TSDB_DATA_TYPE_ARRAY) {  // this is only for string array
weixin_48148422's avatar
weixin_48148422 已提交
220 221 222 223 224 225 226
    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);
    }
227
  }
H
Haojun Liao 已提交
228

H
Haojun Liao 已提交
229
  if (pDst->nType != TSDB_DATA_TYPE_ARRAY) {
H
Haojun Liao 已提交
230
    pDst->nLen = tDataTypes[pDst->nType].bytes;
H
Haojun Liao 已提交
231
  }
232 233
}

234
int32_t tVariantCompare(const tVariant* p1, const tVariant* p2) {
H
Haojun Liao 已提交
235 236 237 238 239 240 241 242 243 244 245
  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;
  }
246

247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270
  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;
    }
271 272 273
  }
}

274 275 276 277 278 279 280 281 282 283 284 285
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 已提交
286
      taosUcs4ToMbs(pVar->wpz, (twcslen(pVar->wpz) + 1) * TSDB_NCHAR_SIZE, dst + 1);
287
      int32_t len = (int32_t)strlen(dst);
288 289 290 291 292 293 294 295 296
      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:
297 298 299 300
    case TSDB_DATA_TYPE_UTINYINT:
    case TSDB_DATA_TYPE_USMALLINT:
    case TSDB_DATA_TYPE_UINT:
      return sprintf(dst, "%d", (int32_t)pVar->i64);
301 302
    
    case TSDB_DATA_TYPE_BIGINT:
303 304 305
      return sprintf(dst, "%" PRId64, pVar->i64);
    case TSDB_DATA_TYPE_UBIGINT:
      return sprintf(dst, "%" PRIu64, pVar->u64);
306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331
    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;
332 333
  } else if (memcmp(pstr, L"null", wcslen(L"null")) == 0) {
    return TSDB_DATA_BOOL_NULL;
334 335 336 337 338 339
  } else {
    return -1;
  }
}

static int32_t toBinary(tVariant *pVariant, char **pDest, int32_t *pDestSize) {
340
  const int32_t INITIAL_ALLOC_SIZE = 40;
341
  char *        pBuf = NULL;
342 343

  // it is a in-place convert type for tVariant, local buffer is needed
344 345 346 347 348 349 350
  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 已提交
351
      if (newSize >= INITIAL_ALLOC_SIZE) {
352 353 354
        pBuf = realloc(pBuf, newSize + 1);
      }
      
355
      taosUcs4ToMbs(pVariant->wpz, (int32_t)newSize, pBuf);
356 357 358
      free(pVariant->wpz);
      pBuf[newSize] = 0;
    } else {
359
      taosUcs4ToMbs(pVariant->wpz, (int32_t)newSize, *pDest);
360 361 362
    }
    
  } else {
363 364
    if (IS_SIGNED_NUMERIC_TYPE(pVariant->nType)) {
      sprintf(pBuf == NULL ? *pDest : pBuf, "%" PRId64, pVariant->i64);
365 366 367
    } 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) {
368
      sprintf(pBuf == NULL ? *pDest : pBuf, "%s", (pVariant->i64 == TSDB_TRUE) ? "TRUE" : "FALSE");
369 370 371 372 373 374 375 376 377
    } else if (pVariant->nType == 0) {  // null data
      setNull(pBuf == NULL ? *pDest : pBuf, TSDB_DATA_TYPE_BINARY, 0);
    }
  }
  
  if (pBuf != NULL) {
    *pDest = pBuf;
  }
  
378
  *pDestSize = (int32_t)strlen(*pDest);
379 380 381 382 383 384 385 386
  return 0;
}

static int32_t toNchar(tVariant *pVariant, char **pDest, int32_t *pDestSize) {
  char tmpBuf[40] = {0};
  
  char *  pDst = tmpBuf;
  int32_t nLen = 0;
387 388 389 390 391 392

  // 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);
393 394 395 396 397 398
  } 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) {
399
    nLen = sprintf(pDst, "%s", (pVariant->i64 == TSDB_TRUE) ? "TRUE" : "FALSE");
400 401 402 403
  }
  
  if (*pDest == pVariant->pz) {
    wchar_t *pWStr = calloc(1, (nLen + 1) * TSDB_NCHAR_SIZE);
404 405 406 407 408
    bool ret = taosMbsToUcs4(pDst, nLen, (char *)pWStr, (nLen + 1) * TSDB_NCHAR_SIZE, NULL);
    if (!ret) {
      return -1;
    }

409 410 411 412 413 414
    // 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 已提交
415
    *pDestSize = twcslen(pVariant->wpz);
416 417 418 419 420 421 422
    
    // 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 {
423
    int32_t output = 0;
424

425 426 427 428 429
    bool ret = taosMbsToUcs4(pDst, nLen, *pDest, (nLen + 1) * TSDB_NCHAR_SIZE, &output);
    if (!ret) {
      return -1;
    }

H
hjxilinx 已提交
430
    if (pDestSize != NULL) {
431
      *pDestSize = output;
H
hjxilinx 已提交
432
    }
433 434 435 436 437 438
  }
  
  return 0;
}

static FORCE_INLINE int32_t convertToDouble(char *pStr, int32_t len, double *value) {
H
Haojun Liao 已提交
439
  SStrToken stoken = {.z = pStr, .n = len};
440
  if (TK_ILLEGAL == tGetNumericStringType(&stoken)) {
441 442 443 444 445 446 447
    return -1;
  }
  
  *value = strtod(pStr, NULL);
  return 0;
}

448
static FORCE_INLINE int32_t convertToInteger(tVariant *pVariant, int64_t *result, int32_t type, bool issigned, bool releaseVariantPtr) {
449
  if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
H
Haojun Liao 已提交
450
    setNull((char *)result, type, tDataTypes[type].bytes);
451 452
    return 0;
  }
453 454

  errno = 0;
D
fix bug  
dapan1121 已提交
455
  if (IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || (pVariant->nType == TSDB_DATA_TYPE_BOOL)) {
456 457 458 459 460
    *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;
461
  } else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
462
    SStrToken token = {.z = pVariant->pz, .n = pVariant->nLen};
463
    /*int32_t n = */tGetToken(pVariant->pz, &token.type);
464

465 466 467 468 469
    if (token.type == TK_NULL) {
      if (releaseVariantPtr) {
        free(pVariant->pz);
        pVariant->nLen = 0;
      }
470

H
Haojun Liao 已提交
471
      setNull((char *)result, type, tDataTypes[type].bytes);
472 473
      return 0;
    }
474 475 476 477

    // decide if it is a valid number
    token.type = tGetNumericStringType(&token);
    if (token.type == TK_ILLEGAL) {
478 479
      return -1;
    }
480 481

    int64_t res = 0;
482 483
    int32_t t = tStrToInteger(token.z, token.type, token.n, &res, issigned);
    if (t != 0) {
484 485
      return -1;
    }
486 487 488 489 490 491 492

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

    *result = res;
493 494 495 496
  } else if (pVariant->nType == TSDB_DATA_TYPE_NCHAR) {
    errno = 0;
    wchar_t *endPtr = NULL;
    
H
Haojun Liao 已提交
497
    SStrToken token = {0};
498
    token.n = tGetToken(pVariant->pz, &token.type);
499 500
    
    if (token.type == TK_MINUS || token.type == TK_PLUS) {
501
      token.n = tGetToken(pVariant->pz + token.n, &token.type);
502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520
    }
    
    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 已提交
521
      setNull((char *)result, type, tDataTypes[type].bytes);
522 523 524 525 526 527 528 529 530 531 532 533 534 535 536
      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;
    }
  }
537 538

  bool code = false;
539 540

  uint64_t ui = 0;
541 542 543 544 545 546 547 548 549 550
  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:
551 552
      ui = *result;
      code = IS_VALID_UTINYINT(ui); break;
553
    case TSDB_DATA_TYPE_USMALLINT:
554 555
      ui = *result;
      code = IS_VALID_USMALLINT(ui); break;
556
    case TSDB_DATA_TYPE_UINT:
557 558
      ui = *result;
      code = IS_VALID_UINT(ui); break;
559
    case TSDB_DATA_TYPE_UBIGINT:
560 561
      ui = *result;
      code = IS_VALID_UBIGINT(ui); break;
562
  }
563 564

  return code? 0:-1;
565 566 567 568
}

static int32_t convertToBool(tVariant *pVariant, int64_t *pDest) {
  if (pVariant->nType == TSDB_DATA_TYPE_BOOL) {
569 570 571
    *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);
572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598
  } 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
 */
599
int32_t tVariantDump(tVariant *pVariant, char *payload, int16_t type, bool includeLengthPrefix) {
600
  if (pVariant == NULL || (pVariant->nType != 0 && !isValidDataType(pVariant->nType))) {
601 602
    return -1;
  }
603

604
  errno = 0;  // reset global error code
605 606
  int64_t result;

607 608
  switch (type) {
    case TSDB_DATA_TYPE_BOOL: {
609
      if (convertToBool(pVariant, &result) < 0) {
610 611
        return -1;
      }
612 613

      *(int8_t *)payload = (int8_t)result;
614 615 616 617
      break;
    }
    
    case TSDB_DATA_TYPE_TINYINT: {
618
      if (convertToInteger(pVariant, &result, type, true, false) < 0) {
619 620
        return -1;
      }
621 622 623 624 625 626 627 628 629
      *((int8_t *)payload) = (int8_t) result;
      break;
    }

    case TSDB_DATA_TYPE_UTINYINT: {
      if (convertToInteger(pVariant, &result, type, false, false) < 0) {
        return -1;
      }
      *((uint8_t *)payload) = (uint8_t) result;
630 631 632 633
      break;
    }
    
    case TSDB_DATA_TYPE_SMALLINT: {
634
      if (convertToInteger(pVariant, &result, type, true, false) < 0) {
635 636 637 638 639
        return -1;
      }
      *((int16_t *)payload) = (int16_t)result;
      break;
    }
640 641 642 643 644 645 646 647

    case TSDB_DATA_TYPE_USMALLINT: {
      if (convertToInteger(pVariant, &result, type, false, false) < 0) {
        return -1;
      }
      *((uint16_t *)payload) = (uint16_t)result;
      break;
    }
648 649
    
    case TSDB_DATA_TYPE_INT: {
650
      if (convertToInteger(pVariant, &result, type, true, false) < 0) {
651 652 653 654 655
        return -1;
      }
      *((int32_t *)payload) = (int32_t)result;
      break;
    }
656 657 658 659 660 661 662 663

    case TSDB_DATA_TYPE_UINT: {
      if (convertToInteger(pVariant, &result, type, false, false) < 0) {
        return -1;
      }
      *((uint32_t *)payload) = (uint32_t)result;
      break;
    }
664 665
    
    case TSDB_DATA_TYPE_BIGINT: {
666
      if (convertToInteger(pVariant, &result, type, true, false) < 0) {
667 668 669 670 671
        return -1;
      }
      *((int64_t *)payload) = (int64_t)result;
      break;
    }
672 673 674 675 676 677 678 679 680

    case TSDB_DATA_TYPE_UBIGINT: {
      if (convertToInteger(pVariant, &result, type, false, false) < 0) {
        return -1;
      }
      *((uint64_t *)payload) = (uint64_t)result;
      break;
    }

681 682 683 684 685 686 687
    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 已提交
688
          double  value = -1;
689
          int32_t ret = convertToDouble(pVariant->pz, pVariant->nLen, &value);
690 691 692 693
          if ((errno == ERANGE && (float)value == -1) || (ret != 0)) {
            return -1;
          }

S
TD-1530  
Shengliang Guan 已提交
694
          SET_FLOAT_VAL(payload, value);
695
        }
696
      } else if (pVariant->nType == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
697
        SET_FLOAT_VAL(payload, pVariant->i64);
698
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
S
TD-1530  
Shengliang Guan 已提交
699
        SET_FLOAT_VAL(payload, pVariant->dKey);
700
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
701
        *((uint32_t *)payload) = TSDB_DATA_FLOAT_NULL;
702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724
        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 已提交
725
          SET_DOUBLE_VAL(payload, value);
726
        }
727
      } else if (pVariant->nType == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
728
        SET_DOUBLE_VAL(payload, pVariant->i64);
729
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
S
TD-1530  
Shengliang Guan 已提交
730
        SET_DOUBLE_VAL(payload, pVariant->dKey);
731 732 733 734 735 736
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
        *((int64_t *)payload) = TSDB_DATA_DOUBLE_NULL;
        return 0;
      }

      double dv = GET_DOUBLE_VAL(payload);
737
      if (errno == ERANGE || isinf(dv) || isnan(dv)) {
738 739
        return -1;
      }
740

741 742 743 744
      break;
    }
    
    case TSDB_DATA_TYPE_BINARY: {
745 746 747 748 749 750 751 752 753 754
      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);
          }
        }
755
      } else {
756 757
        if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
          setVardataNull(payload, TSDB_DATA_TYPE_BINARY);
758
        } else {
759 760 761 762 763 764 765 766 767 768
          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));
769 770 771 772 773 774 775 776
        }
      }
      break;
    }
    case TSDB_DATA_TYPE_TIMESTAMP: {
      if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
        *((int64_t *)payload) = TSDB_DATA_BIGINT_NULL;
      } else {
777
        *((int64_t *)payload) = pVariant->i64;
778 779 780 781
      }
      break;
    }
    case TSDB_DATA_TYPE_NCHAR: {
L
liu0x54 已提交
782
      int32_t newlen = 0;
783 784 785 786 787
      if (!includeLengthPrefix) {
        if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
          *(uint32_t *)payload = TSDB_DATA_NCHAR_NULL;
        } else {
          if (pVariant->nType != TSDB_DATA_TYPE_NCHAR) {
788 789 790
            if (toNchar(pVariant, &payload, &newlen) != 0) {
              return -1;
            }
791 792 793 794
          } else {
            wcsncpy((wchar_t *)payload, pVariant->wpz, pVariant->nLen);
          }
        }
795
      } else {
796 797
        if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
          setVardataNull(payload, TSDB_DATA_TYPE_NCHAR);
798
        } else {
799 800 801
          char *p = varDataVal(payload);

          if (pVariant->nType != TSDB_DATA_TYPE_NCHAR) {
802 803 804
            if (toNchar(pVariant, &p, &newlen) != 0) {
              return -1;
            }
805
          } else {
D
fix bug  
dapan1121 已提交
806
            memcpy(p, pVariant->wpz, pVariant->nLen);
L
liu0x54 已提交
807
            newlen = pVariant->nLen;
808 809
          }

L
liu0x54 已提交
810
          varDataSetLen(payload, newlen);  // the length may be changed after toNchar function called
811
          assert(p == varDataVal(payload));
812 813
        }
      }
814
      
815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834
      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
835
      if (convertToBool(pVariant, &pVariant->i64) < 0) {
836 837 838 839 840 841 842 843 844 845
        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: {
846
      convertToInteger(pVariant, &(pVariant->i64), type, true, true);
847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872
      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) {
873
        pVariant->dKey = (double)(pVariant->i64);
874 875 876 877 878 879 880 881 882 883 884 885 886 887
      }
      
      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) {
888
        if (toNchar(pVariant, &pVariant->pz, &pVariant->nLen) != 0) {
889 890
          return -1;
        }
891 892 893 894 895 896 897
      }
      pVariant->nType = type;
      break;
    }
  }
  
  return 0;
D
fix bug  
dapan1121 已提交
898
}