create_table.c 14.4 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
/*
 * 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/>.
 */

#define _DEFAULT_SOURCE
#include "os.h"
S
Shengliang Guan 已提交
18
#include "taos.h"
19
#include "taoserror.h"
S
Shengliang Guan 已提交
20
#include "ulog.h"
21 22 23 24 25

#define GREEN "\033[1;32m"
#define NC "\033[0m"

char    dbName[32] = "db";
S
Shengliang Guan 已提交
26
char    stbName[64] = "st";
S
Shengliang Guan 已提交
27
int32_t numOfThreads = 1;
S
Shengliang Guan 已提交
28
int64_t numOfTables = 200000;
S
Shengliang Guan 已提交
29 30
int32_t createTable = 1;
int32_t insertData = 0;
P
plum-lihui 已提交
31 32
int32_t batchNumOfTbl = 100;
int32_t batchNumOfRow = 1;
S
Shengliang Guan 已提交
33
int32_t numOfVgroups = 2;
L
lihui 已提交
34
int32_t showTablesFlag = 0;
P
plum-lihui 已提交
35 36 37 38
int32_t queryFlag = 0;

int64_t startTimestamp = 1640966400000; // 2020-01-01 00:00:00.000

39 40

typedef struct {
S
Shengliang Guan 已提交
41 42
  int64_t   tableBeginIndex;
  int64_t   tableEndIndex;
43 44
  int32_t   threadIndex;
  char      dbName[32];
S
Shengliang Guan 已提交
45
  char      stbName[64];
46
  float     createTableSpeed;
S
Shengliang Guan 已提交
47
  float     insertDataSpeed;
S
Shengliang Guan 已提交
48
  int64_t   startMs;
L
lihui 已提交
49 50
  int64_t   maxDelay;
  int64_t   minDelay;
51 52 53
  pthread_t thread;
} SThreadInfo;

54 55 56
// void  parseArgument(int32_t argc, char *argv[]);
// void *threadFunc(void *param);
// void  createDbAndStb();
57

S
Shengliang Guan 已提交
58
void createDbAndStb() {
59 60
  pPrint("start to create db and stable");
  char qstr[64000];
S
Shengliang Guan 已提交
61

62 63
  TAOS *con = taos_connect(NULL, "root", "taosdata", NULL, 0);
  if (con == NULL) {
L
lihui 已提交
64
    pError("failed to connect to DB, reason:%s", taos_errstr(NULL));
65 66 67
    exit(1);
  }

S
Shengliang Guan 已提交
68
  sprintf(qstr, "create database if not exists %s vgroups %d", dbName, numOfVgroups);
L
lihui 已提交
69 70
  TAOS_RES *pRes = taos_query(con, qstr);
  int32_t   code = taos_errno(pRes);
71
  if (code != 0) {
72 73
    pError("failed to create database:%s, sql:%s, code:%d reason:%s", dbName, qstr, taos_errno(pRes),
           taos_errstr(pRes));
74 75
    exit(0);
  }
L
lihui 已提交
76
  taos_free_result(pRes);
77 78

  sprintf(qstr, "use %s", dbName);
L
lihui 已提交
79 80
  pRes = taos_query(con, qstr);
  code = taos_errno(pRes);
81
  if (code != 0) {
L
lihui 已提交
82
    pError("failed to use db, code:%d reason:%s", taos_errno(pRes), taos_errstr(pRes));
83 84
    exit(0);
  }
L
lihui 已提交
85
  taos_free_result(pRes);
86

87
  sprintf(qstr, "create table %s (ts timestamp, i int) tags (j bigint)", stbName);
L
lihui 已提交
88 89
  pRes = taos_query(con, qstr);
  code = taos_errno(pRes);
S
Shengliang Guan 已提交
90
  if (code != 0) {
L
lihui 已提交
91
    pError("failed to use db, code:%d reason:%s", taos_errno(pRes), taos_errstr(pRes));
S
Shengliang Guan 已提交
92 93
    exit(0);
  }
L
lihui 已提交
94
  taos_free_result(pRes);
S
Shengliang Guan 已提交
95

96 97 98
  taos_close(con);
}

