ClickHouse/dbms/Parsers/ParserExtendedRoleSet.cpp

140 lines
4.3 KiB
C++
Raw Normal View History

#include <Parsers/ParserExtendedRoleSet.h>
2019-12-01 22:01:05 +00:00
#include <Parsers/CommonParsers.h>
#include <Parsers/ExpressionElementParsers.h>
#include <Parsers/ASTLiteral.h>
#include <Parsers/ASTExtendedRoleSet.h>
#include <Parsers/parseUserName.h>
2019-12-01 22:01:05 +00:00
#include <boost/range/algorithm/find.hpp>
namespace DB
{
namespace
2019-12-01 22:01:05 +00:00
{
bool parseRoleNameOrID(IParserBase::Pos & pos, Expected & expected, bool parse_id, String & res)
{
return IParserBase::wrapParseImpl(pos, [&]
{
if (!parse_id)
return parseRoleName(pos, expected, res);
if (!ParserKeyword{"ID"}.ignore(pos, expected))
return false;
if (!ParserToken(TokenType::OpeningRoundBracket).ignore(pos, expected))
return false;
ASTPtr ast;
if (!ParserStringLiteral{}.parse(pos, ast, expected))
return false;
String id = ast->as<ASTLiteral &>().value.safeGet<String>();
if (!ParserToken(TokenType::ClosingRoundBracket).ignore(pos, expected))
return false;
res = std::move(id);
return true;
});
}
bool parseBeforeExcept(
IParserBase::Pos & pos,
Expected & expected,
bool id_mode,
bool all_keyword_enabled,
bool current_user_keyword_enabled,
Strings & names,
bool & all,
bool & current_user)
2019-12-01 22:01:05 +00:00
{
return IParserBase::wrapParseImpl(pos, [&]
2019-12-01 22:01:05 +00:00
{
bool res_all = false;
bool res_current_user = false;
Strings res_names;
while (true)
2019-12-01 22:01:05 +00:00
{
if (ParserKeyword{"NONE"}.ignore(pos, expected))
{
}
else if (
current_user_keyword_enabled
&& (ParserKeyword{"CURRENT_USER"}.ignore(pos, expected) || ParserKeyword{"currentUser"}.ignore(pos, expected)))
{
if (ParserToken{TokenType::OpeningRoundBracket}.ignore(pos, expected))
{
if (!ParserToken{TokenType::ClosingRoundBracket}.ignore(pos, expected))
return false;
}
res_current_user = true;
}
else if (all_keyword_enabled && ParserKeyword{"ALL"}.ignore(pos, expected))
{
res_all = true;
}
else
{
String name;
if (!parseRoleNameOrID(pos, expected, id_mode, name))
return false;
res_names.push_back(name);
}
if (!ParserToken{TokenType::Comma}.ignore(pos, expected))
break;
2019-12-01 22:01:05 +00:00
}
all = res_all;
current_user = res_current_user;
names = std::move(res_names);
return true;
});
2019-12-01 22:01:05 +00:00
}
bool parseExceptAndAfterExcept(
IParserBase::Pos & pos,
Expected & expected,
bool id_mode,
bool current_user_keyword_enabled,
Strings & except_names,
bool & except_current_user)
2019-12-01 22:01:05 +00:00
{
return IParserBase::wrapParseImpl(pos, [&]
{
if (!ParserKeyword{"EXCEPT"}.ignore(pos, expected))
return false;
bool dummy;
return parseBeforeExcept(pos, expected, id_mode, false, current_user_keyword_enabled, except_names, dummy, except_current_user);
});
2019-12-01 22:01:05 +00:00
}
}
bool ParserExtendedRoleSet::parseImpl(Pos & pos, ASTPtr & node, Expected & expected)
{
Strings names;
bool current_user = false;
bool all = false;
Strings except_names;
bool except_current_user = false;
if (!parseBeforeExcept(pos, expected, id_mode, all_keyword, current_user_keyword, names, all, current_user))
return false;
parseExceptAndAfterExcept(pos, expected, id_mode, current_user_keyword, except_names, except_current_user);
if (all)
names.clear();
2019-12-01 22:01:05 +00:00
auto result = std::make_shared<ASTExtendedRoleSet>();
result->names = std::move(names);
2019-12-01 22:01:05 +00:00
result->current_user = current_user;
result->all = all;
result->except_names = std::move(except_names);
2019-12-01 22:01:05 +00:00
result->except_current_user = except_current_user;
result->id_mode = id_mode;
2019-12-01 22:01:05 +00:00
node = result;
return true;
}
}