/* * Copyright (c) 2019 TAOS Data, Inc. * * 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 . */ #ifndef TDENGINE_TTOKEN_H #define TDENGINE_TTOKEN_H #ifdef __cplusplus extern "C" { #endif #include "os.h" #include "ttokendef.h" // used to denote the minimum unite in sql parsing typedef struct SToken { uint32_t n; uint32_t type; char *z; } SToken; /** * check if it is a number or not * @param pToken * @return */ #define isNumber(tk) \ ((tk)->type == TK_NK_INTEGER || (tk)->type == TK_NK_FLOAT || (tk)->type == TK_NK_HEX || (tk)->type == TK_NK_BIN) /** * tokenizer for sql string * @param z * @param tokenType * @return */ uint32_t tGetToken(const char *z, uint32_t *tokenType); /** * enhanced tokenizer for sql string. * * @param str * @param i * @param isPrevOptr * @return */ SToken tStrGetToken(const char *str, int32_t *i, bool isPrevOptr, bool *pIgnoreComma); /** * check if it is a keyword or not * @param z * @param len * @return */ bool taosIsKeyWordToken(const char *z, int32_t len); /** * check if it is a token or not * @param pToken * @return token type, if it is not a number, TK_NK_ILLEGAL will return */ static FORCE_INLINE int32_t tGetNumericStringType(const SToken *pToken) { const char *z = pToken->z; int32_t type = TK_NK_ILLEGAL; uint32_t i = 0; for (; i < pToken->n; ++i) { switch (z[i]) { case '+': case '-': { break; } case '.': { /* * handle the the float number with out integer part * .123 * .123e4 */ if (!isdigit(z[i + 1])) { return TK_NK_ILLEGAL; } for (i += 2; isdigit(z[i]); i++) { } if ((z[i] == 'e' || z[i] == 'E') && (isdigit(z[i + 1]) || ((z[i + 1] == '+' || z[i + 1] == '-') && isdigit(z[i + 2])))) { i += 2; while (isdigit(z[i])) { i++; } } type = TK_NK_FLOAT; goto _end; } case '0': { char next = z[i + 1]; if (next == 'b') { // bin number type = TK_NK_BIN; for (i += 2; (z[i] == '0' || z[i] == '1'); ++i) { } goto _end; } else if (next == 'x') { // hex number type = TK_NK_HEX; for (i += 2; isdigit(z[i]) || (z[i] >= 'a' && z[i] <= 'f') || (z[i] >= 'A' && z[i] <= 'F'); ++i) { } goto _end; } } case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': { type = TK_NK_INTEGER; for (; isdigit(z[i]); i++) { } int32_t seg = 0; while (z[i] == '.' && isdigit(z[i + 1])) { i += 2; while (isdigit(z[i])) { i++; } seg++; type = TK_NK_FLOAT; } if (seg > 1) { return TK_NK_ILLEGAL; } if ((z[i] == 'e' || z[i] == 'E') && (isdigit(z[i + 1]) || ((z[i + 1] == '+' || z[i + 1] == '-') && isdigit(z[i + 2])))) { i += 2; while (isdigit(z[i])) { i++; } type = TK_NK_FLOAT; } goto _end; } default: return TK_NK_ILLEGAL; } } _end: return (i < pToken->n) ? TK_NK_ILLEGAL : type; } void taosCleanupKeywordsTable(); #ifdef __cplusplus } #endif #endif // TDENGINE_TTOKEN_H