S
Shengliang Guan 已提交
99
void printCreateProgress(SThreadInfo *pInfo, int64_t t) {
S
Shengliang Guan 已提交
100
  int64_t endMs = taosGetTimestampMs();
S
Shengliang Guan 已提交
101
  int64_t totalTables = t - pInfo->tableBeginIndex;
S
Shengliang Guan 已提交
102 103 104
  float   seconds = (endMs - pInfo->startMs) / 1000.0;
  float   speed = totalTables / seconds;
  pInfo->createTableSpeed = speed;
S
Shengliang Guan 已提交
105 106
  pPrint("thread:%d, %" PRId64 " tables created, time:%.2f sec, speed:%.1f tables/second, ", pInfo->threadIndex,
         totalTables, seconds, speed);
S
Shengliang Guan 已提交
107 108
}

S
Shengliang Guan 已提交
109
void printInsertProgress(SThreadInfo *pInfo, int64_t t) {
S
Shengliang Guan 已提交
110
  int64_t endMs = taosGetTimestampMs();
S
Shengliang Guan 已提交
111
  int64_t totalTables = t - pInfo->tableBeginIndex;
S
Shengliang Guan 已提交
112 113 114
  float   seconds = (endMs - pInfo->startMs) / 1000.0;
  float   speed = totalTables / seconds;
  pInfo->insertDataSpeed = speed;
S
Shengliang Guan 已提交
115 116
  pPrint("thread:%d, %" PRId64 " rows inserted, time:%.2f sec, speed:%.1f rows/second, ", pInfo->threadIndex,
         totalTables, seconds, speed);
S
Shengliang Guan 已提交
117 118
}

L
lihui 已提交
119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
static int64_t getResult(TAOS_RES *tres) {
  TAOS_ROW row = taos_fetch_row(tres);
  if (row == NULL) {
    return 0;
  }

  int         num_fields = taos_num_fields(tres);
  TAOS_FIELD *fields = taos_fetch_fields(tres);
  int         precision = taos_result_precision(tres);

  int64_t numOfRows = 0;
  do {
    numOfRows++;
    row = taos_fetch_row(tres);
  } while (row != NULL);

  return numOfRows;
}

138
void showTables() {
L
lihui 已提交
139
  pPrint("start to show tables");
140
  char qstr[128];
L
lihui 已提交
141 142 143 144 145 146 147

  TAOS *con = taos_connect(NULL, "root", "taosdata", NULL, 0);
  if (con == NULL) {
    pError("failed to connect to DB, reason:%s", taos_errstr(NULL));
    exit(1);
  }

148
  snprintf(qstr, 128, "use %s", dbName);
L
lihui 已提交
149
  TAOS_RES *pRes = taos_query(con, qstr);
150
  int       code = taos_errno(pRes);
L
lihui 已提交
151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
  if (code != 0) {
    pError("failed to use db, code:%d reason:%s", taos_errno(pRes), taos_errstr(pRes));
    exit(1);
  }
  taos_free_result(pRes);

  sprintf(qstr, "show tables");
  pRes = taos_query(con, qstr);
  code = taos_errno(pRes);
  if (code != 0) {
    pError("failed to show tables, code:%d reason:%s", taos_errno(pRes), taos_errstr(pRes));
    exit(0);
  }

  int64_t totalTableNum = getResult(pRes);
  taos_free_result(pRes);

168
  pPrint("%s database: %s, total %" PRId64 " tables %s", GREEN, dbName, totalTableNum, NC);
L
lihui 已提交
169 170 171 172

  taos_close(con);
}

