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 "ttime.h"
21
#include "ttoken.h"
22
#include "ttokendef.h"
23
#include "ttype.h"
24 25
#include "tutil.h"
#include "tvariant.h"
26

W
wpan 已提交
27 28 29 30 31 32 33
#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)

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

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

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

50 51
      break;
    }
52

53 54 55
    case TSDB_DATA_TYPE_TINYINT:
    case TSDB_DATA_TYPE_SMALLINT:
    case TSDB_DATA_TYPE_BIGINT:
56 57 58
    case TSDB_DATA_TYPE_INT:{
      ret = tStrToInteger(token->z, token->type, token->n, &pVar->i64, true);
      if (ret != 0) {
59
        SStrToken t = {0};
60
        tGetToken(token->z, &t.type);
61 62 63 64 65
        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;
        }

66 67 68 69 70 71
        // 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;
        }
72 73
      }

74
      break;
75 76
    }

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

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

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

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

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

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

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

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

  }

232
  if (IS_NUMERIC_TYPE(pSrc->nType) || (pSrc->nType == TSDB_DATA_TYPE_BOOL)) {
233
    pDst->i64 = pSrc->i64;
W
wpan 已提交
234
  } else if (pSrc->nType == TSDB_DATA_TYPE_POINTER_ARRAY) {  // this is only for string array
weixin_48148422's avatar
weixin_48148422 已提交
235 236 237 238 239 240 241
    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 已提交
242 243 244 245 246 247 248 249 250
  } 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);
      }
251
  }
H
Haojun Liao 已提交
252

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

258
int32_t tVariantCompare(const tVariant* p1, const tVariant* p2) {
H
Haojun Liao 已提交
259 260 261 262 263 264 265 266 267 268 269
  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;
  }
270

271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294
  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;
    }
295 296 297
  }
}

298 299 300 301 302 303 304 305 306 307 308 309
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 已提交
310
      taosUcs4ToMbs(pVar->wpz, (twcslen(pVar->wpz) + 1) * TSDB_NCHAR_SIZE, dst + 1);
311
      int32_t len = (int32_t)strlen(dst);
312 313 314 315 316 317 318 319 320
      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:
321 322 323 324
    case TSDB_DATA_TYPE_UTINYINT:
    case TSDB_DATA_TYPE_USMALLINT:
    case TSDB_DATA_TYPE_UINT:
      return sprintf(dst, "%d", (int32_t)pVar->i64);
325 326
    
    case TSDB_DATA_TYPE_BIGINT:
327 328 329
      return sprintf(dst, "%" PRId64, pVar->i64);
    case TSDB_DATA_TYPE_UBIGINT:
      return sprintf(dst, "%" PRIu64, pVar->u64);
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 355
    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;
356 357
  } else if (memcmp(pstr, L"null", wcslen(L"null")) == 0) {
    return TSDB_DATA_BOOL_NULL;
358 359 360 361 362 363
  } else {
    return -1;
  }
}

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

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

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

  // 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);
417 418 419 420 421 422
  } 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) {
423
    nLen = sprintf(pDst, "%s", (pVariant->i64 == TSDB_TRUE) ? "TRUE" : "FALSE");
424 425 426 427
  }
  
  if (*pDest == pVariant->pz) {
    wchar_t *pWStr = calloc(1, (nLen + 1) * TSDB_NCHAR_SIZE);
428 429
    bool ret = taosMbsToUcs4(pDst, nLen, (char *)pWStr, (nLen + 1) * TSDB_NCHAR_SIZE, NULL);
    if (!ret) {
H
Haojun Liao 已提交
430
      tfree(pWStr);
431 432 433
      return -1;
    }

434 435 436 437 438 439
    // 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 已提交
440
    *pDestSize = twcslen(pVariant->wpz);
441 442 443 444 445 446 447
    
    // 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 {
448
    int32_t output = 0;
449

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

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

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

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

  errno = 0;
D
fix bug  
dapan1121 已提交
480
  if (IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || (pVariant->nType == TSDB_DATA_TYPE_BOOL)) {
481 482 483 484 485
    *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;
486
  } else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
487
    SStrToken token = {.z = pVariant->pz, .n = pVariant->nLen};
488
    /*int32_t n = */tGetToken(pVariant->pz, &token.type);
489

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

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

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

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

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

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

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

567
  bool code = false;
568 569

  uint64_t ui = 0;
570 571 572 573 574 575 576 577 578 579
  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:
580 581
      ui = *result;
      code = IS_VALID_UTINYINT(ui); break;
582
    case TSDB_DATA_TYPE_USMALLINT:
583 584
      ui = *result;
      code = IS_VALID_USMALLINT(ui); break;
585
    case TSDB_DATA_TYPE_UINT:
586 587
      ui = *result;
      code = IS_VALID_UINT(ui); break;
588
    case TSDB_DATA_TYPE_UBIGINT:
589 590
      ui = *result;
      code = IS_VALID_UBIGINT(ui); break;
591
  }
592 593

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

static int32_t convertToBool(tVariant *pVariant, int64_t *pDest) {
  if (pVariant->nType == TSDB_DATA_TYPE_BOOL) {
598 599 600
    *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);
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 627
  } 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
 */
628
int32_t tVariantDump(tVariant *pVariant, char *payload, int16_t type, bool includeLengthPrefix) {
W
wpan 已提交
629 630 631 632 633 634 635 636 637 638 639 640
  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;
  }
  
641
  if (pVariant == NULL || (pVariant->nType != 0 && !isValidDataType(pVariant->nType))) {
642 643
    return -1;
  }
644

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

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

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

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

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

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

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

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

W
wpan 已提交
743 744 745 746 747 748 749 750
          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 已提交
751
          SET_FLOAT_VAL(payload, value);
752
        }
753
      } else if (pVariant->nType == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
W
wpan 已提交
754 755 756 757 758 759 760 761 762
        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;
        }

763
        SET_FLOAT_VAL(payload, pVariant->i64);
764
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
W
wpan 已提交
765 766 767 768 769 770 771 772 773
        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 已提交
774
        SET_FLOAT_VAL(payload, pVariant->dKey);
775
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
776
        *((uint32_t *)payload) = TSDB_DATA_FLOAT_NULL;
777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799
        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 已提交
800
          SET_DOUBLE_VAL(payload, value);
801
        }
802
      } else if (pVariant->nType == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
803
        SET_DOUBLE_VAL(payload, pVariant->i64);
804
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
S
TD-1530  
Shengliang Guan 已提交
805
        SET_DOUBLE_VAL(payload, pVariant->dKey);
806 807 808 809 810 811
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
        *((int64_t *)payload) = TSDB_DATA_DOUBLE_NULL;
        return 0;
      }

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

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

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

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

W
wpan 已提交
897

898 899 900 901 902 903 904 905 906 907 908 909 910
/*
 * 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
911
      if (convertToBool(pVariant, &pVariant->i64) < 0) {
912 913 914 915 916 917 918 919 920 921
        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 已提交
922
      convertToInteger(pVariant, &(pVariant->i64), type, true, true, NULL);
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 948
      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 已提交
949 950
        double tmp = (double) pVariant->i64;
        pVariant->dKey = tmp;
951 952 953 954 955 956 957 958 959 960 961 962 963 964
      }
      
      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) {
965
        if (toNchar(pVariant, &pVariant->pz, &pVariant->nLen) != 0) {
966 967
          return -1;
        }
968 969 970 971 972 973 974
      }
      pVariant->nType = type;
      break;
    }
  }
  
  return 0;
D
fix bug  
dapan1121 已提交
975
}