1341 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			1341 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * 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/>.
 | |
|  */
 | |
| 
 | |
| #include <fstream>
 | |
| 
 | |
| #include "parTestUtil.h"
 | |
| 
 | |
| using namespace std;
 | |
| 
 | |
| namespace ParserTest {
 | |
| 
 | |
| class ParserInitialCTest : public ParserDdlTest {};
 | |
| 
 | |
| /*
 | |
|  * COMPACT DATABASE db_name [START WITH start_time] [END WITH END_time]
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, compact) {
 | |
|   SCompactDbReq expect = {0};
 | |
| 
 | |
|   auto setCompactDbReq = [&](const char* pDb, int64_t start = INT64_MIN, int64_t end = INT64_MAX) {
 | |
|     snprintf(expect.db, sizeof(expect.db), "0.%s", pDb);
 | |
|     expect.timeRange.skey = start;
 | |
|     expect.timeRange.ekey = end;
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_COMPACT_DATABASE_STMT);
 | |
|     ASSERT_EQ(pQuery->pCmdMsg->msgType, TDMT_MND_COMPACT_DB);
 | |
|     SCompactDbReq req = {0};
 | |
|     ASSERT_EQ(tDeserializeSCompactDbReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req), TSDB_CODE_SUCCESS);
 | |
|     ASSERT_EQ(std::string(req.db), std::string(expect.db));
 | |
|     ASSERT_EQ(req.timeRange.skey, expect.timeRange.skey);
 | |
|     ASSERT_EQ(req.timeRange.ekey, expect.timeRange.ekey);
 | |
|   });
 | |
| 
 | |
|   setCompactDbReq("test");
 | |
|   run("COMPACT DATABASE test");
 | |
| 
 | |
|   setCompactDbReq("test", 1678168883000, 1678255283000);
 | |
|   run("COMPACT DATABASE test START WITH '2023-03-07 14:01:23' END WITH '2023-03-08 14:01:23'");
 | |
| 
 | |
|   setCompactDbReq("testus", 1673071283000000000);
 | |
|   run("COMPACT DATABASE testus START WITH TIMESTAMP '2023-01-07 14:01:23'");
 | |
| 
 | |
|   setCompactDbReq("testus", INT64_MIN, 1675749683000000000);
 | |
|   run("COMPACT DATABASE testus END WITH 1675749683000000000");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE ACCOUNT account_name PASS value [create_account_options]
 | |
|  *
 | |
|  * create_account_options:
 | |
|  *     create_account_option ...
 | |
|  *
 | |
|  * create_account_option: {
 | |
|  *     PPS value
 | |
|  *   | TSERIES value
 | |
|  *   | STORAGE value
 | |
|  *   | STREAMS value
 | |
|  *   | QTIME value
 | |
|  *   | DBS value
 | |
|  *   | USERS value
 | |
|  *   | CONNS value
 | |
|  *   | STATE value
 | |
|  * }
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createAccount) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   run("CREATE ACCOUNT ac_wxy PASS '123456'", TSDB_CODE_PAR_EXPRIE_STATEMENT, PARSER_STAGE_PARSE);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE BNODE ON DNODE dnode_id
 | |
|  * the server does not support it temporarily
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * CREATE DATABASE [IF NOT EXISTS] db_name [database_options]
 | |
|  *
 | |
|  * database_options:
 | |
|  *     database_option ...
 | |
|  *
 | |
|  * database_option: {
 | |
|  *     BUFFER value
 | |
|  *   | CACHEMODEL {'none' | 'last_row' | 'last_value' | 'both'}
 | |
|  *   | CACHESIZE value
 | |
|  *   | COMP {0 | 1 | 2}
 | |
|  *   | DURATION value
 | |
|  *   | WAL_FSYNC_PERIOD value
 | |
|  *   | MAXROWS value
 | |
|  *   | MINROWS value
 | |
|  *   | KEEP value
 | |
|  *   | PAGES value
 | |
|  *   | PAGESIZE  value
 | |
|  *   | PRECISION {'ms' | 'us' | 'ns'}
 | |
|  *   | REPLICA value
 | |
|  *   | RETENTIONS ingestion_duration:keep_duration ...
 | |
|  *   | STRICT {'off' | 'on'}  // not support
 | |
|  *   | WAL_LEVEL value
 | |
|  *   | VGROUPS value
 | |
|  *   | SINGLE_STABLE {0 | 1}
 | |
|  *   | WAL_RETENTION_PERIOD value
 | |
|  *   | WAL_ROLL_PERIOD value
 | |
|  *   | WAL_RETENTION_SIZE value
 | |
|  *   | WAL_SEGMENT_SIZE value
 | |
|  * }
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createDatabase) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SCreateDbReq expect = {0};
 | |
| 
 | |
|   auto clearCreateDbReq = [&]() {
 | |
|     tFreeSCreateDbReq(&expect);
 | |
|     memset(&expect, 0, sizeof(SCreateDbReq));
 | |
|   };
 | |
| 
 | |
|   auto setCreateDbReq = [&](const char* pDbname, int8_t igExists = 0) {
 | |
|     int32_t len = snprintf(expect.db, sizeof(expect.db), "0.%s", pDbname);
 | |
|     expect.db[len] = '\0';
 | |
|     expect.ignoreExist = igExists;
 | |
|     expect.buffer = TSDB_DEFAULT_BUFFER_PER_VNODE;
 | |
|     expect.cacheLast = TSDB_DEFAULT_CACHE_MODEL;
 | |
|     expect.cacheLastSize = TSDB_DEFAULT_CACHE_SIZE;
 | |
|     expect.compression = TSDB_DEFAULT_COMP_LEVEL;
 | |
|     expect.daysPerFile = TSDB_DEFAULT_DAYS_PER_FILE;
 | |
|     expect.walFsyncPeriod = TSDB_DEFAULT_FSYNC_PERIOD;
 | |
|     expect.maxRows = TSDB_DEFAULT_MAXROWS_FBLOCK;
 | |
|     expect.minRows = TSDB_DEFAULT_MINROWS_FBLOCK;
 | |
|     expect.daysToKeep0 = TSDB_DEFAULT_KEEP;
 | |
|     expect.daysToKeep1 = TSDB_DEFAULT_KEEP;
 | |
|     expect.daysToKeep2 = TSDB_DEFAULT_KEEP;
 | |
|     expect.pages = TSDB_DEFAULT_PAGES_PER_VNODE;
 | |
|     expect.pageSize = TSDB_DEFAULT_PAGESIZE_PER_VNODE;
 | |
|     expect.precision = TSDB_DEFAULT_PRECISION;
 | |
|     expect.replications = TSDB_DEFAULT_DB_REPLICA;
 | |
|     expect.strict = TSDB_DEFAULT_DB_STRICT;
 | |
|     expect.walLevel = TSDB_DEFAULT_WAL_LEVEL;
 | |
|     expect.numOfVgroups = TSDB_DEFAULT_VN_PER_DB;
 | |
|     expect.numOfStables = TSDB_DEFAULT_DB_SINGLE_STABLE;
 | |
|     expect.schemaless = TSDB_DEFAULT_DB_SCHEMALESS;
 | |
|     expect.walRetentionPeriod = TSDB_REP_DEF_DB_WAL_RET_PERIOD;
 | |
|     expect.walRetentionSize = TSDB_REP_DEF_DB_WAL_RET_SIZE;
 | |
|     expect.walRollPeriod = TSDB_REP_DEF_DB_WAL_ROLL_PERIOD;
 | |
|     expect.walSegmentSize = TSDB_DEFAULT_DB_WAL_SEGMENT_SIZE;
 | |
|     expect.sstTrigger = TSDB_DEFAULT_SST_TRIGGER;
 | |
|     expect.hashPrefix = TSDB_DEFAULT_HASH_PREFIX;
 | |
|     expect.hashSuffix = TSDB_DEFAULT_HASH_SUFFIX;
 | |
|     expect.tsdbPageSize = TSDB_DEFAULT_TSDB_PAGESIZE;
 | |
|   };
 | |
| 
 | |
|   auto setDbBuffer = [&](int32_t buffer) { expect.buffer = buffer; };
 | |
|   auto setDbCachelast = [&](int8_t cachelast) { expect.cacheLast = cachelast; };
 | |
|   auto setDbCachelastSize = [&](int8_t cachelastSize) { expect.cacheLastSize = cachelastSize; };
 | |
|   auto setDbCompression = [&](int8_t compressionLevel) { expect.compression = compressionLevel; };
 | |
|   auto setDbDays = [&](int32_t daysPerFile) { expect.daysPerFile = daysPerFile; };
 | |
|   auto setDbFsync = [&](int32_t fsyncPeriod) { expect.walFsyncPeriod = fsyncPeriod; };
 | |
|   auto setDbMaxRows = [&](int32_t maxRowsPerBlock) { expect.maxRows = maxRowsPerBlock; };
 | |
|   auto setDbMinRows = [&](int32_t minRowsPerBlock) { expect.minRows = minRowsPerBlock; };
 | |
|   auto setDbKeep = [&](int32_t keep0, int32_t keep1 = 0, int32_t keep2 = 0) {
 | |
|     expect.daysToKeep0 = keep0;
 | |
|     expect.daysToKeep1 = 0 == keep1 ? expect.daysToKeep0 : keep1;
 | |
|     expect.daysToKeep2 = 0 == keep2 ? expect.daysToKeep1 : keep2;
 | |
|   };
 | |
|   auto setDbPages = [&](int32_t pages) { expect.pages = pages; };
 | |
|   auto setDbPageSize = [&](int32_t pagesize) { expect.pageSize = pagesize; };
 | |
|   auto setDbPrecision = [&](int8_t precision) { expect.precision = precision; };
 | |
|   auto setDbReplica = [&](int8_t replica) { expect.replications = replica; };
 | |
|   auto setDbStricta = [&](int8_t strict) { expect.strict = strict; };
 | |
|   auto setDbWalLevel = [&](int8_t walLevel) { expect.walLevel = walLevel; };
 | |
|   auto setDbVgroups = [&](int32_t numOfVgroups) { expect.numOfVgroups = numOfVgroups; };
 | |
|   auto setDbSingleStable = [&](int8_t singleStable) { expect.numOfStables = singleStable; };
 | |
|   auto addDbRetention = [&](int64_t freq, int64_t keep, int8_t freqUnit, int8_t keepUnit) {
 | |
|     SRetention retention = {0};
 | |
|     retention.freq = freq;
 | |
|     retention.keep = keep;
 | |
|     retention.freqUnit = freqUnit;
 | |
|     retention.keepUnit = keepUnit;
 | |
|     if (NULL == expect.pRetensions) {
 | |
|       expect.pRetensions = taosArrayInit(TARRAY_MIN_SIZE, sizeof(SRetention));
 | |
|     }
 | |
|     taosArrayPush(expect.pRetensions, &retention);
 | |
|     ++expect.numOfRetensions;
 | |
|   };
 | |
|   auto setDbSchemaless = [&](int8_t schemaless) { expect.schemaless = schemaless; };
 | |
|   auto setDbWalRetentionPeriod = [&](int32_t walRetentionPeriod) { expect.walRetentionPeriod = walRetentionPeriod; };
 | |
|   auto setDbWalRetentionSize = [&](int32_t walRetentionSize) { expect.walRetentionSize = walRetentionSize; };
 | |
|   auto setDbWalRollPeriod = [&](int32_t walRollPeriod) { expect.walRollPeriod = walRollPeriod; };
 | |
|   auto setDbWalSegmentSize = [&](int32_t walSegmentSize) { expect.walSegmentSize = walSegmentSize; };
 | |
|   auto setDbSstTrigger = [&](int32_t sstTrigger) { expect.sstTrigger = sstTrigger; };
 | |
|   auto setDbHashPrefix = [&](int32_t hashPrefix) { expect.hashPrefix = hashPrefix; };
 | |
|   auto setDbHashSuffix = [&](int32_t hashSuffix) { expect.hashSuffix = hashSuffix; };
 | |
|   auto setDbTsdbPageSize = [&](int32_t tsdbPageSize) { expect.tsdbPageSize = tsdbPageSize; };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_DATABASE_STMT);
 | |
|     ASSERT_EQ(pQuery->pCmdMsg->msgType, TDMT_MND_CREATE_DB);
 | |
|     SCreateDbReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS == tDeserializeSCreateDbReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.db), std::string(expect.db));
 | |
|     ASSERT_EQ(req.numOfVgroups, expect.numOfVgroups);
 | |
|     ASSERT_EQ(req.numOfStables, expect.numOfStables);
 | |
|     ASSERT_EQ(req.buffer, expect.buffer);
 | |
|     ASSERT_EQ(req.pageSize, expect.pageSize);
 | |
|     ASSERT_EQ(req.pages, expect.pages);
 | |
|     ASSERT_EQ(req.daysPerFile, expect.daysPerFile);
 | |
|     ASSERT_EQ(req.daysToKeep0, expect.daysToKeep0);
 | |
|     ASSERT_EQ(req.daysToKeep1, expect.daysToKeep1);
 | |
|     ASSERT_EQ(req.daysToKeep2, expect.daysToKeep2);
 | |
|     ASSERT_EQ(req.minRows, expect.minRows);
 | |
|     ASSERT_EQ(req.maxRows, expect.maxRows);
 | |
|     ASSERT_EQ(req.walFsyncPeriod, expect.walFsyncPeriod);
 | |
|     ASSERT_EQ(req.walLevel, expect.walLevel);
 | |
|     ASSERT_EQ(req.precision, expect.precision);
 | |
|     ASSERT_EQ(req.compression, expect.compression);
 | |
|     ASSERT_EQ(req.replications, expect.replications);
 | |
|     ASSERT_EQ(req.strict, expect.strict);
 | |
|     ASSERT_EQ(req.cacheLast, expect.cacheLast);
 | |
|     ASSERT_EQ(req.cacheLastSize, expect.cacheLastSize);
 | |
|     ASSERT_EQ(req.walRetentionPeriod, expect.walRetentionPeriod);
 | |
|     ASSERT_EQ(req.walRetentionSize, expect.walRetentionSize);
 | |
|     ASSERT_EQ(req.walRollPeriod, expect.walRollPeriod);
 | |
|     ASSERT_EQ(req.walSegmentSize, expect.walSegmentSize);
 | |
|     ASSERT_EQ(req.sstTrigger, expect.sstTrigger);
 | |
|     ASSERT_EQ(req.hashPrefix, expect.hashPrefix);
 | |
|     ASSERT_EQ(req.hashSuffix, expect.hashSuffix);
 | |
|     ASSERT_EQ(req.tsdbPageSize, expect.tsdbPageSize);
 | |
|     ASSERT_EQ(req.ignoreExist, expect.ignoreExist);
 | |
|     ASSERT_EQ(req.numOfRetensions, expect.numOfRetensions);
 | |
|     if (expect.numOfRetensions > 0) {
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pRetensions), expect.numOfRetensions);
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pRetensions), taosArrayGetSize(expect.pRetensions));
 | |
|       for (int32_t i = 0; i < expect.numOfRetensions; ++i) {
 | |
|         SRetention* pReten = (SRetention*)taosArrayGet(req.pRetensions, i);
 | |
|         SRetention* pExpectReten = (SRetention*)taosArrayGet(expect.pRetensions, i);
 | |
|         if(i == 0) {
 | |
|           ASSERT_EQ(pReten->freq, 0);
 | |
|         } else {
 | |
|           ASSERT_EQ(pReten->freq, pExpectReten->freq);
 | |
|         }
 | |
|         ASSERT_EQ(pReten->keep, pExpectReten->keep);
 | |
|         ASSERT_EQ(pReten->freqUnit, pExpectReten->freqUnit);
 | |
|         ASSERT_EQ(pReten->keepUnit, pExpectReten->keepUnit);
 | |
|       }
 | |
|     }
 | |
|     tFreeSCreateDbReq(&req);
 | |
|   });
 | |
| 
 | |
|   setCreateDbReq("wxy_db");
 | |
|   run("CREATE DATABASE wxy_db");
 | |
|   clearCreateDbReq();
 | |
| 
 | |
|   setCreateDbReq("wxy_db", 1);
 | |
|   setDbBuffer(64);
 | |
|   setDbCachelast(2);
 | |
|   setDbCachelastSize(20);
 | |
|   setDbCompression(1);
 | |
|   setDbDays(100 * 1440);
 | |
|   setDbFsync(100);
 | |
|   setDbMaxRows(1000);
 | |
|   setDbMinRows(100);
 | |
|   setDbKeep(1440 * 1440);
 | |
|   setDbPages(96);
 | |
|   setDbPageSize(8);
 | |
|   setDbPrecision(TSDB_TIME_PRECISION_NANO);
 | |
|   setDbReplica(3);
 | |
|   addDbRetention(15 * MILLISECOND_PER_SECOND, 7 * MILLISECOND_PER_DAY, TIME_UNIT_SECOND, TIME_UNIT_DAY);
 | |
|   addDbRetention(1 * MILLISECOND_PER_MINUTE, 21 * MILLISECOND_PER_DAY, TIME_UNIT_MINUTE, TIME_UNIT_DAY);
 | |
|   addDbRetention(15 * MILLISECOND_PER_MINUTE, 500 * MILLISECOND_PER_DAY, TIME_UNIT_MINUTE, TIME_UNIT_DAY);
 | |
|   // setDbStricta(1);
 | |
|   setDbWalLevel(2);
 | |
|   setDbVgroups(100);
 | |
|   setDbSingleStable(1);
 | |
|   setDbSchemaless(1);
 | |
|   setDbWalRetentionPeriod(-1);
 | |
|   setDbWalRetentionSize(-1);
 | |
|   setDbWalRollPeriod(10);
 | |
|   setDbWalSegmentSize(20);
 | |
|   setDbSstTrigger(16);
 | |
|   setDbHashPrefix(3);
 | |
|   setDbHashSuffix(4);
 | |
|   setDbTsdbPageSize(32);
 | |
|   run("CREATE DATABASE IF NOT EXISTS wxy_db "
 | |
|       "BUFFER 64 "
 | |
|       "CACHEMODEL 'last_value' "
 | |
|       "CACHESIZE 20 "
 | |
|       "COMP 1 "
 | |
|       "DURATION 100 "
 | |
|       "WAL_FSYNC_PERIOD 100 "
 | |
|       "MAXROWS 1000 "
 | |
|       "MINROWS 100 "
 | |
|       "KEEP 1440 "
 | |
|       "PAGES 96 "
 | |
|       "PAGESIZE 8 "
 | |
|       "PRECISION 'ns' "
 | |
|       "REPLICA 3 "
 | |
|       "RETENTIONS -:7d,1m:21d,15m:500d "
 | |
|       //      "STRICT 'on' "
 | |
|       "WAL_LEVEL 2 "
 | |
|       "VGROUPS 100 "
 | |
|       "SINGLE_STABLE 1 "
 | |
|       "SCHEMALESS 1 "
 | |
|       "WAL_RETENTION_PERIOD -1 "
 | |
|       "WAL_RETENTION_SIZE -1 "
 | |
|       "WAL_ROLL_PERIOD 10 "
 | |
|       "WAL_SEGMENT_SIZE 20 "
 | |
|       "STT_TRIGGER 16 "
 | |
|       "TABLE_PREFIX 3 "
 | |
|       "TABLE_SUFFIX 4 "
 | |
|       "TSDB_PAGESIZE 32");
 | |
|   clearCreateDbReq();
 | |
| 
 | |
|   setCreateDbReq("wxy_db", 1);
 | |
|   setDbDays(100);
 | |
|   setDbKeep(1440, 300 * 60, 400 * 1440);
 | |
|   run("CREATE DATABASE IF NOT EXISTS wxy_db "
 | |
|       "DURATION 100m "
 | |
|       "KEEP 1440m,300h,400d ");
 | |
|   clearCreateDbReq();
 | |
| 
 | |
|   setCreateDbReq("wxy_db", 1);
 | |
|   setDbReplica(3);
 | |
|   setDbWalRetentionPeriod(TSDB_REPS_DEF_DB_WAL_RET_PERIOD);
 | |
|   setDbWalRetentionSize(TSDB_REPS_DEF_DB_WAL_RET_SIZE);
 | |
|   setDbWalRollPeriod(TSDB_REPS_DEF_DB_WAL_ROLL_PERIOD);
 | |
|   run("CREATE DATABASE IF NOT EXISTS wxy_db REPLICA 3");
 | |
|   clearCreateDbReq();
 | |
| }
 | |
| 
 | |
| TEST_F(ParserInitialCTest, createDatabaseSemanticCheck) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   run("create database db2 retentions -:1d,0s:1d", TSDB_CODE_PAR_INVALID_DB_OPTION);
 | |
|   run("create database db2 retentions -:0d", TSDB_CODE_PAR_INVALID_DB_OPTION);
 | |
|   run("create database db2 retentions -:1d,1w:1d", TSDB_CODE_PAR_INVALID_DB_OPTION);
 | |
|   run("create database db2 retentions -:1n,1w:1d", TSDB_CODE_PAR_INVALID_DB_OPTION);
 | |
|   run("create database db2 retentions -:7d,15m:21d,10m:500d", TSDB_CODE_PAR_INVALID_DB_OPTION);
 | |
|   run("create database db2 retentions -:7d,5m:21d,10m:10d", TSDB_CODE_PAR_INVALID_DB_OPTION);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE DNODE {dnode_endpoint | dnode_host_name PORT port_val}
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createDnode) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SCreateDnodeReq expect = {0};
 | |
| 
 | |
|   auto clearCreateDnodeReq = [&]() { memset(&expect, 0, sizeof(SCreateDnodeReq)); };
 | |
| 
 | |
|   auto setCreateDnodeReq = [&](const char* pFqdn, int32_t port = tsServerPort) {
 | |
|     strcpy(expect.fqdn, pFqdn);
 | |
|     expect.port = port;
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_DNODE_STMT);
 | |
|     SCreateDnodeReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS == tDeserializeSCreateDnodeReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.fqdn), std::string(expect.fqdn));
 | |
|     ASSERT_EQ(req.port, expect.port);
 | |
|   });
 | |
| 
 | |
|   setCreateDnodeReq("abc1", 7030);
 | |
|   run("CREATE DNODE 'abc1' PORT 7030");
 | |
|   clearCreateDnodeReq();
 | |
| 
 | |
|   setCreateDnodeReq("1.1.1.1", 8030);
 | |
|   run("CREATE DNODE 1.1.1.1 PORT 8030");
 | |
|   clearCreateDnodeReq();
 | |
| 
 | |
|   setCreateDnodeReq("host1", 9030);
 | |
|   run("CREATE DNODE host1 PORT 9030");
 | |
|   clearCreateDnodeReq();
 | |
| 
 | |
|   setCreateDnodeReq("abc2", 7040);
 | |
|   run("CREATE DNODE 'abc2:7040'");
 | |
|   clearCreateDnodeReq();
 | |
| 
 | |
|   setCreateDnodeReq("1.1.1.2");
 | |
|   run("CREATE DNODE 1.1.1.2");
 | |
|   clearCreateDnodeReq();
 | |
| 
 | |
|   setCreateDnodeReq("host2");
 | |
|   run("CREATE DNODE host2");
 | |
|   clearCreateDnodeReq();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE [OR REPLACE] [AGGREGATE] FUNCTION [IF NOT EXISTS] func_name
 | |
|  *   AS library_path OUTPUTTYPE type_name [BUFSIZE value] [LANGUAGE value]
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createFunction) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SCreateFuncReq expect = {0};
 | |
| 
 | |
|   auto setCreateFuncReq = [&](const char* pUdfName, int8_t outputType, int32_t outputBytes = 0,
 | |
|                               int8_t funcType = TSDB_FUNC_TYPE_SCALAR, int8_t igExists = 0, int32_t bufSize = 0,
 | |
|                               int8_t language = TSDB_FUNC_SCRIPT_BIN_LIB, int8_t orReplace = 0) {
 | |
|     memset(&expect, 0, sizeof(SCreateFuncReq));
 | |
|     strcpy(expect.name, pUdfName);
 | |
|     expect.orReplace = orReplace;
 | |
|     expect.igExists = igExists;
 | |
|     expect.funcType = funcType;
 | |
|     expect.scriptType = language;
 | |
|     expect.outputType = outputType;
 | |
|     expect.outputLen = outputBytes > 0 ? outputBytes : tDataTypes[outputType].bytes;
 | |
|     expect.bufSize = bufSize;
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_FUNCTION_STMT);
 | |
|     SCreateFuncReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS == tDeserializeSCreateFuncReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.name), std::string(expect.name));
 | |
|     ASSERT_EQ(req.orReplace, expect.orReplace);
 | |
|     ASSERT_EQ(req.igExists, expect.igExists);
 | |
|     ASSERT_EQ(req.funcType, expect.funcType);
 | |
|     ASSERT_EQ(req.scriptType, expect.scriptType);
 | |
|     ASSERT_EQ(req.outputType, expect.outputType);
 | |
|     ASSERT_EQ(req.outputLen, expect.outputLen);
 | |
|     ASSERT_EQ(req.bufSize, expect.bufSize);
 | |
| 
 | |
|     tFreeSCreateFuncReq(&req);
 | |
|   });
 | |
| 
 | |
|   struct udfFile {
 | |
|     udfFile(const std::string& filename) : path_(filename) {
 | |
|       std::ofstream file(filename, std::ios::binary);
 | |
|       file << 123 << "abc" << '\n';
 | |
|       file.close();
 | |
|     }
 | |
|     ~udfFile() { remove(path_.c_str()); }
 | |
|     std::string path_;
 | |
|   } udffile("udf");
 | |
| 
 | |
|   setCreateFuncReq("udf1", TSDB_DATA_TYPE_INT);
 | |
|   run("CREATE FUNCTION udf1 AS 'udf' OUTPUTTYPE INT");
 | |
| 
 | |
|   setCreateFuncReq("udf2", TSDB_DATA_TYPE_DOUBLE, 0, TSDB_FUNC_TYPE_AGGREGATE, 1, 8, TSDB_FUNC_SCRIPT_PYTHON, 1);
 | |
|   run("CREATE OR REPLACE AGGREGATE FUNCTION IF NOT EXISTS udf2 AS 'udf' OUTPUTTYPE DOUBLE BUFSIZE 8 LANGUAGE 'python'");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE [ OR REPLACE ] VIEW name [ ( column_name [, ...] ) ] AS query
 | |
|  *
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createView) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SCMCreateStreamReq expect = {0};
 | |
| 
 | |
|   auto clearCreateStreamReq = [&]() {
 | |
|     tFreeSCMCreateStreamReq(&expect);
 | |
|     memset(&expect, 0, sizeof(SCMCreateStreamReq));
 | |
|   };
 | |
| 
 | |
|   auto setCreateStreamReq = [&](const char* pStream, const char* pSrcDb, const char* pSql, const char* pDstStb,
 | |
|                                 int8_t igExists = 0) {
 | |
|     snprintf(expect.name, sizeof(expect.name), "0.%s", pStream);
 | |
|     snprintf(expect.sourceDB, sizeof(expect.sourceDB), "0.%s", pSrcDb);
 | |
|     snprintf(expect.targetStbFullName, sizeof(expect.targetStbFullName), "0.test.%s", pDstStb);
 | |
|     expect.igExists = igExists;
 | |
|     expect.sql = taosStrdup(pSql);
 | |
|   };
 | |
| 
 | |
|   auto setStreamOptions =
 | |
|       [&](int8_t createStb = STREAM_CREATE_STABLE_TRUE, int8_t triggerType = STREAM_TRIGGER_WINDOW_CLOSE,
 | |
|           int64_t maxDelay = 0, int64_t watermark = 0, int8_t igExpired = STREAM_DEFAULT_IGNORE_EXPIRED,
 | |
|           int8_t fillHistory = STREAM_DEFAULT_FILL_HISTORY, int8_t igUpdate = STREAM_DEFAULT_IGNORE_UPDATE) {
 | |
|         expect.createStb = createStb;
 | |
|         expect.triggerType = triggerType;
 | |
|         expect.maxDelay = maxDelay;
 | |
|         expect.watermark = watermark;
 | |
|         expect.fillHistory = fillHistory;
 | |
|         expect.igExpired = igExpired;
 | |
|         expect.igUpdate = igUpdate;
 | |
|       };
 | |
| 
 | |
|   auto addTag = [&](const char* pFieldName, uint8_t type, int32_t bytes = 0) {
 | |
|     SField field = {0};
 | |
|     strcpy(field.name, pFieldName);
 | |
|     field.type = type;
 | |
|     field.bytes = bytes > 0 ? bytes : tDataTypes[type].bytes;
 | |
|     field.flags |= COL_SMA_ON;
 | |
| 
 | |
|     if (NULL == expect.pTags) {
 | |
|       expect.pTags = taosArrayInit(TARRAY_MIN_SIZE, sizeof(SField));
 | |
|     }
 | |
|     taosArrayPush(expect.pTags, &field);
 | |
|     expect.numOfTags += 1;
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_STREAM_STMT);
 | |
|     SCMCreateStreamReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS ==
 | |
|                 tDeserializeSCMCreateStreamReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.name), std::string(expect.name));
 | |
|     ASSERT_EQ(std::string(req.sourceDB), std::string(expect.sourceDB));
 | |
|     ASSERT_EQ(std::string(req.targetStbFullName), std::string(expect.targetStbFullName));
 | |
|     ASSERT_EQ(req.igExists, expect.igExists);
 | |
|     ASSERT_EQ(std::string(req.sql), std::string(expect.sql));
 | |
|     ASSERT_EQ(req.triggerType, expect.triggerType);
 | |
|     ASSERT_EQ(req.maxDelay, expect.maxDelay);
 | |
|     ASSERT_EQ(req.watermark, expect.watermark);
 | |
|     ASSERT_EQ(req.fillHistory, expect.fillHistory);
 | |
|     ASSERT_EQ(req.igExpired, expect.igExpired);
 | |
|     ASSERT_EQ(req.numOfTags, expect.numOfTags);
 | |
|     if (expect.numOfTags > 0) {
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pTags), expect.numOfTags);
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pTags), taosArrayGetSize(expect.pTags));
 | |
|       for (int32_t i = 0; i < expect.numOfTags; ++i) {
 | |
|         SField* pField = (SField*)taosArrayGet(req.pTags, i);
 | |
|         SField* pExpectField = (SField*)taosArrayGet(expect.pTags, i);
 | |
|         ASSERT_EQ(std::string(pField->name), std::string(pExpectField->name));
 | |
|         ASSERT_EQ(pField->type, pExpectField->type);
 | |
|         ASSERT_EQ(pField->bytes, pExpectField->bytes);
 | |
|         ASSERT_EQ(pField->flags, pExpectField->flags);
 | |
|       }
 | |
|     }
 | |
|     ASSERT_EQ(req.checkpointFreq, expect.checkpointFreq);
 | |
|     ASSERT_EQ(req.createStb, expect.createStb);
 | |
|     ASSERT_EQ(req.igUpdate, expect.igUpdate);
 | |
|     tFreeSCMCreateStreamReq(&req);
 | |
|   });
 | |
| 
 | |
|   setCreateStreamReq("s1", "test", "create stream s1 into st3 as select count(*) from t1 interval(10s)", "st3");
 | |
|   setStreamOptions();
 | |
|   run("CREATE STREAM s1 INTO st3 AS SELECT COUNT(*) FROM t1 INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| 
 | |
|   setCreateStreamReq(
 | |
|       "s1", "test",
 | |
|       "create stream if not exists s1 trigger max_delay 20s watermark 10s ignore expired 0 fill_history 0 ignore "
 | |
|       "update 1 into st3 as select count(*) from t1 interval(10s)",
 | |
|       "st3", 1);
 | |
|   setStreamOptions(STREAM_CREATE_STABLE_TRUE, STREAM_TRIGGER_MAX_DELAY, 20 * MILLISECOND_PER_SECOND,
 | |
|                    10 * MILLISECOND_PER_SECOND, 0, 0, 1);
 | |
|   run("CREATE STREAM IF NOT EXISTS s1 TRIGGER MAX_DELAY 20s WATERMARK 10s IGNORE EXPIRED 0 FILL_HISTORY 0 IGNORE "
 | |
|       "UPDATE 1 INTO st3 AS SELECT COUNT(*) FROM t1 INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| 
 | |
|   setCreateStreamReq("s1", "test",
 | |
|                      "create stream s1 into st3 tags(tname varchar(10), id int) subtable(concat('new-', tname)) as "
 | |
|                      "select _wstart wstart, count(*) cnt from st1 partition by tbname tname, tag1 id interval(10s)",
 | |
|                      "st3");
 | |
|   addTag("tname", TSDB_DATA_TYPE_VARCHAR, 10 + VARSTR_HEADER_SIZE);
 | |
|   addTag("id", TSDB_DATA_TYPE_INT);
 | |
|   setStreamOptions();
 | |
|   run("CREATE STREAM s1 INTO st3 TAGS(tname VARCHAR(10), id INT) SUBTABLE(CONCAT('new-', tname)) "
 | |
|       "AS SELECT _WSTART wstart, COUNT(*) cnt FROM st1 PARTITION BY TBNAME tname, tag1 id INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| 
 | |
|   // st1 already exists
 | |
|   setCreateStreamReq(
 | |
|       "s1", "test",
 | |
|       "create stream s1 into st1 tags(tag2) as select max(c1), c2 from t1 partition by tbname tag2 interval(10s)",
 | |
|       "st1");
 | |
|   setStreamOptions(STREAM_CREATE_STABLE_FALSE);
 | |
|   run("CREATE STREAM s1 INTO st1 TAGS(tag2) AS SELECT MAX(c1), c2 FROM t1 PARTITION BY TBNAME tag2 INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * CREATE MNODE ON DNODE dnode_id
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createMnode) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SMCreateMnodeReq expect = {0};
 | |
| 
 | |
|   auto setCreateMnodeReq = [&](int32_t dnodeId) { expect.dnodeId = dnodeId; };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_MNODE_STMT);
 | |
|     SMCreateMnodeReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS ==
 | |
|                 tDeserializeSCreateDropMQSNodeReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(req.dnodeId, expect.dnodeId);
 | |
|   });
 | |
| 
 | |
|   setCreateMnodeReq(1);
 | |
|   run("CREATE MNODE ON DNODE 1");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE QNODE ON DNODE dnode_id
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createQnode) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SMCreateQnodeReq expect = {0};
 | |
| 
 | |
|   auto setCreateQnodeReq = [&](int32_t dnodeId) { expect.dnodeId = dnodeId; };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_QNODE_STMT);
 | |
|     SMCreateQnodeReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS ==
 | |
|                 tDeserializeSCreateDropMQSNodeReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(req.dnodeId, expect.dnodeId);
 | |
|   });
 | |
| 
 | |
|   setCreateQnodeReq(1);
 | |
|   run("CREATE QNODE ON DNODE 1");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE SMA INDEX index_name ON tb_name index_option
 | |
|  *
 | |
|  * index_option:
 | |
|  *     FUNCTION(functions) INTERVAL(interval_val [, interval_offset]) [SLIDING(sliding_val)]
 | |
|  *         [WATERMARK(watermark_val)] [MAX_DELAY(max_delay_val)]
 | |
|  *
 | |
|  * functions:
 | |
|  *     function [, function] ...
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createSmaIndex) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SMCreateSmaReq expect = {0};
 | |
| 
 | |
|   auto setCreateSmacReq = [&](const char* pIndexName, const char* pStbName, int64_t interval, int8_t intervalUnit,
 | |
|                               int64_t offset = 0, int64_t sliding = -1, int8_t slidingUnit = -1, int8_t igExists = 0) {
 | |
|     memset(&expect, 0, sizeof(SMCreateSmaReq));
 | |
|     strcpy(expect.name, pIndexName);
 | |
|     strcpy(expect.stb, pStbName);
 | |
|     expect.igExists = igExists;
 | |
|     expect.intervalUnit = intervalUnit;
 | |
|     expect.slidingUnit = slidingUnit < 0 ? intervalUnit : slidingUnit;
 | |
|     expect.timezone = 0;
 | |
|     expect.dstVgId = 1;
 | |
|     expect.interval = interval;
 | |
|     expect.offset = offset;
 | |
|     expect.sliding = sliding < 0 ? interval : sliding;
 | |
|     expect.maxDelay = -1;
 | |
|     expect.watermark = TSDB_DEFAULT_ROLLUP_WATERMARK;
 | |
|     expect.deleteMark = TSDB_DEFAULT_ROLLUP_DELETE_MARK;
 | |
|   };
 | |
| 
 | |
|   auto setOptionsForCreateSmacReq = [&](int64_t maxDelay, int64_t watermark, int64_t deleteMark) {
 | |
|     expect.maxDelay = maxDelay;
 | |
|     expect.watermark = watermark;
 | |
|     expect.deleteMark = deleteMark;
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_INDEX_STMT);
 | |
|     SMCreateSmaReq req = {0};
 | |
|     ASSERT_TRUE(pQuery->pPrevRoot);
 | |
|     ASSERT_EQ(QUERY_NODE_SELECT_STMT, nodeType(pQuery->pPrevRoot));
 | |
| 
 | |
|     SCreateIndexStmt* pStmt = (SCreateIndexStmt*)pQuery->pRoot;
 | |
|     SCmdMsgInfo* pCmdMsg = (SCmdMsgInfo*)taosMemoryMalloc(sizeof(SCmdMsgInfo));
 | |
|     if (NULL == pCmdMsg) FAIL();
 | |
|     pCmdMsg->msgType = TDMT_MND_CREATE_SMA;
 | |
|     pCmdMsg->msgLen = tSerializeSMCreateSmaReq(NULL, 0, pStmt->pReq);
 | |
|     pCmdMsg->pMsg = taosMemoryMalloc(pCmdMsg->msgLen);
 | |
|     if (!pCmdMsg->pMsg) FAIL();
 | |
|     tSerializeSMCreateSmaReq(pCmdMsg->pMsg, pCmdMsg->msgLen, pStmt->pReq);
 | |
|     ((SQuery*)pQuery)->pCmdMsg = pCmdMsg;
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS == tDeserializeSMCreateSmaReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.name), std::string(expect.name));
 | |
|     ASSERT_EQ(std::string(req.stb), std::string(expect.stb));
 | |
|     ASSERT_EQ(req.igExists, expect.igExists);
 | |
|     ASSERT_EQ(req.intervalUnit, expect.intervalUnit);
 | |
|     ASSERT_EQ(req.slidingUnit, expect.slidingUnit);
 | |
|     ASSERT_EQ(req.timezone, expect.timezone);
 | |
|     ASSERT_EQ(req.dstVgId, expect.dstVgId);
 | |
|     ASSERT_EQ(req.interval, expect.interval);
 | |
|     ASSERT_EQ(req.offset, expect.offset);
 | |
|     ASSERT_EQ(req.sliding, expect.sliding);
 | |
|     ASSERT_EQ(req.maxDelay, expect.maxDelay);
 | |
|     ASSERT_EQ(req.watermark, expect.watermark);
 | |
|     ASSERT_EQ(req.deleteMark, expect.deleteMark);
 | |
|     ASSERT_GT(req.exprLen, 0);
 | |
|     ASSERT_EQ(req.tagsFilterLen, 0);
 | |
|     ASSERT_GT(req.sqlLen, 0);
 | |
|     ASSERT_GT(req.astLen, 0);
 | |
|     ASSERT_NE(req.expr, nullptr);
 | |
|     ASSERT_EQ(req.tagsFilter, nullptr);
 | |
|     ASSERT_NE(req.sql, nullptr);
 | |
|     ASSERT_NE(req.ast, nullptr);
 | |
|     tFreeSMCreateSmaReq(&req);
 | |
|   });
 | |
| 
 | |
|   setCreateSmacReq("0.test.index1", "0.test.t1", 10 * MILLISECOND_PER_SECOND, 's');
 | |
|   run("CREATE SMA INDEX index1 ON t1 FUNCTION(MAX(c1), MIN(c3 + 10), SUM(c4)) INTERVAL(10s)");
 | |
| 
 | |
|   setCreateSmacReq("0.test.index2", "0.test.st1", 5 * MILLISECOND_PER_SECOND, 's');
 | |
|   setOptionsForCreateSmacReq(10 * MILLISECOND_PER_SECOND, 20 * MILLISECOND_PER_SECOND, 1000 * MILLISECOND_PER_SECOND);
 | |
|   run("CREATE SMA INDEX index2 ON st1 FUNCTION(MAX(c1), MIN(tag1)) INTERVAL(5s) WATERMARK 20s MAX_DELAY 10s "
 | |
|       "DELETE_MARK 1000s");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE SNODE ON DNODE dnode_id
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createSnode) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SMCreateSnodeReq expect = {0};
 | |
| 
 | |
|   auto setCreateSnodeReq = [&](int32_t dnodeId) { expect.dnodeId = dnodeId; };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_SNODE_STMT);
 | |
|     SMCreateSnodeReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS ==
 | |
|                 tDeserializeSCreateDropMQSNodeReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(req.dnodeId, expect.dnodeId);
 | |
|   });
 | |
| 
 | |
|   setCreateSnodeReq(1);
 | |
|   run("CREATE SNODE ON DNODE 1");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE STABLE [IF NOT EXISTS] stb_name (create_definition [, create_definitionn] ...)
 | |
|  *     TAGS (create_definition [, create_definition] ...) [table_options]
 | |
|  *
 | |
|  * create_definition:
 | |
|  *     col_name column_definition
 | |
|  *
 | |
|  * column_definition:
 | |
|  *     type_name [COMMENT 'string_value']
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createStable) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SMCreateStbReq expect = {0};
 | |
| 
 | |
|   auto clearCreateStbReq = [&]() {
 | |
|     tFreeSMCreateStbReq(&expect);
 | |
|     memset(&expect, 0, sizeof(SMCreateStbReq));
 | |
|   };
 | |
| 
 | |
|   auto setCreateStbReq =
 | |
|       [&](const char* pDbName, const char* pTbName, int8_t igExists = 0, int64_t delay1 = -1, int64_t delay2 = -1,
 | |
|           int64_t watermark1 = TSDB_DEFAULT_ROLLUP_WATERMARK, int64_t watermark2 = TSDB_DEFAULT_ROLLUP_WATERMARK,
 | |
|           int64_t deleteMark1 = TSDB_DEFAULT_ROLLUP_DELETE_MARK, int64_t deleteMark2 = TSDB_DEFAULT_ROLLUP_DELETE_MARK,
 | |
|           int32_t ttl = TSDB_DEFAULT_TABLE_TTL, const char* pComment = nullptr) {
 | |
|         int32_t len = snprintf(expect.name, sizeof(expect.name), "0.%s.%s", pDbName, pTbName);
 | |
|         expect.name[len] = '\0';
 | |
|         expect.igExists = igExists;
 | |
|         expect.delay1 = delay1;
 | |
|         expect.delay2 = delay2;
 | |
|         expect.watermark1 = watermark1;
 | |
|         expect.watermark2 = watermark2;
 | |
|         expect.deleteMark1 = deleteMark1;
 | |
|         expect.deleteMark2 = deleteMark2;
 | |
|         // expect.ttl = ttl;
 | |
|         if (nullptr != pComment) {
 | |
|           expect.pComment = taosStrdup(pComment);
 | |
|           expect.commentLen = strlen(pComment);
 | |
|         }
 | |
|       };
 | |
| 
 | |
|   auto addFieldToCreateStbReq = [&](bool col, const char* pFieldName, uint8_t type, int32_t bytes = 0,
 | |
|                                     int8_t flags = COL_SMA_ON) {
 | |
|     SField field = {0};
 | |
|     strcpy(field.name, pFieldName);
 | |
|     field.type = type;
 | |
|     field.bytes = bytes > 0 ? bytes : tDataTypes[type].bytes;
 | |
|     field.flags = flags;
 | |
| 
 | |
|     if (col) {
 | |
|       if (NULL == expect.pColumns) {
 | |
|         expect.pColumns = taosArrayInit(TARRAY_MIN_SIZE, sizeof(SField));
 | |
|       }
 | |
|       taosArrayPush(expect.pColumns, &field);
 | |
|       expect.numOfColumns += 1;
 | |
|     } else {
 | |
|       if (NULL == expect.pTags) {
 | |
|         expect.pTags = taosArrayInit(TARRAY_MIN_SIZE, sizeof(SField));
 | |
|       }
 | |
|       taosArrayPush(expect.pTags, &field);
 | |
|       expect.numOfTags += 1;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_TABLE_STMT);
 | |
|     SMCreateStbReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS == tDeserializeSMCreateStbReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.name), std::string(expect.name));
 | |
|     ASSERT_EQ(req.igExists, expect.igExists);
 | |
|     ASSERT_EQ(req.delay1, expect.delay1);
 | |
|     ASSERT_EQ(req.delay2, expect.delay2);
 | |
|     ASSERT_EQ(req.watermark1, expect.watermark1);
 | |
|     ASSERT_EQ(req.watermark2, expect.watermark2);
 | |
|     ASSERT_EQ(req.ttl, expect.ttl);
 | |
|     ASSERT_EQ(req.numOfColumns, expect.numOfColumns);
 | |
|     ASSERT_EQ(req.numOfTags, expect.numOfTags);
 | |
|     //    ASSERT_EQ(req.commentLen, expect.commentLen);
 | |
|     ASSERT_EQ(req.ast1Len, expect.ast1Len);
 | |
|     ASSERT_EQ(req.ast2Len, expect.ast2Len);
 | |
| 
 | |
|     if (expect.numOfColumns > 0) {
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pColumns), expect.numOfColumns);
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pColumns), taosArrayGetSize(expect.pColumns));
 | |
|       for (int32_t i = 0; i < expect.numOfColumns; ++i) {
 | |
|         SField* pField = (SField*)taosArrayGet(req.pColumns, i);
 | |
|         SField* pExpectField = (SField*)taosArrayGet(expect.pColumns, i);
 | |
|         ASSERT_EQ(std::string(pField->name), std::string(pExpectField->name));
 | |
|         ASSERT_EQ(pField->type, pExpectField->type);
 | |
|         ASSERT_EQ(pField->bytes, pExpectField->bytes);
 | |
|         ASSERT_EQ(pField->flags, pExpectField->flags);
 | |
|       }
 | |
|     }
 | |
|     if (expect.numOfTags > 0) {
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pTags), expect.numOfTags);
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pTags), taosArrayGetSize(expect.pTags));
 | |
|       for (int32_t i = 0; i < expect.numOfTags; ++i) {
 | |
|         SField* pField = (SField*)taosArrayGet(req.pTags, i);
 | |
|         SField* pExpectField = (SField*)taosArrayGet(expect.pTags, i);
 | |
|         ASSERT_EQ(std::string(pField->name), std::string(pExpectField->name));
 | |
|         ASSERT_EQ(pField->type, pExpectField->type);
 | |
|         ASSERT_EQ(pField->bytes, pExpectField->bytes);
 | |
|         ASSERT_EQ(pField->flags, pExpectField->flags);
 | |
|       }
 | |
|     }
 | |
|     if (expect.commentLen > 0) {
 | |
|       ASSERT_EQ(std::string(req.pComment), std::string(expect.pComment));
 | |
|     }
 | |
|     if (expect.ast1Len > 0) {
 | |
|       ASSERT_EQ(std::string(req.pAst1), std::string(expect.pAst1));
 | |
|     }
 | |
|     if (expect.ast2Len > 0) {
 | |
|       ASSERT_EQ(std::string(req.pAst2), std::string(expect.pAst2));
 | |
|     }
 | |
|     tFreeSMCreateStbReq(&req);
 | |
|   });
 | |
| 
 | |
|   setCreateStbReq("test", "t1");
 | |
|   addFieldToCreateStbReq(true, "ts", TSDB_DATA_TYPE_TIMESTAMP);
 | |
|   addFieldToCreateStbReq(true, "c1", TSDB_DATA_TYPE_INT);
 | |
|   addFieldToCreateStbReq(false, "id", TSDB_DATA_TYPE_INT);
 | |
|   run("CREATE STABLE t1(ts TIMESTAMP, c1 INT) TAGS(id INT)");
 | |
|   clearCreateStbReq();
 | |
| 
 | |
|   setCreateStbReq("rollup_db", "t1", 1, 100 * MILLISECOND_PER_SECOND, 10 * MILLISECOND_PER_MINUTE, 10,
 | |
|                   1 * MILLISECOND_PER_MINUTE, 1000 * MILLISECOND_PER_SECOND, 200 * MILLISECOND_PER_MINUTE, 100,
 | |
|                   "test create table");
 | |
|   addFieldToCreateStbReq(true, "ts", TSDB_DATA_TYPE_TIMESTAMP, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c1", TSDB_DATA_TYPE_INT);
 | |
|   addFieldToCreateStbReq(true, "c2", TSDB_DATA_TYPE_UINT);
 | |
|   addFieldToCreateStbReq(true, "c3", TSDB_DATA_TYPE_BIGINT);
 | |
|   addFieldToCreateStbReq(true, "c4", TSDB_DATA_TYPE_UBIGINT, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c5", TSDB_DATA_TYPE_FLOAT, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c6", TSDB_DATA_TYPE_DOUBLE, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c7", TSDB_DATA_TYPE_BINARY, 20 + VARSTR_HEADER_SIZE, 0);
 | |
|   addFieldToCreateStbReq(true, "c8", TSDB_DATA_TYPE_SMALLINT, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c9", TSDB_DATA_TYPE_USMALLINT, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c10", TSDB_DATA_TYPE_TINYINT, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c11", TSDB_DATA_TYPE_UTINYINT, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c12", TSDB_DATA_TYPE_BOOL, 0, 0);
 | |
|   addFieldToCreateStbReq(true, "c13", TSDB_DATA_TYPE_NCHAR, 30 * TSDB_NCHAR_SIZE + VARSTR_HEADER_SIZE, 0);
 | |
|   addFieldToCreateStbReq(true, "c14", TSDB_DATA_TYPE_VARCHAR, 50 + VARSTR_HEADER_SIZE, 0);
 | |
|   addFieldToCreateStbReq(false, "a1", TSDB_DATA_TYPE_TIMESTAMP);
 | |
|   addFieldToCreateStbReq(false, "a2", TSDB_DATA_TYPE_INT);
 | |
|   addFieldToCreateStbReq(false, "a3", TSDB_DATA_TYPE_UINT);
 | |
|   addFieldToCreateStbReq(false, "a4", TSDB_DATA_TYPE_BIGINT);
 | |
|   addFieldToCreateStbReq(false, "a5", TSDB_DATA_TYPE_UBIGINT);
 | |
|   addFieldToCreateStbReq(false, "a6", TSDB_DATA_TYPE_FLOAT);
 | |
|   addFieldToCreateStbReq(false, "a7", TSDB_DATA_TYPE_DOUBLE);
 | |
|   addFieldToCreateStbReq(false, "a8", TSDB_DATA_TYPE_BINARY, 20 + VARSTR_HEADER_SIZE);
 | |
|   addFieldToCreateStbReq(false, "a9", TSDB_DATA_TYPE_SMALLINT);
 | |
|   addFieldToCreateStbReq(false, "a10", TSDB_DATA_TYPE_USMALLINT);
 | |
|   addFieldToCreateStbReq(false, "a11", TSDB_DATA_TYPE_TINYINT);
 | |
|   addFieldToCreateStbReq(false, "a12", TSDB_DATA_TYPE_UTINYINT);
 | |
|   addFieldToCreateStbReq(false, "a13", TSDB_DATA_TYPE_BOOL);
 | |
|   addFieldToCreateStbReq(false, "a14", TSDB_DATA_TYPE_NCHAR, 30 * TSDB_NCHAR_SIZE + VARSTR_HEADER_SIZE);
 | |
|   addFieldToCreateStbReq(false, "a15", TSDB_DATA_TYPE_VARCHAR, 50 + VARSTR_HEADER_SIZE);
 | |
|   run("CREATE STABLE IF NOT EXISTS rollup_db.t1("
 | |
|       "ts TIMESTAMP, c1 INT, c2 INT UNSIGNED, c3 BIGINT, c4 BIGINT UNSIGNED, c5 FLOAT, c6 DOUBLE, c7 BINARY(20), "
 | |
|       "c8 SMALLINT, c9 SMALLINT UNSIGNED, c10 TINYINT, c11 TINYINT UNSIGNED, c12 BOOL, "
 | |
|       "c13 NCHAR(30), c14 VARCHAR(50)) "
 | |
|       "TAGS (a1 TIMESTAMP, a2 INT, a3 INT UNSIGNED, a4 BIGINT, a5 BIGINT UNSIGNED, a6 FLOAT, a7 DOUBLE, "
 | |
|       "a8 BINARY(20), a9 SMALLINT, a10 SMALLINT UNSIGNED, a11 TINYINT, "
 | |
|       "a12 TINYINT UNSIGNED, a13 BOOL, a14 NCHAR(30), a15 VARCHAR(50)) "
 | |
|       "COMMENT 'test create table' SMA(c1, c2, c3) ROLLUP (MIN) MAX_DELAY 100s,10m WATERMARK 10a,1m "
 | |
|       "DELETE_MARK 1000s,200m");
 | |
|   clearCreateStbReq();
 | |
| }
 | |
| 
 | |
| TEST_F(ParserInitialCTest, createStableSemanticCheck) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   run("CREATE STABLE rollup_db.stb2 (ts TIMESTAMP, c1 INT) TAGS (tag1 INT) ROLLUP(CEIL)",
 | |
|       TSDB_CODE_PAR_INVALID_TABLE_OPTION);
 | |
| 
 | |
|   run("CREATE STABLE rollup_db.stb2 (ts TIMESTAMP, c1 INT) TAGS (tag1 INT) ROLLUP(MAX) MAX_DELAY 0s WATERMARK 1m",
 | |
|       TSDB_CODE_PAR_INVALID_TABLE_OPTION);
 | |
| 
 | |
|   run("CREATE STABLE rollup_db.stb2 (ts TIMESTAMP, c1 INT) TAGS (tag1 INT) ROLLUP(MAX) MAX_DELAY 10s WATERMARK 18m",
 | |
|       TSDB_CODE_PAR_INVALID_TABLE_OPTION);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE STREAM [IF NOT EXISTS] stream_name [stream_options]
 | |
|  *     INTO stb_name [TAGS (create_definition [, create_definition] ...)] [SUBTABLE (expr)] AS subquery
 | |
|  *
 | |
|  * stream_options:
 | |
|  *     stream_option ...
 | |
|  *
 | |
|  * stream_option: {
 | |
|  *     TRIGGER    [AT_ONCE | WINDOW_CLOSE | MAX_DELAY time]
 | |
|  *   | WATERMARK   time
 | |
|  *   | IGNORE EXPIRED value
 | |
|  *   | FILL_HISTORY value
 | |
|  * }
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createStream) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SCMCreateStreamReq expect = {0};
 | |
| 
 | |
|   auto clearCreateStreamReq = [&]() {
 | |
|     tFreeSCMCreateStreamReq(&expect);
 | |
|     memset(&expect, 0, sizeof(SCMCreateStreamReq));
 | |
|   };
 | |
| 
 | |
|   auto setCreateStreamReq = [&](const char* pStream, const char* pSrcDb, const char* pSql, const char* pDstStb,
 | |
|                                 int8_t igExists = 0) {
 | |
|     snprintf(expect.name, sizeof(expect.name), "0.%s", pStream);
 | |
|     snprintf(expect.sourceDB, sizeof(expect.sourceDB), "0.%s", pSrcDb);
 | |
|     snprintf(expect.targetStbFullName, sizeof(expect.targetStbFullName), "0.test.%s", pDstStb);
 | |
|     expect.igExists = igExists;
 | |
|     expect.sql = taosStrdup(pSql);
 | |
|   };
 | |
| 
 | |
|   auto setStreamOptions =
 | |
|       [&](int8_t createStb = STREAM_CREATE_STABLE_TRUE, int8_t triggerType = STREAM_TRIGGER_WINDOW_CLOSE,
 | |
|           int64_t maxDelay = 0, int64_t watermark = 0, int8_t igExpired = STREAM_DEFAULT_IGNORE_EXPIRED,
 | |
|           int8_t fillHistory = STREAM_DEFAULT_FILL_HISTORY, int8_t igUpdate = STREAM_DEFAULT_IGNORE_UPDATE) {
 | |
|         expect.createStb = createStb;
 | |
|         expect.triggerType = triggerType;
 | |
|         expect.maxDelay = maxDelay;
 | |
|         expect.watermark = watermark;
 | |
|         expect.fillHistory = fillHistory;
 | |
|         expect.igExpired = igExpired;
 | |
|         expect.igUpdate = igUpdate;
 | |
|       };
 | |
| 
 | |
|   auto addTag = [&](const char* pFieldName, uint8_t type, int32_t bytes = 0) {
 | |
|     SField field = {0};
 | |
|     strcpy(field.name, pFieldName);
 | |
|     field.type = type;
 | |
|     field.bytes = bytes > 0 ? bytes : tDataTypes[type].bytes;
 | |
|     field.flags |= COL_SMA_ON;
 | |
| 
 | |
|     if (NULL == expect.pTags) {
 | |
|       expect.pTags = taosArrayInit(TARRAY_MIN_SIZE, sizeof(SField));
 | |
|     }
 | |
|     taosArrayPush(expect.pTags, &field);
 | |
|     expect.numOfTags += 1;
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_STREAM_STMT);
 | |
|     SCMCreateStreamReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS ==
 | |
|                 tDeserializeSCMCreateStreamReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.name), std::string(expect.name));
 | |
|     ASSERT_EQ(std::string(req.sourceDB), std::string(expect.sourceDB));
 | |
|     ASSERT_EQ(std::string(req.targetStbFullName), std::string(expect.targetStbFullName));
 | |
|     ASSERT_EQ(req.igExists, expect.igExists);
 | |
|     ASSERT_EQ(std::string(req.sql), std::string(expect.sql));
 | |
|     ASSERT_EQ(req.triggerType, expect.triggerType);
 | |
|     ASSERT_EQ(req.maxDelay, expect.maxDelay);
 | |
|     ASSERT_EQ(req.watermark, expect.watermark);
 | |
|     ASSERT_EQ(req.fillHistory, expect.fillHistory);
 | |
|     ASSERT_EQ(req.igExpired, expect.igExpired);
 | |
|     ASSERT_EQ(req.numOfTags, expect.numOfTags);
 | |
|     if (expect.numOfTags > 0) {
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pTags), expect.numOfTags);
 | |
|       ASSERT_EQ(taosArrayGetSize(req.pTags), taosArrayGetSize(expect.pTags));
 | |
|       for (int32_t i = 0; i < expect.numOfTags; ++i) {
 | |
|         SField* pField = (SField*)taosArrayGet(req.pTags, i);
 | |
|         SField* pExpectField = (SField*)taosArrayGet(expect.pTags, i);
 | |
|         ASSERT_EQ(std::string(pField->name), std::string(pExpectField->name));
 | |
|         ASSERT_EQ(pField->type, pExpectField->type);
 | |
|         ASSERT_EQ(pField->bytes, pExpectField->bytes);
 | |
|         ASSERT_EQ(pField->flags, pExpectField->flags);
 | |
|       }
 | |
|     }
 | |
|     ASSERT_EQ(req.checkpointFreq, expect.checkpointFreq);
 | |
|     ASSERT_EQ(req.createStb, expect.createStb);
 | |
|     ASSERT_EQ(req.igUpdate, expect.igUpdate);
 | |
|     tFreeSCMCreateStreamReq(&req);
 | |
|   });
 | |
| 
 | |
|   setCreateStreamReq("s1", "test", "create stream s1 into st3 as select count(*) from t1 interval(10s)", "st3");
 | |
|   setStreamOptions();
 | |
|   run("CREATE STREAM s1 INTO st3 AS SELECT COUNT(*) FROM t1 INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| 
 | |
|   setCreateStreamReq(
 | |
|       "s1", "test",
 | |
|       "create stream if not exists s1 trigger max_delay 20s watermark 10s ignore expired 0 fill_history 0 ignore "
 | |
|       "update 1 into st3 as select count(*) from t1 interval(10s)",
 | |
|       "st3", 1);
 | |
|   setStreamOptions(STREAM_CREATE_STABLE_TRUE, STREAM_TRIGGER_MAX_DELAY, 20 * MILLISECOND_PER_SECOND,
 | |
|                    10 * MILLISECOND_PER_SECOND, 0, 0, 1);
 | |
|   run("CREATE STREAM IF NOT EXISTS s1 TRIGGER MAX_DELAY 20s WATERMARK 10s IGNORE EXPIRED 0 FILL_HISTORY 0 IGNORE "
 | |
|       "UPDATE 1 INTO st3 AS SELECT COUNT(*) FROM t1 INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| 
 | |
|   setCreateStreamReq("s1", "test",
 | |
|                      "create stream s1 into st3 tags(tname varchar(10), id int) subtable(concat('new-', tname)) as "
 | |
|                      "select _wstart wstart, count(*) cnt from st1 partition by tbname tname, tag1 id interval(10s)",
 | |
|                      "st3");
 | |
|   addTag("tname", TSDB_DATA_TYPE_VARCHAR, 10 + VARSTR_HEADER_SIZE);
 | |
|   addTag("id", TSDB_DATA_TYPE_INT);
 | |
|   setStreamOptions();
 | |
|   run("CREATE STREAM s1 INTO st3 TAGS(tname VARCHAR(10), id INT) SUBTABLE(CONCAT('new-', tname)) "
 | |
|       "AS SELECT _WSTART wstart, COUNT(*) cnt FROM st1 PARTITION BY TBNAME tname, tag1 id INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| 
 | |
|   // st1 already exists
 | |
|   setCreateStreamReq(
 | |
|       "s1", "test",
 | |
|       "create stream s1 into st1 tags(tag2) as select max(c1), c2 from t1 partition by tbname tag2 interval(10s)",
 | |
|       "st1");
 | |
|   setStreamOptions(STREAM_CREATE_STABLE_FALSE);
 | |
|   run("CREATE STREAM s1 INTO st1 TAGS(tag2) AS SELECT MAX(c1), c2 FROM t1 PARTITION BY TBNAME tag2 INTERVAL(10S)");
 | |
|   clearCreateStreamReq();
 | |
| }
 | |
| 
 | |
| TEST_F(ParserInitialCTest, createStreamSemanticCheck) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   run("CREATE STREAM s1 INTO st1 AS SELECT PERCENTILE(c1, 30) FROM t1 INTERVAL(10S)",
 | |
|       TSDB_CODE_PAR_STREAM_NOT_ALLOWED_FUNC);
 | |
|   run("CREATE STREAM s2 INTO st1 AS SELECT ts, to_json('{c1:1}') FROM st1 PARTITION BY TBNAME",
 | |
|       TSDB_CODE_PAR_INVALID_STREAM_QUERY);
 | |
|   run("CREATE STREAM s3 INTO st3 TAGS(tname VARCHAR(10), id INT) SUBTABLE(CONCAT('new-', tbname)) "
 | |
|       "AS SELECT _WSTART wstart, COUNT(*) cnt FROM st1 INTERVAL(10S)", TSDB_CODE_PAR_INVALID_STREAM_QUERY);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE TABLE [IF NOT EXISTS] [db_name.]tb_name (create_definition [, create_definitionn] ...) [table_options]
 | |
|  *
 | |
|  * CREATE TABLE create_subtable_clause
 | |
|  *
 | |
|  * CREATE TABLE [IF NOT EXISTS] [db_name.]tb_name (create_definition [, create_definitionn] ...)
 | |
|  *     [TAGS (create_definition [, create_definitionn] ...)]
 | |
|  *     [table_options]
 | |
|  *
 | |
|  * create_subtable_clause: {
 | |
|  *     create_subtable_clause [create_subtable_clause] ...
 | |
|  *   | [IF NOT EXISTS] [db_name.]tb_name USING [db_name.]stb_name [(tag_name [, tag_name] ...)]
 | |
|  *         TAGS (tag_value [, tag_value] ...)
 | |
|  * }
 | |
|  *
 | |
|  * create_definition:
 | |
|  *     col_name column_definition
 | |
|  *
 | |
|  * column_definition:
 | |
|  *     type_name [comment 'string_value']
 | |
|  *
 | |
|  * table_options:
 | |
|  *     table_option ...
 | |
|  *
 | |
|  * table_option: {
 | |
|  *     COMMENT 'string_value'
 | |
|  *   | WATERMARK duration[,duration]
 | |
|  *   | MAX_DELAY duration[,duration]
 | |
|  *   | ROLLUP(func_name [, func_name] ...)
 | |
|  *   | SMA(col_name [, col_name] ...)
 | |
|  *   | TTL value
 | |
|  * }
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createTable) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SVCreateTbBatchReq expect = {0};
 | |
| 
 | |
|   auto addCreateTbReq = [&](const char* pName, bool ignoreExists = false, int32_t ttl = TSDB_DEFAULT_TABLE_TTL,
 | |
|                             const char* pComment = nullptr) {
 | |
|     SVCreateTbReq req = {0};
 | |
|     req.name = taosStrdup(pName);
 | |
|     if (ignoreExists) {
 | |
|       req.flags |= TD_CREATE_IF_NOT_EXISTS;
 | |
|     }
 | |
|     req.ttl = ttl;
 | |
|     if (nullptr != pComment) {
 | |
|       req.comment = taosStrdup(pComment);
 | |
|       req.commentLen = strlen(pComment);
 | |
|     }
 | |
|     ++expect.nReqs;
 | |
|     if (nullptr == expect.pArray) {
 | |
|       expect.pArray = taosArrayInit(TARRAY_MIN_SIZE, sizeof(SVCreateTbReq));
 | |
|     }
 | |
|     taosArrayPush(expect.pArray, &req);
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     return;  // todo
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_VNODE_MODIFY_STMT);
 | |
|     SVnodeModifyOpStmt* pStmt = (SVnodeModifyOpStmt*)pQuery->pRoot;
 | |
| 
 | |
|     ASSERT_EQ(pStmt->sqlNodeType, QUERY_NODE_CREATE_TABLE_STMT);
 | |
|     ASSERT_NE(pStmt->pDataBlocks, nullptr);
 | |
|     int32_t numOfBlocks = taosArrayGetSize(pStmt->pDataBlocks);
 | |
|     for (int32_t i = 0; i < numOfBlocks; ++i) {
 | |
|       SVgDataBlocks*     pVgData = (SVgDataBlocks*)taosArrayGetP(pStmt->pDataBlocks, i);
 | |
|       void*              pBuf = POINTER_SHIFT(pVgData->pData, sizeof(SMsgHead));
 | |
|       SVCreateTbBatchReq req = {0};
 | |
|       SDecoder           coder = {0};
 | |
|       tDecoderInit(&coder, (uint8_t*)pBuf, pVgData->size);
 | |
|       ASSERT_EQ(tDecodeSVCreateTbBatchReq(&coder, &req), TSDB_CODE_SUCCESS);
 | |
|       ASSERT_EQ(req.nReqs, expect.nReqs);
 | |
|       for (int32_t j = 0; j < req.nReqs; ++j) {
 | |
|         SVCreateTbReq* pReq = req.pReqs + j;
 | |
|         SVCreateTbReq* pExpect = (SVCreateTbReq*)taosArrayGet(expect.pArray, j);
 | |
|         ASSERT_EQ(pReq->flags, pExpect->flags);
 | |
|         ASSERT_EQ(std::string(pReq->name), std::string(pExpect->name));
 | |
|         ASSERT_EQ(pReq->uid, pExpect->uid);
 | |
|         ASSERT_EQ(pReq->btime, pExpect->btime);
 | |
|         ASSERT_EQ(pReq->ttl, pExpect->ttl);
 | |
|         ASSERT_EQ(pReq->commentLen, pExpect->commentLen);
 | |
|         ASSERT_EQ(std::string(pReq->comment), std::string(pExpect->comment));
 | |
|         ASSERT_EQ(pReq->type, pExpect->type);
 | |
|         if (TD_NORMAL_TABLE == pExpect->type) {
 | |
|           ASSERT_EQ(pReq->ntb.schemaRow.version, pExpect->ntb.schemaRow.version);
 | |
|           ASSERT_EQ(pReq->ntb.schemaRow.nCols, pExpect->ntb.schemaRow.nCols);
 | |
|         } else if (TD_CHILD_TABLE == pExpect->type) {
 | |
|           ASSERT_EQ(std::string(pReq->ctb.stbName), std::string(pExpect->ctb.stbName));
 | |
|           ASSERT_EQ(pReq->ctb.tagNum, pExpect->ctb.tagNum);
 | |
|           ASSERT_EQ(pReq->ctb.suid, pExpect->ctb.suid);
 | |
|         }
 | |
|       }
 | |
|       tDecoderClear(&coder);
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   run("CREATE TABLE t1(ts TIMESTAMP, c1 INT)");
 | |
| 
 | |
|   run("CREATE TABLE IF NOT EXISTS test.t1("
 | |
|       "ts TIMESTAMP, c1 INT, c2 INT UNSIGNED, c3 BIGINT, c4 BIGINT UNSIGNED, c5 FLOAT, c6 DOUBLE, c7 BINARY(20), "
 | |
|       "c8 SMALLINT, c9 SMALLINT UNSIGNED, c10 TINYINT, c11 TINYINT UNSIGNED, c12 BOOL, "
 | |
|       "c13 NCHAR(30), c15 VARCHAR(50)) "
 | |
|       "TTL 100 COMMENT 'test create table' SMA(c1, c2, c3)");
 | |
| 
 | |
|   run("CREATE TABLE IF NOT EXISTS rollup_db.t1("
 | |
|       "ts TIMESTAMP, c1 INT, c2 INT UNSIGNED, c3 BIGINT, c4 BIGINT UNSIGNED, c5 FLOAT, c6 DOUBLE, c7 BINARY(20), "
 | |
|       "c8 SMALLINT, c9 SMALLINT UNSIGNED, c10 TINYINT, c11 TINYINT UNSIGNED, c12 BOOL, "
 | |
|       "c13 NCHAR(30), c14 VARCHAR(50)) "
 | |
|       "TAGS (a1 TIMESTAMP, a2 INT, a3 INT UNSIGNED, a4 BIGINT, a5 BIGINT UNSIGNED, a6 FLOAT, a7 DOUBLE, a8 BINARY(20), "
 | |
|       "a9 SMALLINT, a10 SMALLINT UNSIGNED, a11 TINYINT, a12 TINYINT UNSIGNED, a13 BOOL, "
 | |
|       "a14 NCHAR(30), a15 VARCHAR(50)) "
 | |
|       "COMMENT 'test create table' SMA(c1, c2, c3) ROLLUP (MIN)");
 | |
| 
 | |
|   run("CREATE TABLE IF NOT EXISTS t1 USING st1 TAGS(1, 'wxy', NOW)");
 | |
| 
 | |
|   run("CREATE TABLE "
 | |
|       "IF NOT EXISTS test.t1 USING test.st1 (tag1, tag2) TAGS(1, 'abc') "
 | |
|       "IF NOT EXISTS test.t2 USING test.st1 (tag1, tag2) TAGS(2, 'abc') "
 | |
|       "IF NOT EXISTS test.t3 USING test.st1 (tag1, tag2) TAGS(3, 'abc') ");
 | |
| 
 | |
|   run("CREATE TABLE IF NOT EXISTS t1 USING st1 TAGS(1, 'wxy', NOW + 1S)");
 | |
| }
 | |
| 
 | |
| TEST_F(ParserInitialCTest, createTableSemanticCheck) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   string sql = "CREATE TABLE st1(ts TIMESTAMP, ";
 | |
|   for (int32_t i = 1; i < 4096; ++i) {
 | |
|     if (i > 1) {
 | |
|       sql.append(", ");
 | |
|     }
 | |
|     sql.append("c" + to_string(i) + " INT");
 | |
|   }
 | |
|   sql.append(") TAGS (t1 int)");
 | |
| 
 | |
|   run(sql, TSDB_CODE_PAR_TOO_MANY_COLUMNS);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE TOPIC [IF NOT EXISTS] topic_name AS subquery
 | |
|  *
 | |
|  * CREATE TOPIC [IF NOT EXISTS] topic_name [WITH META] AS {DATABASE db_name | STABLE stb_name }
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createTopic) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SCMCreateTopicReq expect = {0};
 | |
| 
 | |
|   auto clearCreateTopicReq = [&]() { memset(&expect, 0, sizeof(SCMCreateTopicReq)); };
 | |
| 
 | |
|   auto setCreateTopicReq = [&](const char* pTopicName, int8_t igExists, const char* pSql, const char* pAst,
 | |
|                                const char* pDbName = nullptr, const char* pTbname = nullptr, int8_t withMeta = 0) {
 | |
|     snprintf(expect.name, sizeof(expect.name), "0.%s", pTopicName);
 | |
|     expect.igExists = igExists;
 | |
|     expect.sql = (char*)pSql;
 | |
|     expect.withMeta = withMeta;
 | |
|     if (nullptr != pTbname) {
 | |
|       expect.subType = TOPIC_SUB_TYPE__TABLE;
 | |
|       snprintf(expect.subStbName, sizeof(expect.subStbName), "0.%s.%s", pDbName, pTbname);
 | |
|     } else if (nullptr != pAst) {
 | |
|       expect.subType = TOPIC_SUB_TYPE__COLUMN;
 | |
|       expect.ast = (char*)pAst;
 | |
|     } else {
 | |
|       expect.subType = TOPIC_SUB_TYPE__DB;
 | |
|       snprintf(expect.subDbName, sizeof(expect.subDbName), "0.%s", pDbName);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_TOPIC_STMT);
 | |
|     SCMCreateTopicReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS ==
 | |
|                 tDeserializeSCMCreateTopicReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(std::string(req.name), std::string(expect.name));
 | |
|     ASSERT_EQ(req.igExists, expect.igExists);
 | |
|     ASSERT_EQ(req.subType, expect.subType);
 | |
|     ASSERT_EQ(std::string(req.sql), std::string(expect.sql));
 | |
|     ASSERT_EQ(req.withMeta, expect.withMeta);
 | |
|     switch (expect.subType) {
 | |
|       case TOPIC_SUB_TYPE__DB:
 | |
|         ASSERT_EQ(std::string(req.subDbName), std::string(expect.subDbName));
 | |
|         break;
 | |
|       case TOPIC_SUB_TYPE__TABLE:
 | |
|         ASSERT_EQ(std::string(req.subStbName), std::string(expect.subStbName));
 | |
|         break;
 | |
|       case TOPIC_SUB_TYPE__COLUMN:
 | |
|         ASSERT_NE(req.ast, nullptr);
 | |
|         break;
 | |
|       default:
 | |
|         ASSERT_TRUE(false);
 | |
|     }
 | |
|     tFreeSCMCreateTopicReq(&req);
 | |
|   });
 | |
| 
 | |
|   setCreateTopicReq("tp1", 0, "create topic tp1 as select * from t1", "ast");
 | |
|   run("CREATE TOPIC tp1 AS SELECT * FROM t1");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
|   setCreateTopicReq("tp1", 1, "create topic if not exists tp1 as select ts, ceil(c1) from t1", "ast");
 | |
|   run("CREATE TOPIC IF NOT EXISTS tp1 AS SELECT ts, CEIL(c1) FROM t1");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
|   setCreateTopicReq("tp1", 0, "create topic tp1 as database test", nullptr, "test");
 | |
|   run("CREATE TOPIC tp1 AS DATABASE test");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
|   setCreateTopicReq("tp1", 0, "create topic tp1 with meta as database test", nullptr, "test", nullptr, 1);
 | |
|   run("CREATE TOPIC tp1 WITH META AS DATABASE test");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
|   setCreateTopicReq("tp1", 1, "create topic if not exists tp1 as stable st1", nullptr, "test", "st1");
 | |
|   run("CREATE TOPIC IF NOT EXISTS tp1 AS STABLE st1");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
|   setCreateTopicReq("tp1", 1, "create topic if not exists tp1 with meta as stable st1", nullptr, "test", "st1", 1);
 | |
|   run("CREATE TOPIC IF NOT EXISTS tp1 WITH META AS STABLE st1");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
|   setCreateTopicReq("tp1", 1, "create topic if not exists tp1 as stable st1 where tag1 > 0", nullptr, "test", "st1");
 | |
|   run("CREATE TOPIC IF NOT EXISTS tp1 AS STABLE st1 WHERE tag1 > 0");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
|   setCreateTopicReq("tp1", 1, "create topic if not exists tp1 with meta as stable st1 where tag1 > 0", nullptr, "test", "st1", 1);
 | |
|   run("CREATE TOPIC IF NOT EXISTS tp1 WITH META AS STABLE st1 WHERE tag1 > 0");
 | |
|   clearCreateTopicReq();
 | |
| 
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * CREATE USER use_name PASS password [SYSINFO value]
 | |
|  */
 | |
| TEST_F(ParserInitialCTest, createUser) {
 | |
|   useDb("root", "test");
 | |
| 
 | |
|   SCreateUserReq expect = {0};
 | |
| 
 | |
|   auto clearCreateUserReq = [&]() { memset(&expect, 0, sizeof(SCreateUserReq)); };
 | |
| 
 | |
|   auto setCreateUserReq = [&](const char* pUser, const char* pPass, int8_t sysInfo = 1) {
 | |
|     strcpy(expect.user, pUser);
 | |
|     strcpy(expect.pass, pPass);
 | |
|     expect.createType = 0;
 | |
|     expect.superUser = 0;
 | |
|     expect.sysInfo = sysInfo;
 | |
|     expect.enable = 1;
 | |
|   };
 | |
| 
 | |
|   setCheckDdlFunc([&](const SQuery* pQuery, ParserStage stage) {
 | |
|     ASSERT_EQ(nodeType(pQuery->pRoot), QUERY_NODE_CREATE_USER_STMT);
 | |
|     SCreateUserReq req = {0};
 | |
|     ASSERT_TRUE(TSDB_CODE_SUCCESS == tDeserializeSCreateUserReq(pQuery->pCmdMsg->pMsg, pQuery->pCmdMsg->msgLen, &req));
 | |
| 
 | |
|     ASSERT_EQ(req.createType, expect.createType);
 | |
|     ASSERT_EQ(req.superUser, expect.superUser);
 | |
|     ASSERT_EQ(req.sysInfo, expect.sysInfo);
 | |
|     ASSERT_EQ(req.enable, expect.enable);
 | |
|     ASSERT_EQ(std::string(req.user), std::string(expect.user));
 | |
|     ASSERT_EQ(std::string(req.pass), std::string(expect.pass));
 | |
|   });
 | |
| 
 | |
|   setCreateUserReq("wxy", "123456");
 | |
|   run("CREATE USER wxy PASS '123456'");
 | |
|   clearCreateUserReq();
 | |
| 
 | |
|   setCreateUserReq("wxy1", "a123456", 1);
 | |
|   run("CREATE USER wxy1 PASS 'a123456' SYSINFO 1");
 | |
|   clearCreateUserReq();
 | |
| }
 | |
| 
 | |
| }  // namespace ParserTest
 |