173 174
void *threadFunc(void *param) {
  SThreadInfo *pInfo = (SThreadInfo *)param;
P
plum-lihui 已提交
175
  char        *qstr = malloc(batchNumOfTbl * batchNumOfRow * 128);
S
Shengliang Guan 已提交
176
  int32_t      code = 0;
177 178 179

  TAOS *con = taos_connect(NULL, "root", "taosdata", NULL, 0);
  if (con == NULL) {
L
lihui 已提交
180
    pError("index:%d, failed to connect to DB, reason:%s", pInfo->threadIndex, taos_errstr(NULL));
181 182 183
    exit(1);
  }

184 185
  // printf("thread:%d, table range: %"PRId64 " - %"PRId64 "\n", pInfo->threadIndex, pInfo->tableBeginIndex,
  // pInfo->tableEndIndex);
186
  sprintf(qstr, "use %s", pInfo->dbName);
L
lihui 已提交
187 188 189
  TAOS_RES *pRes = taos_query(con, qstr);
  taos_free_result(pRes);

S
Shengliang Guan 已提交
190
  if (createTable) {
L
lihui 已提交
191 192 193
    int64_t curMs = 0;
    int64_t beginMs = taosGetTimestampMs();
    pInfo->startMs = beginMs;
194
    int64_t t = pInfo->tableBeginIndex;
L
lihui 已提交
195
    for (; t <= pInfo->tableEndIndex;) {
196 197
      // int64_t batch = (pInfo->tableEndIndex - t);
      // batch = MIN(batch, batchNum);
S
Shengliang Guan 已提交
198 199

      int32_t len = sprintf(qstr, "create table");
P
plum-lihui 已提交
200
      for (int32_t i = 0; i < batchNumOfTbl;) {
L
lihui 已提交
201
        len += sprintf(qstr + len, " %s_t%" PRId64 " using %s tags(%" PRId64 ")", stbName, t, stbName, t);
202 203
        t++;
        i++;
L
lihui 已提交
204
        if (t > pInfo->tableEndIndex) {
205 206
          break;
        }
S
Shengliang Guan 已提交
207
      }
S
Shengliang Guan 已提交
208

209
      int64_t   startTs = taosGetTimestampUs();
L
lihui 已提交
210 211
      TAOS_RES *pRes = taos_query(con, qstr);
      code = taos_errno(pRes);
P
plum-lihui 已提交
212
      if ((code != 0) && (code != TSDB_CODE_RPC_AUTH_REQUIRED)) {
P
plum-lihui 已提交
213
        pError("failed to create table t%" PRId64 ", reason:%s", t, tstrerror(code));
S
TD-1415  
Shengliang Guan 已提交
214
      }
L
lihui 已提交
215
      taos_free_result(pRes);
216 217 218 219
      int64_t endTs = taosGetTimestampUs();
      int64_t delay = endTs - startTs;
      // printf("==== %"PRId64" -  %"PRId64", %"PRId64"\n", startTs, endTs, delay);
      if (delay > pInfo->maxDelay) pInfo->maxDelay = delay;
L
lihui 已提交
220 221
      if (delay < pInfo->minDelay) pInfo->minDelay = delay;

222 223 224 225
      curMs = taosGetTimestampMs();
      if (curMs - beginMs > 10000) {
        beginMs = curMs;
        // printf("==== tableBeginIndex: %"PRId64", t: %"PRId64"\n", pInfo->tableBeginIndex, t);
S
Shengliang Guan 已提交
226 227
        printCreateProgress(pInfo, t);
      }
S
TD-1415  
Shengliang Guan 已提交
228
    }
L
lihui 已提交
229
    printCreateProgress(pInfo, t);
S
Shengliang Guan 已提交
230 231 232
  }

  if (insertData) {
L
lihui 已提交
233
    int64_t curMs = 0;
234
    int64_t beginMs = taosGetTimestampMs();
P
plum-lihui 已提交
235 236 237 238 239
    pInfo->startMs = beginMs;
    int64_t t = pInfo->tableBeginIndex;
    for (; t <= pInfo->tableEndIndex;) {
      // int64_t batch = (pInfo->tableEndIndex - t);
      // batch = MIN(batch, batchNum);
L
lihui 已提交
240

P
plum-lihui 已提交
241 242 243 244 245 246 247 248 249 250 251 252 253 254 255
      int32_t len = sprintf(qstr, "insert into ");
	  
      for (int32_t i = 0; i < batchNumOfTbl;) {
		int64_t ts = startTimestamp;	  	
        len += sprintf(qstr + len, "%s_t%" PRId64 " values ", stbName, t);
        for (int32_t j = 0; j < batchNumOfRow; j++) {
		  len += sprintf(qstr + len, "(%" PRId64 ", 6666) ", ts++);
        }
		
        t++;
        i++;
        if (t > pInfo->tableEndIndex) {
          break;
        }
      }	  
S
Shengliang Guan 已提交
256

P
plum-lihui 已提交
257
      int64_t   startTs = taosGetTimestampUs();
L
lihui 已提交
258 259
      TAOS_RES *pRes = taos_query(con, qstr);
      code = taos_errno(pRes);
P
plum-lihui 已提交
260 261
      if ((code != 0) && (code != TSDB_CODE_RPC_AUTH_REQUIRED)) {
        pError("failed to insert %s_t%" PRId64 ", reason:%s", stbName, t, tstrerror(code));
S
TD-1415  
Shengliang Guan 已提交
262
      }
L
lihui 已提交
263
      taos_free_result(pRes);
P
plum-lihui 已提交
264 265 266 267 268
      int64_t endTs = taosGetTimestampUs();
      int64_t delay = endTs - startTs;
      // printf("==== %"PRId64" -  %"PRId64", %"PRId64"\n", startTs, endTs, delay);
      if (delay > pInfo->maxDelay) pInfo->maxDelay = delay;
      if (delay < pInfo->minDelay) pInfo->minDelay = delay;
S
Shengliang Guan 已提交
269

L
lihui 已提交
270
      curMs = taosGetTimestampMs();
271
      if (curMs - beginMs > 10000) {
P
plum-lihui 已提交
272 273
        beginMs = curMs;
        // printf("==== tableBeginIndex: %"PRId64", t: %"PRId64"\n", pInfo->tableBeginIndex, t);
S
Shengliang Guan 已提交
274 275
        printInsertProgress(pInfo, t);
      }
276
    }
P
plum-lihui 已提交
277
    printInsertProgress(pInfo, t);
278 279 280
  }

  taos_close(con);
S
Shengliang Guan 已提交
281
  free(qstr);
282 283 284 285 286 287 288 289 290 291 292 293
  return 0;
}

