2011-08-18 18:48:00 +00:00
|
|
|
#pragma once
|
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Parsers/IParserBase.h>
|
|
|
|
#include <Parsers/ExpressionElementParsers.h>
|
|
|
|
#include <Parsers/ExpressionListParsers.h>
|
|
|
|
#include <Parsers/ASTNameTypePair.h>
|
|
|
|
#include <Parsers/ASTColumnDeclaration.h>
|
|
|
|
#include <Parsers/ASTIdentifier.h>
|
|
|
|
#include <Parsers/CommonParsers.h>
|
|
|
|
#include <Common/typeid_cast.h>
|
2014-10-07 09:09:59 +00:00
|
|
|
#include <Poco/String.h>
|
2011-08-18 18:48:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2017-05-27 17:29:55 +00:00
|
|
|
/** A nested table. For example, Nested(UInt32 CounterID, FixedString(2) UserAgentMajor)
|
2013-07-11 17:35:56 +00:00
|
|
|
*/
|
|
|
|
class ParserNestedTable : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "nested table"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2013-07-11 17:35:56 +00:00
|
|
|
};
|
|
|
|
|
2014-06-26 00:58:14 +00:00
|
|
|
|
2017-05-27 17:29:55 +00:00
|
|
|
/** Parametric type or Storage. For example:
|
|
|
|
* FixedString(10) or
|
|
|
|
* Partitioned(Log, ChunkID) or
|
2017-04-01 07:20:54 +00:00
|
|
|
* Nested(UInt32 CounterID, FixedString(2) UserAgentMajor)
|
2017-05-27 17:29:55 +00:00
|
|
|
* Result of parsing - ASTFunction with or without parameters.
|
|
|
|
*/
|
2013-07-11 17:35:56 +00:00
|
|
|
class ParserIdentifierWithParameters : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "identifier with parameters"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2013-07-11 17:35:56 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2016-12-29 23:17:51 +00:00
|
|
|
/** Data type or table engine, possibly with parameters. For example, UInt8 or see examples from ParserIdentifierWithParameters
|
|
|
|
* Parse result is ASTFunction, with or without arguments.
|
2011-08-18 18:48:00 +00:00
|
|
|
*/
|
|
|
|
class ParserIdentifierWithOptionalParameters : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "identifier with optional parameters"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2011-08-18 18:48:00 +00:00
|
|
|
};
|
|
|
|
|
2015-12-24 17:14:10 +00:00
|
|
|
class ParserTypeInCastExpression : public ParserIdentifierWithOptionalParameters
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "type in cast expression"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2015-12-24 17:14:10 +00:00
|
|
|
};
|
|
|
|
|
2011-08-18 18:48:00 +00:00
|
|
|
|
2014-05-20 16:46:33 +00:00
|
|
|
template <class NameParser>
|
|
|
|
class IParserNameTypePair : public IParserBase
|
2011-08-18 18:48:00 +00:00
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "name and type pair"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2011-08-18 18:48:00 +00:00
|
|
|
};
|
|
|
|
|
2017-05-27 17:29:55 +00:00
|
|
|
/** The name and type are separated by a space. For example, URL String. */
|
2016-05-28 10:35:44 +00:00
|
|
|
using ParserNameTypePair = IParserNameTypePair<ParserIdentifier>;
|
2017-05-27 17:29:55 +00:00
|
|
|
/** Name and type separated by a space. The name can contain a dot. For example, Hits.URL String. */
|
2016-05-28 10:35:44 +00:00
|
|
|
using ParserCompoundNameTypePair = IParserNameTypePair<ParserCompoundIdentifier>;
|
2014-05-20 16:46:33 +00:00
|
|
|
|
|
|
|
template <class NameParser>
|
2015-04-11 03:10:23 +00:00
|
|
|
bool IParserNameTypePair<NameParser>::parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected)
|
2014-05-20 16:46:33 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
NameParser name_parser;
|
|
|
|
ParserIdentifierWithOptionalParameters type_parser;
|
2017-06-15 17:55:57 +00:00
|
|
|
ParserWhitespaceOrComments ws_parser;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
Pos begin = pos;
|
|
|
|
|
|
|
|
ASTPtr name, type;
|
|
|
|
if (name_parser.parse(pos, end, name, max_parsed_pos, expected)
|
|
|
|
&& ws_parser.ignore(pos, end, max_parsed_pos, expected)
|
|
|
|
&& type_parser.parse(pos, end, type, max_parsed_pos, expected))
|
|
|
|
{
|
|
|
|
auto name_type_pair = std::make_shared<ASTNameTypePair>(StringRange(begin, pos));
|
|
|
|
name_type_pair->name = typeid_cast<const ASTIdentifier &>(*name).name;
|
|
|
|
name_type_pair->type = type;
|
|
|
|
name_type_pair->children.push_back(type);
|
|
|
|
node = name_type_pair;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2014-05-20 16:46:33 +00:00
|
|
|
}
|
2011-08-18 18:48:00 +00:00
|
|
|
|
2017-05-27 17:29:55 +00:00
|
|
|
/** List of columns. */
|
2013-07-11 17:35:56 +00:00
|
|
|
class ParserNameTypePairList : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "name and type pair list"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2014-09-24 14:44:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template <class NameParser>
|
|
|
|
class IParserColumnDeclaration : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "column declaration"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2014-09-24 14:44:57 +00:00
|
|
|
};
|
|
|
|
|
2016-05-28 10:35:44 +00:00
|
|
|
using ParserColumnDeclaration = IParserColumnDeclaration<ParserIdentifier>;
|
|
|
|
using ParserCompoundColumnDeclaration = IParserColumnDeclaration<ParserCompoundIdentifier>;
|
2014-09-24 14:44:57 +00:00
|
|
|
|
|
|
|
template <class NameParser>
|
2015-04-11 03:10:23 +00:00
|
|
|
bool IParserColumnDeclaration<NameParser>::parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected)
|
2014-09-24 14:44:57 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
NameParser name_parser;
|
|
|
|
ParserIdentifierWithOptionalParameters type_parser;
|
2017-06-15 17:55:57 +00:00
|
|
|
ParserWhitespaceOrComments ws;
|
2017-04-01 07:20:54 +00:00
|
|
|
ParserString s_default{"DEFAULT", true, true};
|
|
|
|
ParserString s_materialized{"MATERIALIZED", true, true};
|
|
|
|
ParserString s_alias{"ALIAS", true, true};
|
|
|
|
ParserTernaryOperatorExpression expr_parser;
|
|
|
|
|
|
|
|
const auto begin = pos;
|
|
|
|
|
|
|
|
/// mandatory column name
|
|
|
|
ASTPtr name;
|
|
|
|
if (!name_parser.parse(pos, end, name, max_parsed_pos, expected))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ws.ignore(pos, end, max_parsed_pos, expected);
|
|
|
|
|
|
|
|
/** column name should be followed by type name if it
|
|
|
|
* is not immediately followed by {DEFAULT, MATERIALIZED, ALIAS}
|
|
|
|
*/
|
|
|
|
ASTPtr type;
|
|
|
|
const auto fallback_pos = pos;
|
|
|
|
if (!s_default.check(pos, end, expected, max_parsed_pos) &&
|
|
|
|
!s_materialized.check(pos, end, expected, max_parsed_pos) &&
|
|
|
|
!s_alias.check(pos, end, expected, max_parsed_pos))
|
|
|
|
{
|
|
|
|
if (type_parser.parse(pos, end, type, max_parsed_pos, expected))
|
|
|
|
ws.ignore(pos, end, max_parsed_pos, expected);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
pos = fallback_pos;
|
|
|
|
|
|
|
|
/// parse {DEFAULT, MATERIALIZED, ALIAS}
|
|
|
|
String default_specifier;
|
|
|
|
ASTPtr default_expression;
|
|
|
|
const auto pos_before_specifier = pos;
|
|
|
|
if (s_default.ignore(pos, end, max_parsed_pos, expected) ||
|
|
|
|
s_materialized.ignore(pos, end, max_parsed_pos, expected) ||
|
|
|
|
s_alias.ignore(pos, end, max_parsed_pos, expected))
|
|
|
|
{
|
|
|
|
default_specifier = Poco::toUpper(std::string{pos_before_specifier, pos});
|
|
|
|
|
|
|
|
/// should be followed by an expression
|
|
|
|
ws.ignore(pos, end, max_parsed_pos, expected);
|
|
|
|
|
|
|
|
if (!expr_parser.parse(pos, end, default_expression, max_parsed_pos, expected))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if (!type)
|
|
|
|
return false; /// reject sole column name without type
|
|
|
|
|
|
|
|
const auto column_declaration = std::make_shared<ASTColumnDeclaration>(StringRange{begin, pos});
|
|
|
|
node = column_declaration;
|
|
|
|
column_declaration->name = typeid_cast<ASTIdentifier &>(*name).name;
|
|
|
|
if (type)
|
|
|
|
{
|
|
|
|
column_declaration->type = type;
|
|
|
|
column_declaration->children.push_back(std::move(type));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (default_expression)
|
|
|
|
{
|
|
|
|
column_declaration->default_specifier = default_specifier;
|
|
|
|
column_declaration->default_expression = default_expression;
|
|
|
|
column_declaration->children.push_back(std::move(default_expression));
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2014-09-24 14:44:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class ParserColumnDeclarationList : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "column declaration list"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2013-07-11 17:35:56 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2011-11-01 15:16:04 +00:00
|
|
|
/** ENGINE = name. */
|
|
|
|
class ParserEngine : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "ENGINE"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2011-11-01 15:16:04 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2017-05-27 17:29:55 +00:00
|
|
|
/** Query like this:
|
2011-10-31 06:37:12 +00:00
|
|
|
* CREATE|ATTACH TABLE [IF NOT EXISTS] [db.]name
|
2011-08-18 18:48:00 +00:00
|
|
|
* (
|
2017-04-01 07:20:54 +00:00
|
|
|
* name1 type1,
|
|
|
|
* name2 type2,
|
|
|
|
* ...
|
2011-08-18 18:48:00 +00:00
|
|
|
* ) ENGINE = engine
|
2011-10-31 06:37:12 +00:00
|
|
|
*
|
2017-05-27 17:29:55 +00:00
|
|
|
* Or:
|
2012-07-12 20:06:45 +00:00
|
|
|
* CREATE|ATTACH TABLE [IF NOT EXISTS] [db.]name AS [db2.]name2 [ENGINE = engine]
|
2011-11-01 15:16:04 +00:00
|
|
|
*
|
2017-05-27 17:29:55 +00:00
|
|
|
* Or:
|
2012-12-11 20:32:08 +00:00
|
|
|
* CREATE|ATTACH TABLE [IF NOT EXISTS] [db.]name AS ENGINE = engine SELECT ...
|
2011-11-05 23:31:19 +00:00
|
|
|
*
|
2017-05-27 17:29:55 +00:00
|
|
|
* Or:
|
2016-03-19 01:18:49 +00:00
|
|
|
* CREATE|ATTACH DATABASE db [ENGINE = engine]
|
2014-04-24 18:49:07 +00:00
|
|
|
*
|
2017-05-27 17:29:55 +00:00
|
|
|
* Or:
|
2014-04-24 18:49:07 +00:00
|
|
|
* CREATE|ATTACH [MATERIALIZED] VIEW [IF NOT EXISTS] [db.]name [ENGINE = engine] [POPULATE] AS SELECT ...
|
2011-08-18 18:48:00 +00:00
|
|
|
*/
|
|
|
|
class ParserCreateQuery : public IParserBase
|
|
|
|
{
|
|
|
|
protected:
|
2017-04-01 07:20:54 +00:00
|
|
|
const char * getName() const { return "CREATE TABLE or ATTACH TABLE query"; }
|
|
|
|
bool parseImpl(Pos & pos, Pos end, ASTPtr & node, Pos & max_parsed_pos, Expected & expected);
|
2011-08-18 18:48:00 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|