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

H
hjxilinx 已提交
17
#include "taos.h"
H
Haojun Liao 已提交
18
#include "thash.h"
H
hjxilinx 已提交
19
#include "taosdef.h"
20
#include "ttime.h"
21
#include "ttokendef.h"
H
Haojun Liao 已提交
22
#include "ttypes.h"
23 24
#include "tutil.h"
#include "tvariant.h"
25

W
wpan 已提交
26 27 28 29 30 31
#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)
H
Haojun Liao 已提交
32 33
#if 0
void tVariantCreate(SVariant *pVar, SToken *token) {
34 35
  int32_t ret = 0;
  int32_t type = token->type;
36

H
Haojun Liao 已提交
37
  memset(pVar, 0, sizeof(SVariant));
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) {
H
Haojun Liao 已提交
58
        SToken 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
    case TSDB_DATA_TYPE_FLOAT: {
H
Haojun Liao 已提交
78
      pVar->d = 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
    
    default: {  // nType == 0 means the null value
      type = TSDB_DATA_TYPE_NULL;
    }
  }
  
  pVar->nType = type;
}
H
Haojun Liao 已提交
99
#endif
100 101

/**
H
Haojun Liao 已提交
102
 * create SVariant from binary string, not ascii data
103 104 105 106 107
 * @param pVar
 * @param pz
 * @param len
 * @param type
 */
H
Haojun Liao 已提交
108
void tVariantCreateFromBinary(SVariant *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;
H
Haojun Liao 已提交
154
      pVar->d = GET_DOUBLE_VAL(pz);
155 156 157
      break;
    }
    case TSDB_DATA_TYPE_FLOAT: {
158
      pVar->nLen = tDataTypes[type].bytes;
H
Haojun Liao 已提交
159
      pVar->d = GET_FLOAT_VAL(pz);
160 161 162
      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
  }
  
  pVar->nType = type;
}

H
Haojun Liao 已提交
186
void tVariantDestroy(SVariant *pVar) {
187 188 189
  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
}

H
Haojun Liao 已提交
209
bool tVariantIsValid(SVariant *pVar) {
210 211 212 213
  assert(pVar != NULL);
  return isValidDataType(pVar->nType);
}

H
Haojun Liao 已提交
214
void tVariantAssign(SVariant *pDst, const SVariant *pSrc) {
215 216
  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
}

H
Haojun Liao 已提交
258
int32_t tVariantCompare(const SVariant* p1, const SVariant* 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
  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) {
H
Haojun Liao 已提交
278
    if (p1->d == p2->d) {
279 280
      return 0;
    } else {
H
Haojun Liao 已提交
281
      return p1->d > p2->d? 1:-1;
282 283 284 285 286 287 288 289 290 291 292 293 294
    }
  } 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
  }
}

H
Haojun Liao 已提交
298
int32_t tVariantToString(SVariant *pVar, char *dst) {
299 300 301 302 303 304 305 306 307 308 309
  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
    case TSDB_DATA_TYPE_FLOAT:
    case TSDB_DATA_TYPE_DOUBLE:
H
Haojun Liao 已提交
332
      return sprintf(dst, "%.9lf", pVar->d);
333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355
    
    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
  } else {
    return -1;
  }
}

H
Haojun Liao 已提交
363
static int32_t toBinary(SVariant *pVariant, char **pDest, int32_t *pDestSize) {
364
  const int32_t INITIAL_ALLOC_SIZE = 40;
365
  char *        pBuf = NULL;
366

H
Haojun Liao 已提交
367
  // it is a in-place convert type for SVariant, 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
    } else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
H
Haojun Liao 已提交
390
      sprintf(pBuf == NULL ? *pDest : pBuf, "%lf", pVariant->d);
391
    } 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
  return 0;
}

H
Haojun Liao 已提交
406
static int32_t toNchar(SVariant *pVariant, char **pDest, int32_t *pDestSize) {
407 408 409 410
  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
  } else if (pVariant->nType == TSDB_DATA_TYPE_DOUBLE || pVariant->nType == TSDB_DATA_TYPE_FLOAT) {
H
Haojun Liao 已提交
418
    nLen = sprintf(pDst, "%lf", pVariant->d);
419 420 421 422
  } 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 465 466 467 468 469