void printHelp() {
  char indent[10] = "        ";
  printf("Used to test the performance while create table\n");

  printf("%s%s\n", indent, "-c");
  printf("%s%s%s%s\n", indent, indent, "Configuration directory, default is ", configDir);
  printf("%s%s\n", indent, "-d");
  printf("%s%s%s%s\n", indent, indent, "The name of the database to be created, default is ", dbName);
  printf("%s%s\n", indent, "-s");
S
Shengliang Guan 已提交
294
  printf("%s%s%s%s\n", indent, indent, "The name of the super table to be created, default is ", stbName);
295 296 297
  printf("%s%s\n", indent, "-t");
  printf("%s%s%s%d\n", indent, indent, "numOfThreads, default is ", numOfThreads);
  printf("%s%s\n", indent, "-n");
S
Shengliang Guan 已提交
298
  printf("%s%s%s%" PRId64 "\n", indent, indent, "numOfTables, default is ", numOfTables);
S
Shengliang Guan 已提交
299 300 301 302 303 304 305
  printf("%s%s\n", indent, "-v");
  printf("%s%s%s%d\n", indent, indent, "numOfVgroups, default is ", numOfVgroups);
  printf("%s%s\n", indent, "-a");
  printf("%s%s%s%d\n", indent, indent, "createTable, default is ", createTable);
  printf("%s%s\n", indent, "-i");
  printf("%s%s%s%d\n", indent, indent, "insertData, default is ", insertData);
  printf("%s%s\n", indent, "-b");
P
plum-lihui 已提交
306
  printf("%s%s%s%d\n", indent, indent, "batchNumOfTbl, default is ", batchNumOfTbl);
L
lihui 已提交
307 308
  printf("%s%s\n", indent, "-w");
  printf("%s%s%s%d\n", indent, indent, "showTablesFlag, default is ", showTablesFlag);
P
plum-lihui 已提交
309 310 311 312
  printf("%s%s\n", indent, "-q");
  printf("%s%s%s%d\n", indent, indent, "queryFlag, default is ", queryFlag);
  printf("%s%s\n", indent, "-l");
  printf("%s%s%s%d\n", indent, indent, "batchNumOfRow, default is ", batchNumOfRow);
S
Shengliang Guan 已提交
313

314 315 316
  exit(EXIT_SUCCESS);
}

