Skip to content

Navigation Menu

Sign in
Appearance settings

Search code, repositories, users, issues, pull requests...

Provide feedback

We read every piece of feedback, and take your input very seriously.

Saved searches

Use saved searches to filter your results more quickly

Appearance settings

Latest commit

 

History

History
History
136 lines (114 loc) · 4.51 KB

File metadata and controls

136 lines (114 loc) · 4.51 KB
Copy raw file
Download raw file
Open symbols panel
Edit and raw actions
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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
#include "parser/parser.h"
#include "absl/status/statusor.h"
#include "absl/strings/str_format.h"
#include "absl/types/optional.h"
#include "parser/cel_grammar.inc/cel_grammar/CelLexer.h"
#include "parser/cel_grammar.inc/cel_grammar/CelParser.h"
#include "parser/source_factory.h"
#include "parser/visitor.h"
#include "antlr4-runtime.h"
namespace google {
namespace api {
namespace expr {
namespace parser {
using antlr4::ANTLRInputStream;
using antlr4::CommonTokenStream;
using antlr4::ParseCancellationException;
using antlr4::ParserRuleContext;
using antlr4::tree::ErrorNode;
using antlr4::tree::TerminalNode;
using google::api::expr::v1alpha1::Expr;
using google::api::expr::v1alpha1::ParsedExpr;
namespace {
// ExprRecursionListener extends the standard ANTLR CelParser to ensure that
// recursive entries into the 'expr' rule are limited to a configurable depth so
// as to prevent stack overflows.
class ExprRecursionListener : public ::antlr4::tree::ParseTreeListener {
public:
ExprRecursionListener(
const int max_recursion_depth = kDefaultMaxRecursionDepth)
: max_recursion_depth_(max_recursion_depth), recursion_depth_(0) {}
void visitTerminal(TerminalNode* node) override{};
void visitErrorNode(ErrorNode* error) override{};
void enterEveryRule(ParserRuleContext* ctx) override;
void exitEveryRule(ParserRuleContext* ctx) override;
private:
const int max_recursion_depth_;
int recursion_depth_;
};
void ExprRecursionListener::enterEveryRule(ParserRuleContext* ctx) {
// Throw a ParseCancellationException since the parsing would otherwise
// continue if this were treated as a syntax error and the problem would
// continue to manifest.
if (ctx->getRuleIndex() == ::cel_grammar::CelParser::RuleExpr) {
if (recursion_depth_ >= max_recursion_depth_) {
throw ParseCancellationException(
absl::StrFormat("Expression recursion limit exceeded. limit: %d",
max_recursion_depth_));
}
recursion_depth_++;
}
}
void ExprRecursionListener::exitEveryRule(ParserRuleContext* ctx) {
if (ctx->getRuleIndex() == ::cel_grammar::CelParser::RuleExpr) {
recursion_depth_--;
}
}
} // namespace
absl::StatusOr<ParsedExpr> Parse(const std::string& expression,
const std::string& description,
const int max_recursion_depth) {
return ParseWithMacros(expression, Macro::AllMacros(), description,
max_recursion_depth);
}
absl::StatusOr<ParsedExpr> ParseWithMacros(const std::string& expression,
const std::vector<Macro>& macros,
const std::string& description,
const int max_recursion_depth) {
auto result =
EnrichedParse(expression, macros, description, max_recursion_depth);
if (result.ok()) {
return result->parsed_expr();
}
return result.status();
}
absl::StatusOr<VerboseParsedExpr> EnrichedParse(
const std::string& expression, const std::vector<Macro>& macros,
const std::string& description, const int max_recursion_depth) {
ANTLRInputStream input(expression);
::cel_grammar::CelLexer lexer(&input);
CommonTokenStream tokens(&lexer);
::cel_grammar::CelParser parser(&tokens);
ExprRecursionListener listener(max_recursion_depth);
ParserVisitor visitor(description, expression, max_recursion_depth, macros);
lexer.removeErrorListeners();
parser.removeErrorListeners();
lexer.addErrorListener(&visitor);
parser.addErrorListener(&visitor);
parser.addParseListener(&listener);
// if we were to ignore errors completely:
// std::shared_ptr<BailErrorStrategy> error_strategy(new BailErrorStrategy());
// parser.setErrorHandler(error_strategy);
::cel_grammar::CelParser::StartContext* root;
try {
root = parser.start();
} catch (ParseCancellationException& e) {
return absl::CancelledError(e.what());
} catch (std::exception& e) {
return absl::AbortedError(e.what());
}
Expr expr = visitor.visit(root).as<Expr>();
if (visitor.hasErrored()) {
return absl::InvalidArgumentError(visitor.errorMessage());
}
// root is deleted as part of the parser context
ParsedExpr parsed_expr;
parsed_expr.mutable_expr()->CopyFrom(expr);
parsed_expr.mutable_source_info()->CopyFrom(visitor.sourceInfo());
auto enriched_source_info = visitor.enrichedSourceInfo();
return VerboseParsedExpr(parsed_expr, enriched_source_info);
}
} // namespace parser
} // namespace expr
} // namespace api
} // namespace google
Morty Proxy This is a proxified and sanitized view of the page, visit original site.