//  SToken stoken = {.z = pStr, .n = len};
//  if (TK_ILLEGAL == tGetNumericStringType(&stoken)) {
//    return -1;
//  }
//
//  *value = strtod(pStr, NULL);
470 471 472
  return 0;
}

H
Haojun Liao 已提交
473
static FORCE_INLINE int32_t convertToInteger(SVariant *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;
  }
H
Haojun Liao 已提交
478
#if 0
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
    *result = pVariant->i64;
  } else if (IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
    *result = pVariant->u64;
  } else if (IS_FLOAT_TYPE(pVariant->nType)) {
H
Haojun Liao 已提交
485
    *result = (int64_t) pVariant->d;
486
  } else if (pVariant->nType == TSDB_DATA_TYPE_BINARY) {
H
Haojun Liao 已提交
487
    SToken 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
    SToken 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

H
Haojun Liao 已提交
593

594
  return code? 0:-1;
H
Haojun Liao 已提交
595 596
#endif
  return 0;
597 598
}

H
Haojun Liao 已提交
599
static int32_t convertToBool(SVariant *pVariant, int64_t *pDest) {
600
  if (pVariant->nType == TSDB_DATA_TYPE_BOOL) {
601 602 603
    *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);
604
  } else if (pVariant->nType == TSDB_DATA_TYPE_FLOAT || pVariant->nType == TSDB_DATA_TYPE_DOUBLE) {
H
Haojun Liao 已提交
605
    *pDest = ((pVariant->d != 0) ? TSDB_TRUE : TSDB_FALSE);
606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630
  } 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
 */
H
Haojun Liao 已提交
631
int32_t tVariantDumpEx(SVariant *pVariant, char *payload, int16_t type, bool includeLengthPrefix, bool *converted, char *extInfo) {
W
wpan 已提交
632 633 634 635
  if (converted) {
    *converted = false;
  }
  
636
  if (pVariant == NULL || (pVariant->nType != 0 && !isValidDataType(pVariant->nType))) {
637 638
    return -1;
  }
639

640
  errno = 0;  // reset global error code
H
Haojun Liao 已提交
641
  int64_t result = 0;
642

643 644
  switch (type) {
    case TSDB_DATA_TYPE_BOOL: {
645
      if (convertToBool(pVariant, &result) < 0) {
646 647
        return -1;
      }
648 649

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

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

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

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

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

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

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

758
        SET_FLOAT_VAL(payload, pVariant->i64);
759
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
W
wpan 已提交
760 761 762 763
        if (converted) {
          *converted = true;
        }
        
H
Haojun Liao 已提交
764 765
        if (pVariant->d > FLT_MAX || pVariant->d < -FLT_MAX) {          
          SET_EXT_INFO(converted, pVariant->d, -FLT_MAX, FLT_MAX, extInfo);
W
wpan 已提交
766 767 768
          return -1;
        }
      
H
Haojun Liao 已提交
769
        SET_FLOAT_VAL(payload, pVariant->d);
770
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
771
        *((uint32_t *)payload) = TSDB_DATA_FLOAT_NULL;
772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794
        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 已提交
795
          SET_DOUBLE_VAL(payload, value);
796
        }
797
      } else if (pVariant->nType == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(pVariant->nType) || IS_UNSIGNED_NUMERIC_TYPE(pVariant->nType)) {
798
        SET_DOUBLE_VAL(payload, pVariant->i64);
799
      } else if (IS_FLOAT_TYPE(pVariant->nType)) {
H
Haojun Liao 已提交
800
        SET_DOUBLE_VAL(payload, pVariant->d);
801 802 803 804 805 806
      } else if (pVariant->nType == TSDB_DATA_TYPE_NULL) {
        *((int64_t *)payload) = TSDB_DATA_DOUBLE_NULL;
        return 0;
      }

      double dv = GET_DOUBLE_VAL(payload);
807
      if (errno == ERANGE || isinf(dv) || isnan(dv)) {
808 809
        return -1;
      }
810

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

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

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

H
Haojun Liao 已提交
892 893 894 895 896 897 898
/*
 * transfer data from variant serve as the implicit data conversion: from input sql string pVariant->nType
 * to column type defined in schema
 */
int32_t tVariantDump(SVariant *pVariant, char *payload, int16_t type, bool includeLengthPrefix) {
  return tVariantDumpEx(pVariant, payload, type, includeLengthPrefix, NULL, NULL);
}
W
wpan 已提交
899

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