S
Shengliang Guan 已提交
317 318
void parseArgument(int32_t argc, char *argv[]) {
  for (int32_t i = 1; i < argc; i++) {
319 320 321 322 323 324 325 326
    if (strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--help") == 0) {
      printHelp();
      exit(0);
    } else if (strcmp(argv[i], "-d") == 0) {
      strcpy(dbName, argv[++i]);
    } else if (strcmp(argv[i], "-c") == 0) {
      strcpy(configDir, argv[++i]);
    } else if (strcmp(argv[i], "-s") == 0) {
S
Shengliang Guan 已提交
327
      strcpy(stbName, argv[++i]);
328 329 330
    } else if (strcmp(argv[i], "-t") == 0) {
      numOfThreads = atoi(argv[++i]);
    } else if (strcmp(argv[i], "-n") == 0) {
S
Shengliang Guan 已提交
331
      numOfTables = atoll(argv[++i]);
L
lihui 已提交
332
    } else if (strcmp(argv[i], "-v") == 0) {
S
Shengliang Guan 已提交
333 334 335 336 337 338
      numOfVgroups = atoi(argv[++i]);
    } else if (strcmp(argv[i], "-a") == 0) {
      createTable = atoi(argv[++i]);
    } else if (strcmp(argv[i], "-i") == 0) {
      insertData = atoi(argv[++i]);
    } else if (strcmp(argv[i], "-b") == 0) {
P
plum-lihui 已提交
339 340 341
      batchNumOfTbl = atoi(argv[++i]);
    } else if (strcmp(argv[i], "-l") == 0) {
      batchNumOfRow = atoi(argv[++i]);
L
lihui 已提交
342 343
    } else if (strcmp(argv[i], "-w") == 0) {
      showTablesFlag = atoi(argv[++i]);
P
plum-lihui 已提交
344 345
    } else if (strcmp(argv[i], "-q") == 0) {
      queryFlag = atoi(argv[++i]);
346
    } else {
P
plum-lihui 已提交
347
	  pPrint("%s unknow para: %s %s", GREEN, argv[++i], NC);
348 349 350 351
    }
  }

  pPrint("%s dbName:%s %s", GREEN, dbName, NC);
S
Shengliang Guan 已提交
352
  pPrint("%s stbName:%s %s", GREEN, stbName, NC);
353
  pPrint("%s configDir:%s %s", GREEN, configDir, NC);
S
Shengliang Guan 已提交
354
  pPrint("%s numOfTables:%" PRId64 " %s", GREEN, numOfTables, NC);
355
  pPrint("%s numOfThreads:%d %s", GREEN, numOfThreads, NC);
S
Shengliang Guan 已提交
356 357 358
  pPrint("%s numOfVgroups:%d %s", GREEN, numOfVgroups, NC);
  pPrint("%s createTable:%d %s", GREEN, createTable, NC);
  pPrint("%s insertData:%d %s", GREEN, insertData, NC);
P
plum-lihui 已提交
359 360
  pPrint("%s batchNumOfTbl:%d %s", GREEN, batchNumOfTbl, NC);
  pPrint("%s batchNumOfRow:%d %s", GREEN, batchNumOfRow, NC);
L
lihui 已提交
361
  pPrint("%s showTablesFlag:%d %s", GREEN, showTablesFlag, NC);
P
plum-lihui 已提交
362
  pPrint("%s queryFlag:%d %s", GREEN, queryFlag, NC);
S
Shengliang Guan 已提交
363

364 365
  pPrint("%s start create table performace test %s", GREEN, NC);
}
L
lihui 已提交
366 367 368 369 370

