Merge pull request #9329 from taosdata/feature/3.0_wxy
TD-12450 perfect parser interface
This commit is contained in:
commit
fe12d1d6ee
|
@ -160,6 +160,13 @@ typedef struct SInsertStmtInfo {
|
||||||
const char* sql; // current sql statement position
|
const char* sql; // current sql statement position
|
||||||
} SInsertStmtInfo;
|
} SInsertStmtInfo;
|
||||||
|
|
||||||
|
typedef struct SDclStmtInfo {
|
||||||
|
int16_t nodeType;
|
||||||
|
int16_t msgType;
|
||||||
|
char* pMsg;
|
||||||
|
int32_t msgLen;
|
||||||
|
} SDclStmtInfo;
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -22,20 +22,11 @@ extern "C" {
|
||||||
|
|
||||||
#include "parsenodes.h"
|
#include "parsenodes.h"
|
||||||
|
|
||||||
/**
|
|
||||||
* True will be returned if the input sql string is insert, false otherwise.
|
|
||||||
* @param pStr sql string
|
|
||||||
* @param length length of the sql string
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
bool qIsInsertSql(const char* pStr, size_t length);
|
|
||||||
|
|
||||||
typedef struct SParseContext {
|
typedef struct SParseContext {
|
||||||
SParseBasicCtx ctx;
|
SParseBasicCtx ctx;
|
||||||
void *pRpc;
|
void *pRpc;
|
||||||
struct SCatalog *pCatalog;
|
struct SCatalog *pCatalog;
|
||||||
const SEpSet *pEpSet;
|
const SEpSet *pEpSet;
|
||||||
int64_t id; // query id, generated by uuid generator
|
|
||||||
int8_t schemaAttached; // denote if submit block is built with table schema or not
|
int8_t schemaAttached; // denote if submit block is built with table schema or not
|
||||||
const char *pSql; // sql string
|
const char *pSql; // sql string
|
||||||
size_t sqlLen; // length of the sql string
|
size_t sqlLen; // length of the sql string
|
||||||
|
@ -51,17 +42,9 @@ typedef struct SParseContext {
|
||||||
* @param msg extended error message if exists.
|
* @param msg extended error message if exists.
|
||||||
* @return error code
|
* @return error code
|
||||||
*/
|
*/
|
||||||
int32_t qParseQuerySql(const char* pStr, size_t length, SParseBasicCtx* pParseCtx, int32_t* type, void** pOutput, int32_t* outputLen, char* msg, int32_t msgLen);
|
int32_t qParseQuerySql(SParseContext* pContext, SQueryNode** pQuery);
|
||||||
|
|
||||||
/**
|
bool qIsDclQuery(const SQueryNode* pQuery);
|
||||||
* Parse the insert sql statement.
|
|
||||||
* @param pStr sql string
|
|
||||||
* @param length length of the sql string
|
|
||||||
* @param id operator id, generated by uuid generator.
|
|
||||||
* @param msg extended error message if exists to help avoid the problem in sql statement.
|
|
||||||
* @return data in binary format to submit to vnode directly.
|
|
||||||
*/
|
|
||||||
int32_t qParseInsertSql(SParseContext* pContext, struct SInsertStmtInfo** pInfo);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a normal sql statement to only query tags information to enable that the subscribe client can be aware quickly of the true vgroup ids that
|
* Convert a normal sql statement to only query tags information to enable that the subscribe client can be aware quickly of the true vgroup ids that
|
||||||
|
|
|
@ -144,72 +144,66 @@ TAOS_RES *taos_query_l(TAOS *taos, const char *sql, int sqlLen) {
|
||||||
|
|
||||||
tscDebugL("0x%"PRIx64" SQL: %s", pRequest->requestId, pRequest->sqlstr);
|
tscDebugL("0x%"PRIx64" SQL: %s", pRequest->requestId, pRequest->sqlstr);
|
||||||
|
|
||||||
int32_t code = 0;
|
SParseContext cxt = {
|
||||||
if (qIsInsertSql(pRequest->sqlstr, sqlLen)) {
|
.ctx = {.requestId = pRequest->requestId, .acctId = pTscObj->acctId, .db = getConnectionDB(pTscObj)},
|
||||||
// todo add
|
.pSql = pRequest->sqlstr,
|
||||||
} else {
|
.sqlLen = sqlLen,
|
||||||
int32_t type = 0;
|
.pMsg = pRequest->msgBuf,
|
||||||
void* output = NULL;
|
.msgLen = ERROR_MSG_BUF_DEFAULT_SIZE
|
||||||
int32_t outputLen = 0;
|
};
|
||||||
|
SQueryNode* pQuery = NULL;
|
||||||
|
int32_t code = qParseQuerySql(&cxt, &pQuery);
|
||||||
|
if (qIsDclQuery(pQuery)) {
|
||||||
|
SDclStmtInfo* pDcl = (SDclStmtInfo*)pQuery;
|
||||||
|
pRequest->type = pDcl->msgType;
|
||||||
|
pRequest->body.requestMsg = (SReqMsgInfo){.pMsg = pDcl->pMsg, .len = pDcl->msgLen};
|
||||||
|
|
||||||
SParseBasicCtx c = {.requestId = pRequest->requestId, .acctId = pTscObj->acctId, .db = getConnectionDB(pTscObj)};
|
SRequestMsgBody body = buildRequestMsgImpl(pRequest);
|
||||||
code = qParseQuerySql(pRequest->sqlstr, sqlLen, &c, &type, &output, &outputLen, pRequest->msgBuf, ERROR_MSG_BUF_DEFAULT_SIZE);
|
SEpSet* pEpSet = &pTscObj->pAppInfo->mgmtEp.epSet;
|
||||||
if (type == TSDB_MSG_TYPE_CREATE_USER || type == TSDB_MSG_TYPE_SHOW || type == TSDB_MSG_TYPE_DROP_USER ||
|
|
||||||
type == TSDB_MSG_TYPE_DROP_ACCT || type == TSDB_MSG_TYPE_CREATE_DB || type == TSDB_MSG_TYPE_CREATE_ACCT ||
|
|
||||||
type == TSDB_MSG_TYPE_CREATE_TABLE || type == TSDB_MSG_TYPE_CREATE_STB || type == TSDB_MSG_TYPE_USE_DB ||
|
|
||||||
type == TSDB_MSG_TYPE_DROP_DB || type == TSDB_MSG_TYPE_DROP_STB) {
|
|
||||||
pRequest->type = type;
|
|
||||||
pRequest->body.requestMsg = (SReqMsgInfo){.pMsg = output, .len = outputLen};
|
|
||||||
|
|
||||||
SRequestMsgBody body = buildRequestMsgImpl(pRequest);
|
if (pDcl->msgType == TSDB_MSG_TYPE_CREATE_TABLE) {
|
||||||
SEpSet* pEpSet = &pTscObj->pAppInfo->mgmtEp.epSet;
|
struct SCatalog* pCatalog = NULL;
|
||||||
|
|
||||||
if (type == TSDB_MSG_TYPE_CREATE_TABLE) {
|
char buf[12] = {0};
|
||||||
struct SCatalog* pCatalog = NULL;
|
sprintf(buf, "%d", pTscObj->pAppInfo->clusterId);
|
||||||
|
code = catalogGetHandle(buf, &pCatalog);
|
||||||
char buf[12] = {0};
|
if (code != 0) {
|
||||||
sprintf(buf, "%d", pTscObj->pAppInfo->clusterId);
|
pRequest->code = code;
|
||||||
code = catalogGetHandle(buf, &pCatalog);
|
return pRequest;
|
||||||
if (code != 0) {
|
|
||||||
pRequest->code = code;
|
|
||||||
return pRequest;
|
|
||||||
}
|
|
||||||
|
|
||||||
SCreateTableMsg* pMsg = body.msgInfo.pMsg;
|
|
||||||
|
|
||||||
SName t = {0};
|
|
||||||
tNameFromString(&t, pMsg->name, T_NAME_ACCT|T_NAME_DB|T_NAME_TABLE);
|
|
||||||
|
|
||||||
char db[TSDB_DB_NAME_LEN + TS_PATH_DELIMITER_LEN + TSDB_ACCT_ID_LEN] = {0};
|
|
||||||
tNameGetFullDbName(&t, db);
|
|
||||||
|
|
||||||
SVgroupInfo info = {0};
|
|
||||||
catalogGetTableHashVgroup(pCatalog, pTscObj->pTransporter, pEpSet, db, tNameGetTableName(&t), &info);
|
|
||||||
|
|
||||||
int64_t transporterId = 0;
|
|
||||||
SEpSet ep = {0};
|
|
||||||
ep.inUse = info.inUse;
|
|
||||||
ep.numOfEps = info.numOfEps;
|
|
||||||
for(int32_t i = 0; i < ep.numOfEps; ++i) {
|
|
||||||
ep.port[i] = info.epAddr[i].port;
|
|
||||||
tstrncpy(ep.fqdn[i], info.epAddr[i].fqdn, tListLen(ep.fqdn[i]));
|
|
||||||
}
|
|
||||||
|
|
||||||
sendMsgToServer(pTscObj->pTransporter, &ep, &body, &transporterId);
|
|
||||||
} else {
|
|
||||||
int64_t transporterId = 0;
|
|
||||||
sendMsgToServer(pTscObj->pTransporter, pEpSet, &body, &transporterId);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
tsem_wait(&pRequest->body.rspSem);
|
SCreateTableMsg* pMsg = body.msgInfo.pMsg;
|
||||||
destroyRequestMsgBody(&body);
|
|
||||||
|
SName t = {0};
|
||||||
|
tNameFromString(&t, pMsg->name, T_NAME_ACCT|T_NAME_DB|T_NAME_TABLE);
|
||||||
|
|
||||||
|
char db[TSDB_DB_NAME_LEN + TS_PATH_DELIMITER_LEN + TSDB_ACCT_ID_LEN] = {0};
|
||||||
|
tNameGetFullDbName(&t, db);
|
||||||
|
|
||||||
|
SVgroupInfo info = {0};
|
||||||
|
catalogGetTableHashVgroup(pCatalog, pTscObj->pTransporter, pEpSet, db, tNameGetTableName(&t), &info);
|
||||||
|
|
||||||
|
int64_t transporterId = 0;
|
||||||
|
SEpSet ep = {0};
|
||||||
|
ep.inUse = info.inUse;
|
||||||
|
ep.numOfEps = info.numOfEps;
|
||||||
|
for(int32_t i = 0; i < ep.numOfEps; ++i) {
|
||||||
|
ep.port[i] = info.epAddr[i].port;
|
||||||
|
tstrncpy(ep.fqdn[i], info.epAddr[i].fqdn, tListLen(ep.fqdn[i]));
|
||||||
|
}
|
||||||
|
|
||||||
|
sendMsgToServer(pTscObj->pTransporter, &ep, &body, &transporterId);
|
||||||
} else {
|
} else {
|
||||||
assert(0);
|
int64_t transporterId = 0;
|
||||||
|
sendMsgToServer(pTscObj->pTransporter, pEpSet, &body, &transporterId);
|
||||||
}
|
}
|
||||||
|
|
||||||
tfree(c.db);
|
tsem_wait(&pRequest->body.rspSem);
|
||||||
|
destroyRequestMsgBody(&body);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tfree(cxt.ctx.db);
|
||||||
|
|
||||||
if (code != TSDB_CODE_SUCCESS) {
|
if (code != TSDB_CODE_SUCCESS) {
|
||||||
pRequest->code = code;
|
pRequest->code = code;
|
||||||
return pRequest;
|
return pRequest;
|
||||||
|
|
|
@ -68,7 +68,7 @@ int32_t qParserValidateSqlNode(struct SCatalog* pCatalog, SSqlInfo* pSqlInfo, SQ
|
||||||
* @param type
|
* @param type
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void** output, int32_t* outputLen, int32_t* type, char* msgBuf, int32_t msgBufLen);
|
int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, SDclStmtInfo* pDcl, char* msgBuf, int32_t msgBufLen);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Evaluate the numeric and timestamp arithmetic expression in the WHERE clause.
|
* Evaluate the numeric and timestamp arithmetic expression in the WHERE clause.
|
||||||
|
|
|
@ -4306,7 +4306,7 @@ int32_t doCheckForCreateTable(SSqlInfo* pInfo, SMsgBuf* pMsgBuf) {
|
||||||
return TSDB_CODE_SUCCESS;
|
return TSDB_CODE_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void** output, int32_t* outputLen, int32_t* type, char* msgBuf, int32_t msgBufLen) {
|
int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, SDclStmtInfo* pDcl, char* msgBuf, int32_t msgBufLen) {
|
||||||
int32_t code = 0;
|
int32_t code = 0;
|
||||||
|
|
||||||
SMsgBuf m = {.buf = msgBuf, .len = msgBufLen};
|
SMsgBuf m = {.buf = msgBuf, .len = msgBufLen};
|
||||||
|
@ -4357,8 +4357,8 @@ int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void**
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
*output = buildUserManipulationMsg(pInfo, outputLen, pCtx->requestId, msgBuf, msgBufLen);
|
pDcl->pMsg = (char*)buildUserManipulationMsg(pInfo, &pDcl->msgLen, pCtx->requestId, msgBuf, msgBufLen);
|
||||||
*type = (pInfo->type == TSDB_SQL_CREATE_USER)? TSDB_MSG_TYPE_CREATE_USER:TSDB_MSG_TYPE_ALTER_USER;
|
pDcl->msgType = (pInfo->type == TSDB_SQL_CREATE_USER)? TSDB_MSG_TYPE_CREATE_USER:TSDB_MSG_TYPE_ALTER_USER;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4394,21 +4394,21 @@ int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void**
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
*output = buildAcctManipulationMsg(pInfo, outputLen, pCtx->requestId, msgBuf, msgBufLen);
|
pDcl->pMsg = (char*)buildAcctManipulationMsg(pInfo, &pDcl->msgLen, pCtx->requestId, msgBuf, msgBufLen);
|
||||||
*type = (pInfo->type == TSDB_SQL_CREATE_ACCT)? TSDB_MSG_TYPE_CREATE_ACCT:TSDB_MSG_TYPE_ALTER_ACCT;
|
pDcl->msgType = (pInfo->type == TSDB_SQL_CREATE_ACCT)? TSDB_MSG_TYPE_CREATE_ACCT:TSDB_MSG_TYPE_ALTER_ACCT;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case TSDB_SQL_DROP_ACCT:
|
case TSDB_SQL_DROP_ACCT:
|
||||||
case TSDB_SQL_DROP_USER: {
|
case TSDB_SQL_DROP_USER: {
|
||||||
*output = buildDropUserMsg(pInfo, outputLen, pCtx->requestId, msgBuf, msgBufLen);
|
pDcl->pMsg = (char*)buildDropUserMsg(pInfo, &pDcl->msgLen, pCtx->requestId, msgBuf, msgBufLen);
|
||||||
*type = (pInfo->type == TSDB_SQL_DROP_ACCT)? TSDB_MSG_TYPE_DROP_ACCT:TSDB_MSG_TYPE_DROP_USER;
|
pDcl->msgType = (pInfo->type == TSDB_SQL_DROP_ACCT)? TSDB_MSG_TYPE_DROP_ACCT:TSDB_MSG_TYPE_DROP_USER;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case TSDB_SQL_SHOW: {
|
case TSDB_SQL_SHOW: {
|
||||||
code = setShowInfo(&pInfo->pMiscInfo->showOpt, pCtx, output, outputLen, pMsgBuf);
|
code = setShowInfo(&pInfo->pMiscInfo->showOpt, pCtx, (void**)&pDcl->pMsg, &pDcl->msgLen, pMsgBuf);
|
||||||
*type = TSDB_MSG_TYPE_SHOW;
|
pDcl->msgType = TSDB_MSG_TYPE_SHOW;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4429,9 +4429,9 @@ int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void**
|
||||||
SUseDbMsg *pUseDbMsg = (SUseDbMsg *) calloc(1, sizeof(SUseDbMsg));
|
SUseDbMsg *pUseDbMsg = (SUseDbMsg *) calloc(1, sizeof(SUseDbMsg));
|
||||||
tNameExtractFullName(&n, pUseDbMsg->db);
|
tNameExtractFullName(&n, pUseDbMsg->db);
|
||||||
|
|
||||||
*output = pUseDbMsg;
|
pDcl->pMsg = (char*)pUseDbMsg;
|
||||||
*outputLen = sizeof(SUseDbMsg);
|
pDcl->msgLen = sizeof(SUseDbMsg);
|
||||||
*type = TSDB_MSG_TYPE_USE_DB;
|
pDcl->msgType = TSDB_MSG_TYPE_USE_DB;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4457,9 +4457,11 @@ int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void**
|
||||||
return TSDB_CODE_TSC_INVALID_OPERATION;
|
return TSDB_CODE_TSC_INVALID_OPERATION;
|
||||||
}
|
}
|
||||||
|
|
||||||
*output = pCreateMsg;
|
strncpy(pCreateMsg->db, token.z, token.n);
|
||||||
*outputLen = sizeof(SCreateDbMsg);
|
|
||||||
*type = (pInfo->type == TSDB_SQL_CREATE_DB)? TSDB_MSG_TYPE_CREATE_DB:TSDB_MSG_TYPE_ALTER_DB;
|
pDcl->pMsg = (char*)pCreateMsg;
|
||||||
|
pDcl->msgLen = sizeof(SCreateDbMsg);
|
||||||
|
pDcl->msgType = (pInfo->type == TSDB_SQL_CREATE_DB)? TSDB_MSG_TYPE_CREATE_DB:TSDB_MSG_TYPE_ALTER_DB;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4481,9 +4483,9 @@ int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void**
|
||||||
pDropDbMsg->ignoreNotExists = pInfo->pMiscInfo->existsCheck ? 1 : 0;
|
pDropDbMsg->ignoreNotExists = pInfo->pMiscInfo->existsCheck ? 1 : 0;
|
||||||
assert(code == TSDB_CODE_SUCCESS && name.type == TSDB_DB_NAME_T);
|
assert(code == TSDB_CODE_SUCCESS && name.type == TSDB_DB_NAME_T);
|
||||||
|
|
||||||
*type = TSDB_MSG_TYPE_DROP_DB;
|
pDcl->msgType = TSDB_MSG_TYPE_DROP_DB;
|
||||||
*outputLen = sizeof(SDropDbMsg);
|
pDcl->msgLen = sizeof(SDropDbMsg);
|
||||||
*output = pDropDbMsg;
|
pDcl->pMsg = (char*)pDropDbMsg;
|
||||||
return TSDB_CODE_SUCCESS;
|
return TSDB_CODE_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4494,9 +4496,8 @@ int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void**
|
||||||
if ((code = doCheckForCreateTable(pInfo, pMsgBuf)) != TSDB_CODE_SUCCESS) {
|
if ((code = doCheckForCreateTable(pInfo, pMsgBuf)) != TSDB_CODE_SUCCESS) {
|
||||||
return code;
|
return code;
|
||||||
}
|
}
|
||||||
|
pDcl->pMsg = (char*)buildCreateTableMsg(pCreateTable, &pDcl->msgLen, pCtx, pMsgBuf);
|
||||||
*output = buildCreateTableMsg(pCreateTable, outputLen, pCtx, pMsgBuf);
|
pDcl->msgType = (pCreateTable->type == TSQL_CREATE_TABLE)? TSDB_MSG_TYPE_CREATE_TABLE:TSDB_MSG_TYPE_CREATE_STB;
|
||||||
*type = (pCreateTable->type == TSQL_CREATE_TABLE)? TSDB_MSG_TYPE_CREATE_TABLE:TSDB_MSG_TYPE_CREATE_STB;
|
|
||||||
} else if (pCreateTable->type == TSQL_CREATE_CTABLE) {
|
} else if (pCreateTable->type == TSQL_CREATE_CTABLE) {
|
||||||
// if ((code = doCheckForCreateFromStable(pSql, pInfo)) != TSDB_CODE_SUCCESS) {
|
// if ((code = doCheckForCreateFromStable(pSql, pInfo)) != TSDB_CODE_SUCCESS) {
|
||||||
// return code;
|
// return code;
|
||||||
|
@ -4511,12 +4512,12 @@ int32_t qParserValidateDclSqlNode(SSqlInfo* pInfo, SParseBasicCtx* pCtx, void**
|
||||||
}
|
}
|
||||||
|
|
||||||
case TSDB_SQL_DROP_TABLE: {
|
case TSDB_SQL_DROP_TABLE: {
|
||||||
*output = buildDropTableMsg(pInfo, outputLen, pCtx, pMsgBuf);
|
pDcl->pMsg = (char*)buildDropTableMsg(pInfo, &pDcl->msgLen, pCtx, pMsgBuf);
|
||||||
if (output == NULL) {
|
if (pDcl->pMsg == NULL) {
|
||||||
return terrno;
|
return terrno;
|
||||||
}
|
}
|
||||||
|
|
||||||
*type = TSDB_MSG_TYPE_DROP_STB;
|
pDcl->msgType = TSDB_MSG_TYPE_DROP_STB;
|
||||||
return TSDB_CODE_SUCCESS;
|
return TSDB_CODE_SUCCESS;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -909,6 +909,7 @@ int32_t parseInsertSql(SParseContext* pContext, SInsertStmtInfo** pInfo) {
|
||||||
}
|
}
|
||||||
|
|
||||||
*pInfo = context.pOutput;
|
*pInfo = context.pOutput;
|
||||||
|
context.pOutput->nodeType = TSDB_SQL_INSERT;
|
||||||
context.pOutput->schemaAttache = pContext->schemaAttached;
|
context.pOutput->schemaAttache = pContext->schemaAttached;
|
||||||
context.pOutput->payloadType = PAYLOAD_TYPE_KV;
|
context.pOutput->payloadType = PAYLOAD_TYPE_KV;
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
#include "function.h"
|
#include "function.h"
|
||||||
#include "insertParser.h"
|
#include "insertParser.h"
|
||||||
|
|
||||||
bool qIsInsertSql(const char* pStr, size_t length) {
|
bool isInsertSql(const char* pStr, size_t length) {
|
||||||
int32_t index = 0;
|
int32_t index = 0;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
|
@ -31,18 +31,28 @@ bool qIsInsertSql(const char* pStr, size_t length) {
|
||||||
} while (1);
|
} while (1);
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t qParseQuerySql(const char* pStr, size_t length, SParseBasicCtx* pParseCtx, int32_t *type, void** pOutput, int32_t* outputLen, char* msg, int32_t msgLen) {
|
bool qIsDclQuery(const SQueryNode* pQuery) {
|
||||||
SSqlInfo info = doGenerateAST(pStr);
|
return TSDB_SQL_INSERT != pQuery->type && TSDB_SQL_SELECT != pQuery->type;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t parseQuerySql(SParseContext* pCxt, SQueryNode** pQuery) {
|
||||||
|
SSqlInfo info = doGenerateAST(pCxt->pSql);
|
||||||
if (!info.valid) {
|
if (!info.valid) {
|
||||||
strncpy(msg, info.msg, msgLen);
|
strncpy(pCxt->pMsg, info.msg, pCxt->msgLen);
|
||||||
terrno = TSDB_CODE_TSC_SQL_SYNTAX_ERROR;
|
terrno = TSDB_CODE_TSC_SQL_SYNTAX_ERROR;
|
||||||
return terrno;
|
return terrno;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isDqlSqlStatement(&info)) {
|
if (!isDqlSqlStatement(&info)) {
|
||||||
int32_t code = qParserValidateDclSqlNode(&info, pParseCtx, pOutput, outputLen, type, msg, msgLen);
|
SDclStmtInfo* pDcl = calloc(1, sizeof(SQueryStmtInfo));
|
||||||
|
if (NULL == pDcl) {
|
||||||
|
terrno = TSDB_CODE_TSC_OUT_OF_MEMORY; // set correct error code.
|
||||||
|
return terrno;
|
||||||
|
}
|
||||||
|
pDcl->nodeType = info.type;
|
||||||
|
int32_t code = qParserValidateDclSqlNode(&info, &pCxt->ctx, pDcl, pCxt->pMsg, pCxt->msgLen);
|
||||||
if (code == TSDB_CODE_SUCCESS) {
|
if (code == TSDB_CODE_SUCCESS) {
|
||||||
// do nothing
|
*pQuery = (SQueryNode*)pDcl;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
SQueryStmtInfo* pQueryInfo = calloc(1, sizeof(SQueryStmtInfo));
|
SQueryStmtInfo* pQueryInfo = calloc(1, sizeof(SQueryStmtInfo));
|
||||||
|
@ -53,9 +63,9 @@ int32_t qParseQuerySql(const char* pStr, size_t length, SParseBasicCtx* pParseCt
|
||||||
|
|
||||||
struct SCatalog* pCatalog = NULL;
|
struct SCatalog* pCatalog = NULL;
|
||||||
int32_t code = catalogGetHandle(NULL, &pCatalog);
|
int32_t code = catalogGetHandle(NULL, &pCatalog);
|
||||||
code = qParserValidateSqlNode(pCatalog, &info, pQueryInfo, pParseCtx->requestId, msg, msgLen);
|
code = qParserValidateSqlNode(pCatalog, &info, pQueryInfo, pCxt->ctx.requestId, pCxt->pMsg, pCxt->msgLen);
|
||||||
if (code == TSDB_CODE_SUCCESS) {
|
if (code == TSDB_CODE_SUCCESS) {
|
||||||
*pOutput = pQueryInfo;
|
*pQuery = (SQueryNode*)pQueryInfo;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,8 +73,12 @@ int32_t qParseQuerySql(const char* pStr, size_t length, SParseBasicCtx* pParseCt
|
||||||
return TSDB_CODE_SUCCESS;
|
return TSDB_CODE_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t qParseInsertSql(SParseContext* pContext, SInsertStmtInfo** pInfo) {
|
int32_t qParseQuerySql(SParseContext* pCxt, SQueryNode** pQuery) {
|
||||||
return parseInsertSql(pContext, pInfo);
|
if (isInsertSql(pCxt->pSql, pCxt->sqlLen)) {
|
||||||
|
return parseInsertSql(pCxt, (SInsertStmtInfo**)pQuery);
|
||||||
|
} else {
|
||||||
|
return parseQuerySql(pCxt, pQuery);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t qParserConvertSql(const char* pStr, size_t length, char** pConvertSql) {
|
int32_t qParserConvertSql(const char* pStr, size_t length, char** pConvertSql) {
|
||||||
|
|
|
@ -23,20 +23,20 @@
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
void generateTestT1(MockCatalogService* mcs) {
|
void generateTestT1(MockCatalogService* mcs) {
|
||||||
ITableBuilder& builder = mcs->createTableBuilder("root.test", "t1", TSDB_NORMAL_TABLE, 3)
|
ITableBuilder& builder = mcs->createTableBuilder("test", "t1", TSDB_NORMAL_TABLE, 3)
|
||||||
.setPrecision(TSDB_TIME_PRECISION_MILLI).setVgid(1).addColumn("ts", TSDB_DATA_TYPE_TIMESTAMP)
|
.setPrecision(TSDB_TIME_PRECISION_MILLI).setVgid(1).addColumn("ts", TSDB_DATA_TYPE_TIMESTAMP)
|
||||||
.addColumn("c1", TSDB_DATA_TYPE_INT).addColumn("c2", TSDB_DATA_TYPE_BINARY, 20);
|
.addColumn("c1", TSDB_DATA_TYPE_INT).addColumn("c2", TSDB_DATA_TYPE_BINARY, 20);
|
||||||
builder.done();
|
builder.done();
|
||||||
}
|
}
|
||||||
|
|
||||||
void generateTestST1(MockCatalogService* mcs) {
|
void generateTestST1(MockCatalogService* mcs) {
|
||||||
ITableBuilder& builder = mcs->createTableBuilder("root.test", "st1", TSDB_SUPER_TABLE, 3, 2)
|
ITableBuilder& builder = mcs->createTableBuilder("test", "st1", TSDB_SUPER_TABLE, 3, 2)
|
||||||
.setPrecision(TSDB_TIME_PRECISION_MILLI).addColumn("ts", TSDB_DATA_TYPE_TIMESTAMP)
|
.setPrecision(TSDB_TIME_PRECISION_MILLI).addColumn("ts", TSDB_DATA_TYPE_TIMESTAMP)
|
||||||
.addTag("tag1", TSDB_DATA_TYPE_INT).addTag("tag2", TSDB_DATA_TYPE_BINARY, 20)
|
.addTag("tag1", TSDB_DATA_TYPE_INT).addTag("tag2", TSDB_DATA_TYPE_BINARY, 20)
|
||||||
.addColumn("c1", TSDB_DATA_TYPE_INT).addColumn("c2", TSDB_DATA_TYPE_BINARY, 20);
|
.addColumn("c1", TSDB_DATA_TYPE_INT).addColumn("c2", TSDB_DATA_TYPE_BINARY, 20);
|
||||||
builder.done();
|
builder.done();
|
||||||
mcs->createSubTable("root.test", "st1", "st1s1", 1);
|
mcs->createSubTable("test", "st1", "st1s1", 1);
|
||||||
mcs->createSubTable("root.test", "st1", "st1s2", 2);
|
mcs->createSubTable("test", "st1", "st1s2", 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -94,9 +94,9 @@ public:
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t catalogGetTableMeta(const char* pDBName, const char* pTableName, STableMeta** pTableMeta) const {
|
int32_t catalogGetTableMeta(const char* pDbFullName, const char* pTableName, STableMeta** pTableMeta) const {
|
||||||
std::unique_ptr<STableMeta> table;
|
std::unique_ptr<STableMeta> table;
|
||||||
int32_t code = copyTableSchemaMeta(pDBName, pTableName, &table);
|
int32_t code = copyTableSchemaMeta(toDbname(pDbFullName), pTableName, &table);
|
||||||
if (TSDB_CODE_SUCCESS != code) {
|
if (TSDB_CODE_SUCCESS != code) {
|
||||||
return code;
|
return code;
|
||||||
}
|
}
|
||||||
|
@ -104,7 +104,7 @@ public:
|
||||||
return TSDB_CODE_SUCCESS;
|
return TSDB_CODE_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t catalogGetTableHashVgroup(const char* pDBName, const char* pTableName, SVgroupInfo* vgInfo) const {
|
int32_t catalogGetTableHashVgroup(const char* pDbFullName, const char* pTableName, SVgroupInfo* vgInfo) const {
|
||||||
// todo
|
// todo
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -195,6 +195,14 @@ private:
|
||||||
typedef std::map<std::string, std::shared_ptr<MockTableMeta>> TableMetaCache;
|
typedef std::map<std::string, std::shared_ptr<MockTableMeta>> TableMetaCache;
|
||||||
typedef std::map<std::string, TableMetaCache> DbMetaCache;
|
typedef std::map<std::string, TableMetaCache> DbMetaCache;
|
||||||
|
|
||||||
|
std::string toDbname(const std::string& dbFullName) const {
|
||||||
|
std::string::size_type n = dbFullName.find(".");
|
||||||
|
if (n == std::string::npos) {
|
||||||
|
return dbFullName;
|
||||||
|
}
|
||||||
|
return dbFullName.substr(n + 1);
|
||||||
|
}
|
||||||
|
|
||||||
std::string ttToString(int8_t tableType) const {
|
std::string ttToString(int8_t tableType) const {
|
||||||
switch (tableType) {
|
switch (tableType) {
|
||||||
case TSDB_SUPER_TABLE:
|
case TSDB_SUPER_TABLE:
|
||||||
|
|
|
@ -714,12 +714,9 @@ TEST(testCase, show_user_Test) {
|
||||||
SSqlInfo info1 = doGenerateAST(sql1);
|
SSqlInfo info1 = doGenerateAST(sql1);
|
||||||
ASSERT_EQ(info1.valid, true);
|
ASSERT_EQ(info1.valid, true);
|
||||||
|
|
||||||
void* output = NULL;
|
SDclStmtInfo output;
|
||||||
int32_t type = 0;
|
|
||||||
int32_t len = 0;
|
|
||||||
|
|
||||||
SParseBasicCtx ct= {.db= "abc", .acctId = 1, .requestId = 1};
|
SParseBasicCtx ct= {.db= "abc", .acctId = 1, .requestId = 1};
|
||||||
int32_t code = qParserValidateDclSqlNode(&info1, &ct, &output, &len, &type, msg, buf.len);
|
int32_t code = qParserValidateDclSqlNode(&info1, &ct, &output, msg, buf.len);
|
||||||
ASSERT_EQ(code, 0);
|
ASSERT_EQ(code, 0);
|
||||||
|
|
||||||
// convert the show command to be the select query
|
// convert the show command to be the select query
|
||||||
|
@ -738,12 +735,9 @@ TEST(testCase, create_user_Test) {
|
||||||
ASSERT_EQ(info1.valid, true);
|
ASSERT_EQ(info1.valid, true);
|
||||||
ASSERT_EQ(isDclSqlStatement(&info1), true);
|
ASSERT_EQ(isDclSqlStatement(&info1), true);
|
||||||
|
|
||||||
void* output = NULL;
|
SDclStmtInfo output;
|
||||||
int32_t type = 0;
|
|
||||||
int32_t len = 0;
|
|
||||||
|
|
||||||
SParseBasicCtx ct= {.db= "abc", .acctId = 1, .requestId = 1};
|
SParseBasicCtx ct= {.db= "abc", .acctId = 1, .requestId = 1};
|
||||||
int32_t code = qParserValidateDclSqlNode(&info1, &ct, &output, &len, &type, msg, buf.len);
|
int32_t code = qParserValidateDclSqlNode(&info1, &ct, &output, msg, buf.len);
|
||||||
ASSERT_EQ(code, 0);
|
ASSERT_EQ(code, 0);
|
||||||
|
|
||||||
destroySqlInfo(&info1);
|
destroySqlInfo(&info1);
|
||||||
|
|
|
@ -845,7 +845,7 @@ int32_t subPlanToString(const SSubplan* subplan, char** str, int32_t* len) {
|
||||||
return TSDB_CODE_FAILED;
|
return TSDB_CODE_FAILED;
|
||||||
}
|
}
|
||||||
*str = cJSON_Print(json);
|
*str = cJSON_Print(json);
|
||||||
*len = strlen(*str);
|
*len = strlen(*str) + 1;
|
||||||
return TSDB_CODE_SUCCESS;
|
return TSDB_CODE_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,10 +33,6 @@ protected:
|
||||||
void pushScan(const string& db, const string& table, int32_t scanOp) {
|
void pushScan(const string& db, const string& table, int32_t scanOp) {
|
||||||
shared_ptr<MockTableMeta> meta = mockCatalogService->getTableMeta(db, table);
|
shared_ptr<MockTableMeta> meta = mockCatalogService->getTableMeta(db, table);
|
||||||
EXPECT_TRUE(meta);
|
EXPECT_TRUE(meta);
|
||||||
// typedef struct SQueryPlanNode {
|
|
||||||
// SArray *pExpr; // the query functions or sql aggregations
|
|
||||||
// int32_t numOfExpr; // number of result columns, which is also the number of pExprs
|
|
||||||
// } SQueryPlanNode;
|
|
||||||
unique_ptr<SQueryPlanNode> scan((SQueryPlanNode*)calloc(1, sizeof(SQueryPlanNode)));
|
unique_ptr<SQueryPlanNode> scan((SQueryPlanNode*)calloc(1, sizeof(SQueryPlanNode)));
|
||||||
scan->info.type = scanOp;
|
scan->info.type = scanOp;
|
||||||
scan->numOfCols = meta->schema->tableInfo.numOfColumns;
|
scan->numOfCols = meta->schema->tableInfo.numOfColumns;
|
||||||
|
@ -54,6 +50,21 @@ protected:
|
||||||
return code;
|
return code;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int32_t run(const string& db, const string& sql) {
|
||||||
|
SParseContext cxt;
|
||||||
|
buildParseContext(db, sql, &cxt);
|
||||||
|
SQueryNode* query;
|
||||||
|
int32_t code = qParseQuerySql(&cxt, &query);
|
||||||
|
if (TSDB_CODE_SUCCESS != code) {
|
||||||
|
cout << "error no:" << code << ", msg:" << cxt.pMsg << endl;
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
SQueryDag* dag = nullptr;
|
||||||
|
code = qCreateQueryDag(query, nullptr, &dag);
|
||||||
|
dag_.reset(dag);
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
void explain() {
|
void explain() {
|
||||||
size_t level = taosArrayGetSize(dag_->pSubplans);
|
size_t level = taosArrayGetSize(dag_->pSubplans);
|
||||||
for (size_t i = 0; i < level; ++i) {
|
for (size_t i = 0; i < level; ++i) {
|
||||||
|
@ -64,7 +75,8 @@ protected:
|
||||||
std::cout << "no " << j << ":" << std::endl;
|
std::cout << "no " << j << ":" << std::endl;
|
||||||
int32_t len = 0;
|
int32_t len = 0;
|
||||||
char* str = nullptr;
|
char* str = nullptr;
|
||||||
ASSERT_EQ (TSDB_CODE_SUCCESS, qSubPlanToString((const SSubplan*)taosArrayGetP(subplans, j), &str, &len));
|
ASSERT_EQ(TSDB_CODE_SUCCESS, qSubPlanToString((const SSubplan*)taosArrayGetP(subplans, j), &str, &len));
|
||||||
|
std::cout << "len:" << len << std::endl;
|
||||||
std::cout << str << std::endl;
|
std::cout << str << std::endl;
|
||||||
free(str);
|
free(str);
|
||||||
}
|
}
|
||||||
|
@ -108,6 +120,25 @@ private:
|
||||||
return info;
|
return info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void buildParseContext(const string& db, const string& sql, SParseContext* pCxt) {
|
||||||
|
static string _db;
|
||||||
|
static string _sql;
|
||||||
|
static const int32_t _msgMaxLen = 4096;
|
||||||
|
static char _msg[_msgMaxLen];
|
||||||
|
|
||||||
|
_db = db;
|
||||||
|
_sql = sql;
|
||||||
|
memset(_msg, 0, _msgMaxLen);
|
||||||
|
|
||||||
|
pCxt->ctx.acctId = 1;
|
||||||
|
pCxt->ctx.db = _db.c_str();
|
||||||
|
pCxt->ctx.requestId = 1;
|
||||||
|
pCxt->pSql = _sql.c_str();
|
||||||
|
pCxt->sqlLen = _sql.length();
|
||||||
|
pCxt->pMsg = _msg;
|
||||||
|
pCxt->msgLen = _msgMaxLen;
|
||||||
|
}
|
||||||
|
|
||||||
shared_ptr<MockTableMeta> meta_;
|
shared_ptr<MockTableMeta> meta_;
|
||||||
unique_ptr<SQueryPlanNode> logicPlan_;
|
unique_ptr<SQueryPlanNode> logicPlan_;
|
||||||
unique_ptr<SQueryDag> dag_;
|
unique_ptr<SQueryDag> dag_;
|
||||||
|
@ -115,7 +146,7 @@ private:
|
||||||
|
|
||||||
// select * from table
|
// select * from table
|
||||||
TEST_F(PhyPlanTest, tableScanTest) {
|
TEST_F(PhyPlanTest, tableScanTest) {
|
||||||
pushScan("root.test", "t1", QNODE_TABLESCAN);
|
pushScan("test", "t1", QNODE_TABLESCAN);
|
||||||
ASSERT_EQ(run(), TSDB_CODE_SUCCESS);
|
ASSERT_EQ(run(), TSDB_CODE_SUCCESS);
|
||||||
explain();
|
explain();
|
||||||
SQueryDag* dag = reslut();
|
SQueryDag* dag = reslut();
|
||||||
|
@ -124,9 +155,17 @@ TEST_F(PhyPlanTest, tableScanTest) {
|
||||||
|
|
||||||
// select * from supertable
|
// select * from supertable
|
||||||
TEST_F(PhyPlanTest, superTableScanTest) {
|
TEST_F(PhyPlanTest, superTableScanTest) {
|
||||||
pushScan("root.test", "st1", QNODE_TABLESCAN);
|
pushScan("test", "st1", QNODE_TABLESCAN);
|
||||||
ASSERT_EQ(run(), TSDB_CODE_SUCCESS);
|
ASSERT_EQ(run(), TSDB_CODE_SUCCESS);
|
||||||
explain();
|
explain();
|
||||||
SQueryDag* dag = reslut();
|
SQueryDag* dag = reslut();
|
||||||
// todo check
|
// todo check
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// insert into t values(...)
|
||||||
|
TEST_F(PhyPlanTest, insertTest) {
|
||||||
|
ASSERT_EQ(run("test", "insert into t1 values (now, 1, \"beijing\")"), TSDB_CODE_SUCCESS);
|
||||||
|
explain();
|
||||||
|
SQueryDag* dag = reslut();
|
||||||
|
// todo check
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue