-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathparser.py
102 lines (78 loc) · 3.34 KB
/
parser.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
from antlr4 import CommonTokenStream, InputStream, TerminalNode
from antlr.MyQueryParserVisitor import MyQueryParserVisitor
from antlr.MyQueryLexer import MyQueryLexer
from antlr.MyQueryParserParser import MyQueryParserParser
from typing import Optional, Tuple
class SplitQuery:
def __init__(self):
self.select = None # type: Optional[str]
self.from_ = None # type: Optional[str]
self.where = None # type: Optional[str]
self.group_by = None # type: Optional[str]
self.order_by = None # type: Optional[str]
self.limit = None # type: Optional[int]
self.offset = None # type: Optional[int]
self.set_command = None # type: Optional[Tuple[str, int]]
self.show_command = None # type: Optional[str]
self.schema_command = None # type: Optional[str]
self.has_aggregate = False
class SplitVisitor(MyQueryParserVisitor):
"""Just splits an ODSQL query into its components:
select, where, group_by, order_by, from, limit, offset"""
def __init__(self, input_text):
self._input_text = input_text
self.q = SplitQuery()
def visitSelect_from(self, ctx):
def _raw_text(ctx):
return self._input_text[
ctx.start.start:ctx.stop.stop + 1]
self.q.select = _raw_text(ctx.select_expressions())
# remove quote-escaping of dataset_id (table) containing special char
# '`arbres-parisiens`' -> 'arbres-parisiens'
self.q.from_ = _raw_text(ctx.table).replace('`', '')
if ctx.condition():
self.q.where = _raw_text(ctx.condition())
if ctx.group_by_expressions():
self.q.group_by = _raw_text(ctx.group_by_expressions())
if ctx.order_by_expressions():
self.q.order_by = _raw_text(ctx.order_by_expressions())
if ctx.limit is not None:
self.q.limit = int(ctx.limit.getText())
if ctx.offset is not None:
self.q.offset = int(ctx.offset.getText())
return self.visitChildren(ctx)
def visitSet_command(self, ctx):
self.q.set_command = (
ctx.option_name.getText(),
int(ctx.int_literal().getText()) if ctx.int_literal()
else ctx.string_literal().getText()[1:-1]
)
def visitFunctionAggregation(self, ctx):
self.q.has_aggregate = True
def visitShow_command(self, ctx):
if ctx.option_name:
self.q.show_command = ctx.option_name.getText()
else:
self.q.show_command = "all"
def visitSchema_command(self, ctx):
self.q.schema_command = ctx.dataset.getText()
def print_rule_tree(tree, rule_names, indent=0):
if isinstance(tree, TerminalNode):
print(" " * indent, tree.getSymbol())
else:
idx = tree.getRuleIndex()
print(" " * indent, tree.__class__, rule_names[idx])
for child in tree.getChildren():
print_rule_tree(child, rule_names, indent + 2)
def split_query_or_command(sql, debug=False):
# type: (str, bool) -> SplitQuery
input_stream = InputStream(sql)
lexer = MyQueryLexer(input_stream)
stream = CommonTokenStream(lexer)
parser = MyQueryParserParser(stream)
parsed = parser.cli_command()
if debug:
print_rule_tree(parsed, parser.ruleNames)
visitor = SplitVisitor(sql)
visitor.visit(parsed)
return visitor.q