int32_t main(int32_t argc, char *argv[]) {
  parseArgument(argc, argv);

  if (showTablesFlag) {
371 372
    showTables();
    return 0;
L
lihui 已提交
373
  }
374

P
plum-lihui 已提交
375 376 377 378
  if (queryFlag) {
    //selectRowsFromTable();
    return 0;
  }
L
lihui 已提交
379

P
plum-lihui 已提交
380 381 382 383
  if (createTable) {
    createDbAndStb();
  }
  
384
  pPrint("%d threads are spawned to create %" PRId64 " tables", numOfThreads, numOfTables);
L
lihui 已提交
385 386 387 388 389 390

  pthread_attr_t thattr;
  pthread_attr_init(&thattr);
  pthread_attr_setdetachstate(&thattr, PTHREAD_CREATE_JOINABLE);
  SThreadInfo *pInfo = (SThreadInfo *)calloc(numOfThreads, sizeof(SThreadInfo));

391 392
  // int64_t numOfTablesPerThread = numOfTables / numOfThreads;
  // numOfTables = numOfTablesPerThread * numOfThreads;
L
lihui 已提交
393 394 395 396

  if (numOfThreads < 1) {
    numOfThreads = 1;
  }
397

L
lihui 已提交
398 399
  int64_t a = numOfTables / numOfThreads;
  if (a < 1) {
400 401
    numOfThreads = numOfTables;
    a = 1;
L
lihui 已提交
402
  }
403

L
lihui 已提交
404 405 406 407 408 409 410 411 412
  int64_t b = 0;
  b = numOfTables % numOfThreads;

  int64_t tableFrom = 0;
  for (int32_t i = 0; i < numOfThreads; ++i) {
    pInfo[i].tableBeginIndex = tableFrom;
    pInfo[i].tableEndIndex = i < b ? tableFrom + a : tableFrom + a - 1;
    tableFrom = pInfo[i].tableEndIndex + 1;
    pInfo[i].threadIndex = i;
413
    pInfo[i].minDelay = INT64_MAX;
L
lihui 已提交
414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431
    strcpy(pInfo[i].dbName, dbName);
    strcpy(pInfo[i].stbName, stbName);
    pthread_create(&(pInfo[i].thread), &thattr, threadFunc, (void *)(pInfo + i));
  }

  taosMsleep(300);
  for (int32_t i = 0; i < numOfThreads; i++) {
    pthread_join(pInfo[i].thread, NULL);
  }

  int64_t maxDelay = 0;
  int64_t minDelay = INT64_MAX;

  float createTableSpeed = 0;
  for (int32_t i = 0; i < numOfThreads; ++i) {
    createTableSpeed += pInfo[i].createTableSpeed;

    if (pInfo[i].maxDelay > maxDelay) maxDelay = pInfo[i].maxDelay;
432
    if (pInfo[i].minDelay < minDelay) minDelay = pInfo[i].minDelay;
L
lihui 已提交
433 434 435 436 437 438 439
  }

  float insertDataSpeed = 0;
  for (int32_t i = 0; i < numOfThreads; ++i) {
    insertDataSpeed += pInfo[i].insertDataSpeed;
  }

P
plum-lihui 已提交
440 441
  if (createTable) {
    pPrint("%s total %" PRId64 " tables, %.1f tables/second, threads:%d, maxDelay: %" PRId64 "us, minDelay: %" PRId64
442 443
         "us %s",
         GREEN, numOfTables, createTableSpeed, numOfThreads, maxDelay, minDelay, NC);
P
plum-lihui 已提交
444
  }
L
lihui 已提交
445 446

  if (insertData) {
447 448
    pPrint("%s total %" PRId64 " tables, %.1f rows/second, threads:%d %s", GREEN, numOfTables, insertDataSpeed,
           numOfThreads, NC);
L
lihui 已提交
449 450 451 452 453
  }

  pthread_attr_destroy(&thattr);
  free(pInfo);
}