blob: e50f68f9f11bbdae7b25469db589b4091ea8a703 [file] [log] [blame]
Michael Blow82464fb2017-03-28 18:48:13 -04001//
2// Licensed to the Apache Software Foundation (ASF) under one
3// or more contributor license agreements. See the NOTICE file
4// distributed with this work for additional information
5// regarding copyright ownership. The ASF licenses this file
6// to you under the Apache License, Version 2.0 (the
7// "License"); you may not use this file except in compliance
8// with the License. You may obtain a copy of the License at
9//
10// http://www.apache.org/licenses/LICENSE-2.0
11//
12// Unless required by applicable law or agreed to in writing,
13// software distributed under the License is distributed on an
14// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15// KIND, either express or implied. See the License for the
16// specific language governing permissions and limitations
17// under the License.
18//
Yingyi Bu391f09e2015-10-29 13:49:39 -070019options {
20
21
22 STATIC = false;
23
24}
25
26
27PARSER_BEGIN(SQLPPParser)
28
29package org.apache.asterix.lang.sqlpp.parser;
30
31// For SQL++ ParserTokenManager
Till Westmanne9b2adf2016-10-15 12:39:01 -070032import java.util.ArrayDeque;
Yingyi Bu391f09e2015-10-29 13:49:39 -070033
34import java.io.BufferedReader;
35import java.io.File;
36import java.io.FileInputStream;
37import java.io.FileNotFoundException;
38import java.io.IOException;
39import java.io.InputStreamReader;
40import java.io.Reader;
41import java.io.StringReader;
42import java.util.ArrayList;
Yingyi Budaa549c2016-06-28 22:30:52 -070043import java.util.Collections;
Yingyi Bu391f09e2015-10-29 13:49:39 -070044import java.util.HashMap;
45import java.util.Iterator;
46import java.util.LinkedHashMap;
47import java.util.List;
48import java.util.Map;
49
50import org.apache.asterix.common.annotations.AutoDataGen;
51import org.apache.asterix.common.annotations.DateBetweenYearsDataGen;
52import org.apache.asterix.common.annotations.DatetimeAddRandHoursDataGen;
53import org.apache.asterix.common.annotations.DatetimeBetweenYearsDataGen;
54import org.apache.asterix.common.annotations.FieldIntervalDataGen;
55import org.apache.asterix.common.annotations.FieldValFileDataGen;
56import org.apache.asterix.common.annotations.FieldValFileSameIndexDataGen;
57import org.apache.asterix.common.annotations.IRecordFieldDataGen;
58import org.apache.asterix.common.annotations.InsertRandIntDataGen;
59import org.apache.asterix.common.annotations.ListDataGen;
60import org.apache.asterix.common.annotations.ListValFileDataGen;
61import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
62import org.apache.asterix.common.annotations.TypeDataGen;
63import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
64import org.apache.asterix.common.config.DatasetConfig.DatasetType;
65import org.apache.asterix.common.config.DatasetConfig.IndexType;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080066import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070067import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080068import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070069import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080070import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070071import org.apache.asterix.lang.common.base.AbstractLangExpression;
72import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070073import org.apache.asterix.lang.common.base.Expression;
74import org.apache.asterix.lang.common.base.Literal;
75import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070076import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070077import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070078import org.apache.asterix.lang.common.base.Statement;
79import org.apache.asterix.lang.common.clause.GroupbyClause;
80import org.apache.asterix.lang.common.clause.LetClause;
81import org.apache.asterix.lang.common.clause.LimitClause;
82import org.apache.asterix.lang.common.clause.OrderbyClause;
83import org.apache.asterix.lang.common.clause.UpdateClause;
84import org.apache.asterix.lang.common.clause.WhereClause;
85import org.apache.asterix.lang.common.context.RootScopeFactory;
86import org.apache.asterix.lang.common.context.Scope;
87import org.apache.asterix.lang.common.expression.AbstractAccessor;
88import org.apache.asterix.lang.common.expression.CallExpr;
89import org.apache.asterix.lang.common.expression.FieldAccessor;
90import org.apache.asterix.lang.common.expression.FieldBinding;
91import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
92import org.apache.asterix.lang.common.expression.IfExpr;
93import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -070094import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -070095import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +030096import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -070097import org.apache.asterix.lang.common.expression.LiteralExpr;
98import org.apache.asterix.lang.common.expression.OperatorExpr;
99import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
100import org.apache.asterix.lang.common.expression.QuantifiedExpression;
101import org.apache.asterix.lang.common.expression.RecordConstructor;
102import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
103import org.apache.asterix.lang.common.expression.TypeExpression;
104import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
105import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700106import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
107import org.apache.asterix.lang.common.expression.VariableExpr;
108import org.apache.asterix.lang.common.literal.DoubleLiteral;
109import org.apache.asterix.lang.common.literal.FalseLiteral;
110import org.apache.asterix.lang.common.literal.FloatLiteral;
111import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700112import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700113import org.apache.asterix.lang.common.literal.NullLiteral;
114import org.apache.asterix.lang.common.literal.StringLiteral;
115import org.apache.asterix.lang.common.literal.TrueLiteral;
116import org.apache.asterix.lang.common.parser.ScopeChecker;
117import org.apache.asterix.lang.common.statement.CompactStatement;
118import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800119import org.apache.asterix.lang.common.statement.StartFeedStatement;
120import org.apache.asterix.lang.common.statement.StopFeedStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700121import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
122import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
123import org.apache.asterix.lang.common.statement.CreateFeedStatement;
124import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
125import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700126import org.apache.asterix.lang.common.statement.DatasetDecl;
127import org.apache.asterix.lang.common.statement.DataverseDecl;
128import org.apache.asterix.lang.common.statement.DataverseDropStatement;
129import org.apache.asterix.lang.common.statement.DeleteStatement;
130import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700131import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700132import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
133import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300134import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700135import org.apache.asterix.lang.common.statement.FunctionDecl;
136import org.apache.asterix.lang.common.statement.FunctionDropStatement;
137import org.apache.asterix.lang.common.statement.IndexDropStatement;
138import org.apache.asterix.lang.common.statement.InsertStatement;
139import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
140import org.apache.asterix.lang.common.statement.LoadStatement;
141import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
142import org.apache.asterix.lang.common.statement.NodegroupDecl;
143import org.apache.asterix.lang.common.statement.Query;
144import org.apache.asterix.lang.common.statement.RefreshExternalDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700145import org.apache.asterix.lang.common.statement.SetStatement;
146import org.apache.asterix.lang.common.statement.TypeDecl;
147import org.apache.asterix.lang.common.statement.TypeDropStatement;
148import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800149import org.apache.asterix.lang.common.statement.UpsertStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700150import org.apache.asterix.lang.common.statement.WriteStatement;
151import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800152import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700153import org.apache.asterix.lang.common.struct.QuantifiedPair;
154import org.apache.asterix.lang.common.struct.VarIdentifier;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700155import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700156import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
157import org.apache.asterix.lang.sqlpp.clause.FromClause;
158import org.apache.asterix.lang.sqlpp.clause.FromTerm;
159import org.apache.asterix.lang.sqlpp.clause.HavingClause;
160import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700161import org.apache.asterix.lang.sqlpp.clause.Projection;
162import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
163import org.apache.asterix.lang.sqlpp.clause.SelectClause;
164import org.apache.asterix.lang.sqlpp.clause.SelectElement;
165import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
166import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
167import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800168import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700169import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700170import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700171import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700172import org.apache.asterix.lang.sqlpp.optype.JoinType;
173import org.apache.asterix.lang.sqlpp.optype.SetOpType;
174import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
175import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700176import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700177import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700178import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800179import org.apache.asterix.om.functions.BuiltinFunctions;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700180import org.apache.hyracks.algebricks.common.utils.Pair;
181import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800182import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700183import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
184import org.apache.hyracks.algebricks.core.algebra.expressions.IndexedNLJoinExpressionAnnotation;
185import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700186import org.apache.hyracks.api.exceptions.SourceLocation;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800187import org.apache.hyracks.util.LogRedactionUtil;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700188
Yingyi Bucaea8f02015-11-16 15:12:15 -0800189class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700190
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800191 // tokens parsed as identifiers
192 private static final String CURRENT = "CURRENT";
193 private static final String EXCLUDE = "EXCLUDE";
194 private static final String FOLLOWING = "FOLLOWING";
195 private static final String GROUPS = "GROUPS";
196 private static final String NO = "NO";
197 private static final String OTHERS = "OTHERS";
198 private static final String PARTITION = "PARTITION";
199 private static final String PRECEDING = "PRECEDING";
200 private static final String RANGE = "RANGE";
201 private static final String ROW = "ROW";
202 private static final String ROWS = "ROWS";
203 private static final String TIES = "TIES";
204 private static final String UNBOUNDED = "UNBOUNDED";
205
Yingyi Bu391f09e2015-10-29 13:49:39 -0700206 // optimizer hints
207 private static final String AUTO_HINT = "auto";
208 private static final String BROADCAST_JOIN_HINT = "bcast";
209 private static final String COMPOSE_VAL_FILES_HINT = "compose-val-files";
210 private static final String DATE_BETWEEN_YEARS_HINT = "date-between-years";
211 private static final String DATETIME_ADD_RAND_HOURS_HINT = "datetime-add-rand-hours";
212 private static final String DATETIME_BETWEEN_YEARS_HINT = "datetime-between-years";
213 private static final String HASH_GROUP_BY_HINT = "hash";
214 private static final String INDEXED_NESTED_LOOP_JOIN_HINT = "indexnl";
215 private static final String INMEMORY_HINT = "inmem";
216 private static final String INSERT_RAND_INT_HINT = "insert-rand-int";
217 private static final String INTERVAL_HINT = "interval";
218 private static final String LIST_HINT = "list";
219 private static final String LIST_VAL_FILE_HINT = "list-val-file";
220 private static final String RANGE_HINT = "range";
221 private static final String SKIP_SECONDARY_INDEX_SEARCH_HINT = "skip-index";
222 private static final String VAL_FILE_HINT = "val-files";
223 private static final String VAL_FILE_SAME_INDEX_HINT = "val-file-same-idx";
Yingyi Bu391f09e2015-10-29 13:49:39 -0700224 private static final String GEN_FIELDS_HINT = "gen-fields";
225
226 // data generator hints
227 private static final String DGEN_HINT = "dgen";
228
Till Westmann7199a562016-09-17 16:07:32 -0700229 // error configuration
230 protected static final boolean REPORT_EXPECTED_TOKENS = false;
231
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700232 private int externalVarCounter;
233
Yingyi Bu391f09e2015-10-29 13:49:39 -0700234 private static class IndexParams {
235 public IndexType type;
236 public int gramLength;
237
238 public IndexParams(IndexType type, int gramLength) {
239 this.type = type;
240 this.gramLength = gramLength;
241 }
242 };
243
244 private static class FunctionName {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700245 public String dataverse;
246 public String library;
247 public String function;
248 public String hint;
249 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700250 }
251
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700252 private String getHint(Token t) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700253 if (t.specialToken == null) {
254 return null;
255 }
256 String s = t.specialToken.image;
257 int n = s.length();
258 if (n < 2) {
259 return null;
260 }
261 return s.substring(1).trim();
262 }
263
Ali Alsuliman80225e22018-10-15 14:17:07 -0700264 private static IParser createNewParser(String statement) {
265 return new SQLPPParser(statement);
266 }
267
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700268 private Token getHintToken(Token t) {
269 return t.specialToken;
270 }
271
272 private IRecordFieldDataGen parseFieldDataGen(String hint, Token hintToken) throws ParseException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700273 IRecordFieldDataGen rfdg = null;
274 String splits[] = hint.split(" +");
275 if (splits[0].equals(VAL_FILE_HINT)) {
276 File[] valFiles = new File[splits.length - 1];
277 for (int k=1; k<splits.length; k++) {
278 valFiles[k-1] = new File(splits[k]);
279 }
280 rfdg = new FieldValFileDataGen(valFiles);
281 } else if (splits[0].equals(VAL_FILE_SAME_INDEX_HINT)) {
282 rfdg = new FieldValFileSameIndexDataGen(new File(splits[1]), splits[2]);
283 } else if (splits[0].equals(LIST_VAL_FILE_HINT)) {
284 rfdg = new ListValFileDataGen(new File(splits[1]), Integer.parseInt(splits[2]), Integer.parseInt(splits[3]));
285 } else if (splits[0].equals(LIST_HINT)) {
286 rfdg = new ListDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
287 } else if (splits[0].equals(INTERVAL_HINT)) {
288 FieldIntervalDataGen.ValueType vt;
289 if (splits[1].equals("int")) {
290 vt = FieldIntervalDataGen.ValueType.INT;
291 } else if (splits[1].equals("long")) {
292 vt = FieldIntervalDataGen.ValueType.LONG;
293 } else if (splits[1].equals("float")) {
294 vt = FieldIntervalDataGen.ValueType.FLOAT;
295 } else if (splits[1].equals("double")) {
296 vt = FieldIntervalDataGen.ValueType.DOUBLE;
297 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700298 throw new SqlppParseException(getSourceLocation(hintToken), "Unknown type for interval data gen: " + splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700299 }
300 rfdg = new FieldIntervalDataGen(vt, splits[2], splits[3]);
301 } else if (splits[0].equals(INSERT_RAND_INT_HINT)) {
302 rfdg = new InsertRandIntDataGen(splits[1], splits[2]);
303 } else if (splits[0].equals(DATE_BETWEEN_YEARS_HINT)) {
304 rfdg = new DateBetweenYearsDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
305 } else if (splits[0].equals(DATETIME_BETWEEN_YEARS_HINT)) {
306 rfdg = new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
307 } else if (splits[0].equals(DATETIME_ADD_RAND_HOURS_HINT)) {
308 rfdg = new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
309 } else if (splits[0].equals(AUTO_HINT)) {
310 rfdg = new AutoDataGen(splits[1]);
311 }
312 return rfdg;
313 }
314
Till Westmann7199a562016-09-17 16:07:32 -0700315 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700316 this(new StringReader(s));
317 super.setInput(s);
318 }
319
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800320 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700321 File file = new File(args[0]);
322 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
323 SQLPPParser parser = new SQLPPParser(fis);
324 List<Statement> st = parser.parse();
325 //st.accept(new SQLPPPrintVisitor(), 0);
326 }
327
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800328 public List<Statement> parse() throws CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700329 try {
330 return Statement();
331 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700332 // this is here as the JavaCharStream that's below the lexer sometimes throws Errors that are not handled
Yingyi Bu391f09e2015-10-29 13:49:39 -0700333 // by the ANTLR-generated lexer or parser (e.g it does this for invalid backslash u + 4 hex digits escapes)
Till Westmann60f89982017-08-11 18:14:20 -0700334 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Dmitry Lychagin85142c02018-04-05 17:27:36 -0700335 throw new CompilationException(ErrorCode.PARSE_ERROR, msg);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700336 } catch (SqlppParseException e) {
337 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(), getMessage(e));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700338 } catch (ParseException e) {
Dmitry Lychagin85142c02018-04-05 17:27:36 -0700339 throw new CompilationException(ErrorCode.PARSE_ERROR, getMessage(e));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700340 }
341 }
Till Westmann7199a562016-09-17 16:07:32 -0700342
343 protected String getMessage(ParseException pe) {
344 Token currentToken = pe.currentToken;
345 if (currentToken == null) {
346 return pe.getMessage();
347 }
348 int[][] expectedTokenSequences = pe.expectedTokenSequences;
349 String[] tokenImage = pe.tokenImage;
350 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
351 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
352 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
353 Token tok = currentToken.next;
354 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700355 StringBuilder message = new StringBuilder(128);
356 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700357 for (int i = 0; i < maxSize; i++) {
358 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700359 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700360 }
361 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700362 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700363 break;
364 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700365 final String fixedTokenImage = tokenImage[tok.kind];
366 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700367 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700368 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700369 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700370 tok = tok.next;
371 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700372 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700373 if (REPORT_EXPECTED_TOKENS) {
374 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700375 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700376 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700377 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700378 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700379 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700380 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700381 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700382 }
383
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700384 protected static SourceLocation getSourceLocation(Token token) {
385 return token != null ? new SourceLocation(token.beginLine, token.beginColumn) : null;
386 }
387
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700388 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
389 expr.setSourceLocation(getSourceLocation(token));
390 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700391 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800392
393 private boolean isToken(String image) {
394 return token.image.equalsIgnoreCase(image);
395 }
396
397 private void expectToken(String image) throws SqlppParseException {
398 if (!isToken(image)) {
399 throw createUnexpectedTokenError();
400 }
401 }
402
403 private SqlppParseException createUnexpectedTokenError() {
Ali Alsuliman587b7902019-01-21 14:33:50 -0800404 return new SqlppParseException(getSourceLocation(token), "Unexpected token: " + LogRedactionUtil.userData(token.image));
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800405 }
406
407 private boolean laToken(int idx, int kind, String image) {
408 Token t = getToken(idx);
409 return t.kind == kind && t.image.equalsIgnoreCase(image);
410 }
411
412 private boolean laIdentifier(int idx, String image) {
413 return laToken(idx, IDENTIFIER, image);
414 }
415
416 private boolean laIdentifier(String image) {
417 return laIdentifier(1, image);
418 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700419}
420
421PARSER_END(SQLPPParser)
422
423
424List<Statement> Statement() throws ParseException:
425{
426 scopeStack.push(RootScopeFactory.createRootScope(this));
427 List<Statement> decls = new ArrayList<Statement>();
428 Statement stmt = null;
429}
430{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300431 (
432 (stmt = SingleStatement()
433 {
434 decls.add(stmt);
435 }
436 )?
437 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700438 )*
439 <EOF>
440 {
441 return decls;
442 }
443}
444
445Statement SingleStatement() throws ParseException:
446{
447 Statement stmt = null;
448}
449{
450 (
451 stmt = DataverseDeclaration()
452 | stmt = FunctionDeclaration()
453 | stmt = CreateStatement()
454 | stmt = LoadStatement()
455 | stmt = DropStatement()
456 | stmt = WriteStatement()
457 | stmt = SetStatement()
458 | stmt = InsertStatement()
459 | stmt = DeleteStatement()
460 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800461 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700462 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700463 | stmt = CompactStatement()
Till Westmannef3f0272016-07-27 18:34:01 -0700464 | stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300465 | stmt = Query(false)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700466 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700467 )
468 {
469 return stmt;
470 }
471}
472
473DataverseDecl DataverseDeclaration() throws ParseException:
474{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700475 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700476 String dvName = null;
477}
478{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700479 <USE> { startToken = token; } dvName = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700480 {
481 defaultDataverse = dvName;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700482 DataverseDecl dvDecl = new DataverseDecl(new Identifier(dvName));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700483 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700484 }
485}
486
487Statement CreateStatement() throws ParseException:
488{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700489 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700490 String hint = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700491 Token hintToken = null;
492 boolean hintDGen = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700493 Statement stmt = null;
494}
495{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700496 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700497 (
498 {
499 hint = getHint(token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700500 if (hint != null) {
501 hintToken = getHintToken(token);
502 hintDGen = hint.startsWith(DGEN_HINT);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700503 }
504 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700505 stmt = TypeSpecification(startToken, hint, hintDGen, hintToken)
506 | stmt = NodegroupSpecification(startToken)
507 | stmt = DatasetSpecification(startToken)
508 | stmt = IndexSpecification(startToken)
509 | stmt = DataverseSpecification(startToken)
510 | stmt = FunctionSpecification(startToken)
511 | stmt = FeedSpecification(startToken)
512 | stmt = FeedPolicySpecification(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700513 )
514 {
515 return stmt;
516 }
517}
518
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700519TypeDecl TypeSpecification(Token startStmtToken, String hint, boolean dgen, Token hintToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700520{
521 Pair<Identifier,Identifier> nameComponents = null;
522 boolean ifNotExists = false;
523 TypeExpression typeExpr = null;
524}
525{
526 <TYPE> nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700527 <AS> typeExpr = RecordTypeDef()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700528 {
529 long numValues = -1;
530 String filename = null;
531 if (dgen) {
532 String splits[] = hint.split(" +");
533 if (splits.length != 3) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700534 throw new SqlppParseException(getSourceLocation(hintToken), "Expecting /*+ dgen <filename> <numberOfItems> */");
Yingyi Bu391f09e2015-10-29 13:49:39 -0700535 }
536 filename = splits[1];
537 numValues = Long.parseLong(splits[2]);
538 }
539 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700540 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700541 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700542 }
543}
544
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700545NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700546{
547 String name = null;
548 String tmp = null;
549 boolean ifNotExists = false;
550 List<Identifier>ncNames = null;
551}
552{
553 <NODEGROUP> name = Identifier()
554 ifNotExists = IfNotExists() <ON> tmp = Identifier()
555 {
556 ncNames = new ArrayList<Identifier>();
557 ncNames.add(new Identifier(tmp));
558 }
559 ( <COMMA> tmp = Identifier()
560 {
561 ncNames.add(new Identifier(tmp));
562 }
563 )*
564 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700565 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700566 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700567 }
568}
569
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700570DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700571{
572 Pair<Identifier,Identifier> nameComponents = null;
573 boolean ifNotExists = false;
Your Namedace5f22016-01-12 14:02:48 -0800574 Pair<Identifier,Identifier> typeComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700575 String adapterName = null;
576 Map<String,String> properties = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700577 FunctionSignature appliedFunction = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800578 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700579 String nodeGroupName = null;
580 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700581 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700582 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800583 Pair<Integer, List<String>> filterField = null;
Yingyi Bub9169b62016-02-26 21:21:49 -0800584 Pair<Identifier,Identifier> metaTypeComponents = new Pair<Identifier, Identifier>(null, null);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800585 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700586}
587{
588 (
Yingyi Bu1c0fff52016-03-25 20:23:30 -0700589 <EXTERNAL> Dataset() nameComponents = QualifiedName()
Your Namedace5f22016-01-12 14:02:48 -0800590 <LEFTPAREN> typeComponents = TypeName() <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -0700591 ifNotExists = IfNotExists()
592 <USING> adapterName = AdapterName() properties = Configuration()
Till Westmannf3aa19f2017-12-01 17:42:35 -0800593 ( <ON> nodeGroupName = Identifier() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700594 ( <HINTS> hints = Properties() )?
Till Westmannf3aa19f2017-12-01 17:42:35 -0800595 ( <WITH> withRecord = RecordConstructor() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700596 {
597 ExternalDetailsDecl edd = new ExternalDetailsDecl();
598 edd.setAdapter(adapterName);
599 edd.setProperties(properties);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800600 try{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700601 stmt = new DatasetDecl(nameComponents.first,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700602 nameComponents.second,
Your Namedace5f22016-01-12 14:02:48 -0800603 typeComponents.first,
604 typeComponents.second,
Yingyi Bub9169b62016-02-26 21:21:49 -0800605 metaTypeComponents.first,
606 metaTypeComponents.second,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700607 nodeGroupName != null? new Identifier(nodeGroupName): null,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700608 hints,
609 DatasetType.EXTERNAL,
610 edd,
Till Westmannf3aa19f2017-12-01 17:42:35 -0800611 withRecord,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700612 ifNotExists);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800613 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700614 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Till Westmannf3aa19f2017-12-01 17:42:35 -0800615 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700616 }
617
Murtadha Hubail2c04ae02017-11-21 15:58:01 +0300618 | ( <INTERNAL> )?
Yingyi Bu1c0fff52016-03-25 20:23:30 -0700619 Dataset() nameComponents = QualifiedName()
Your Namedace5f22016-01-12 14:02:48 -0800620 <LEFTPAREN> typeComponents = TypeName() <RIGHTPAREN>
Yingyi Bub9169b62016-02-26 21:21:49 -0800621 (
622 { String name; }
623 <WITH>
624 name = Identifier()
625 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700626 if (!name.equalsIgnoreCase("meta")){
627 throw new SqlppParseException(getSourceLocation(startStmtToken),
628 "We can only support one additional associated field called \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -0800629 }
630 }
631 <LEFTPAREN> metaTypeComponents = TypeName() <RIGHTPAREN>
632 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700633 ifNotExists = IfNotExists()
634 primaryKeyFields = PrimaryKey()
635 (<AUTOGENERATED> { autogenerated = true; } )?
636 (<ON> nodeGroupName = Identifier() )?
637 ( <HINTS> hints = Properties() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700638 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
Till Westmannf3aa19f2017-12-01 17:42:35 -0800639 ( <WITH> withRecord = RecordConstructor() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700640 {
Yingyi Buc9bfe252016-03-01 00:02:40 -0800641 if(filterField!=null && filterField.first!=0){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700642 throw new SqlppParseException(getSourceLocation(startStmtToken),
643 "A filter field can only be a field in the main record of the dataset.");
Yingyi Buc9bfe252016-03-01 00:02:40 -0800644 }
645 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second,
646 primaryKeyFields.first,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700647 autogenerated,
Murtadha Hubail2c04ae02017-11-21 15:58:01 +0300648 filterField == null? null : filterField.second);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800649 try{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700650 stmt = new DatasetDecl(nameComponents.first,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700651 nameComponents.second,
Your Namedace5f22016-01-12 14:02:48 -0800652 typeComponents.first,
653 typeComponents.second,
Yingyi Bub9169b62016-02-26 21:21:49 -0800654 metaTypeComponents.first,
655 metaTypeComponents.second,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700656 nodeGroupName != null ? new Identifier(nodeGroupName) : null,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700657 hints,
658 DatasetType.INTERNAL,
659 idd,
Till Westmannf3aa19f2017-12-01 17:42:35 -0800660 withRecord,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700661 ifNotExists);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800662 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700663 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Till Westmannf3aa19f2017-12-01 17:42:35 -0800664 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700665 }
666 )
667 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700668 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700669 }
670}
671
672RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
673{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700674 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700675 Pair<Identifier,Identifier> nameComponents = null;
676 String datasetName = null;
677}
678{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700679 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700680 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700681 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
682 stmt.setDataverseName(nameComponents.first);
683 stmt.setDatasetName(nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700684 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700685 }
686}
687
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700688CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700689{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700690 CreateIndexStatement stmt = new CreateIndexStatement();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700691 String indexName = null;
692 boolean ifNotExists = false;
693 Pair<Identifier,Identifier> nameComponents = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700694 Pair<Integer, Pair<List<String>, IndexedTypeExpression>> fieldPair = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700695 IndexParams indexType = null;
696 boolean enforced = false;
Ali Alsuliman8351d252017-09-24 00:43:15 -0700697 boolean isPrimaryIdx = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700698}
699{
Ali Alsuliman8351d252017-09-24 00:43:15 -0700700 (
701 (<INDEX> indexName = Identifier()
702 ifNotExists = IfNotExists()
703 <ON> nameComponents = QualifiedName()
704 <LEFTPAREN> ( fieldPair = OpenField()
705 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700706 stmt.addFieldExprPair(fieldPair.second);
707 stmt.addFieldIndexIndicator(fieldPair.first);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700708 }
Ali Alsuliman8351d252017-09-24 00:43:15 -0700709 ) (<COMMA> fieldPair = OpenField()
710 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700711 stmt.addFieldExprPair(fieldPair.second);
712 stmt.addFieldIndexIndicator(fieldPair.first);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700713 }
714 )* <RIGHTPAREN> ( <TYPE> indexType = IndexType() )? ( <ENFORCED> { enforced = true; } )?)
715 |
716 (<PRIMARY> <INDEX> {isPrimaryIdx = true;}
717 (
718 (indexName = Identifier())? ifNotExists = IfNotExists()
719 )
720 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
721 )
722 )
723 {
724 if (isPrimaryIdx && indexName == null) {
725 indexName = "primary_idx_" + nameComponents.second;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700726 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700727 stmt.setIndexName(new Identifier(indexName));
728 stmt.setIfNotExists(ifNotExists);
729 stmt.setDataverseName(nameComponents.first);
730 stmt.setDatasetName(nameComponents.second);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700731 if (indexType != null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700732 stmt.setIndexType(indexType.type);
733 stmt.setGramLength(indexType.gramLength);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700734 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700735 stmt.setEnforced(enforced);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700736 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700737 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700738}
739
740String CompactionPolicy() throws ParseException :
741{
742 String compactionPolicy = null;
743}
744{
745 compactionPolicy = Identifier()
746 {
747 return compactionPolicy;
748 }
749}
750
751String FilterField() throws ParseException :
752{
753 String filterField = null;
754}
755{
756 filterField = Identifier()
757 {
758 return filterField;
759 }
760}
761
762IndexParams IndexType() throws ParseException:
763{
764 IndexType type = null;
765 int gramLength = 0;
766}
767{
768 (<BTREE>
769 {
770 type = IndexType.BTREE;
771 }
772 | <RTREE>
773 {
774 type = IndexType.RTREE;
775 }
776 | <KEYWORD>
777 {
778 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
779 }
Taewoo Kimc49405a2017-01-04 00:30:43 -0800780 |<FULLTEXT>
781 {
782 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
783 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700784 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
785 {
786 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
787 gramLength = Integer.valueOf(token.image);
788 }
789 <RIGHTPAREN>)
790 {
791 return new IndexParams(type, gramLength);
792 }
793}
794
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700795CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -0700796{
797 String dvName = null;
798 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700799}
800{
801 <DATAVERSE> dvName = Identifier()
802 ifNotExists = IfNotExists()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700803 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700804 CreateDataverseStatement stmt = new CreateDataverseStatement(new Identifier(dvName), null, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700805 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700806 }
807}
808
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700809CreateFunctionStatement FunctionSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700810{
811 FunctionSignature signature;
812 boolean ifNotExists = false;
813 List<VarIdentifier> paramList = new ArrayList<VarIdentifier>();
814 String functionBody;
815 VarIdentifier var = null;
816 Expression functionBodyExpr;
817 Token beginPos;
818 Token endPos;
819 FunctionName fctName = null;
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -0800820 String currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700821
822 createNewScope();
823}
824{
825 <FUNCTION> fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -0800826 {
827 defaultDataverse = fctName.dataverse;
828 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700829 ifNotExists = IfNotExists()
830 paramList = ParameterList()
831 <LEFTBRACE>
832 {
833 beginPos = token;
834 }
Xikui Wang3de700a2018-03-15 16:32:55 -0700835 (functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression())
836 <RIGHTBRACE>
Yingyi Bu391f09e2015-10-29 13:49:39 -0700837 {
838 endPos = token;
839 functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine, endPos.beginColumn);
840 // TODO use fctName.library
841 signature = new FunctionSignature(fctName.dataverse, fctName.function, paramList.size());
842 getCurrentScope().addFunctionDescriptor(signature, false);
843 removeCurrentScope();
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -0800844 defaultDataverse = currentDataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700845 CreateFunctionStatement stmt = new CreateFunctionStatement(signature, paramList, functionBody, functionBodyExpr, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700846 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700847 }
848}
849
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700850CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700851{
852 Pair<Identifier,Identifier> nameComponents = null;
853 boolean ifNotExists = false;
854 String adapterName = null;
855 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700856 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700857 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800858 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700859}
860{
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800861 <FEED> nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800862 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800863 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800864 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700865 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700866 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800867 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700868 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800869 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800870 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700871}
872
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700873CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700874{
Michael Blowd6cf6412016-06-30 02:44:35 -0400875 String policyName = null;
876 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700877 String sourcePolicyFile = null;
878 String definition = null;
879 boolean ifNotExists = false;
880 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700881 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700882}
883{
884 (
885 <INGESTION> <POLICY> policyName = Identifier() ifNotExists = IfNotExists()
Yingyi Bu6d57e492016-06-06 21:24:42 -0700886 <FROM>
887 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700888 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700889 stmt = new CreateFeedPolicyStatement(policyName,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700890 basePolicyName, properties, definition, ifNotExists);
891 }
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300892 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700893 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700894 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700895 }
Yingyi Bu6d57e492016-06-06 21:24:42 -0700896 )
Yingyi Bu391f09e2015-10-29 13:49:39 -0700897 )
898 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700899 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700900 }
901}
902
Yingyi Bu391f09e2015-10-29 13:49:39 -0700903List<VarIdentifier> ParameterList() throws ParseException:
904{
905 List<VarIdentifier> paramList = new ArrayList<VarIdentifier>();
906 VarIdentifier var = null;
907}
908{
909 <LEFTPAREN> (<IDENTIFIER>
910 {
Yingyi Buacc12a92016-03-26 17:25:05 -0700911 var = SqlppVariableUtil.toInternalVariableIdentifier(token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700912 paramList.add(var);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700913 }
914 (<COMMA> <IDENTIFIER>
915 {
Yingyi Buacc12a92016-03-26 17:25:05 -0700916 var = SqlppVariableUtil.toInternalVariableIdentifier(token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700917 paramList.add(var);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700918 }
919 )*)? <RIGHTPAREN>
920 {
921 return paramList;
922 }
923}
924
925boolean IfNotExists() throws ParseException:
926{
927}
928{
Yingyi Budaa549c2016-06-28 22:30:52 -0700929 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -0700930 {
931 return true;
932 }
933 )?
934 {
935 return false;
936 }
937}
938
Xikui Wang9d63f622017-05-18 17:50:44 -0700939void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700940{
941 FunctionName functioName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -0700942 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700943}
944{
945 <APPLY> <FUNCTION> functioName = FunctionName()
946 {
Xikui Wang261dc6d2017-03-29 21:23:15 -0700947 fqFunctionName = functioName.library == null ? functioName.function : functioName.library + "#" + functioName.function;
948 funcSigs.add(new FunctionSignature(functioName.dataverse, fqFunctionName, 1));
949 }
950 (
951 <COMMA> functioName = FunctionName()
952 {
953 fqFunctionName = functioName.library == null ? functioName.function : functioName.library + "#" + functioName.function;
954 funcSigs.add(new FunctionSignature(functioName.dataverse, fqFunctionName, 1));
955 }
956 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -0700957}
958
959String GetPolicy() throws ParseException:
960{
961 String policy = null;
962}
963{
964 <USING> <POLICY> policy = Identifier()
965 {
966 return policy;
967 }
968
969}
970
971FunctionSignature FunctionSignature() throws ParseException:
972{
973 FunctionName fctName = null;
974 int arity = 0;
975}
976{
977 fctName = FunctionName() <ATT> <INTEGER_LITERAL>
978 {
979 arity = new Integer(token.image);
980 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700981 throw new SqlppParseException(getSourceLocation(token), "Invalid arity:" + arity);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700982 }
983
984 // TODO use fctName.library
985 String fqFunctionName = fctName.library == null ? fctName.function : fctName.library + "#" + fctName.function;
986 return new FunctionSignature(fctName.dataverse, fqFunctionName, arity);
987 }
988}
989
Yingyi Buc9bfe252016-03-01 00:02:40 -0800990Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700991{
Yingyi Buc9bfe252016-03-01 00:02:40 -0800992 Pair<Integer, List<String>> tmp = null;
993 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700994 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
995}
996{
997 <PRIMARY> <KEY> tmp = NestedField()
998 {
Yingyi Buc9bfe252016-03-01 00:02:40 -0800999 keyFieldSourceIndicators.add(tmp.first);
1000 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001001 }
1002 ( <COMMA> tmp = NestedField()
1003 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001004 keyFieldSourceIndicators.add(tmp.first);
1005 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001006 }
1007 )*
1008 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001009 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001010 }
1011}
1012
1013Statement DropStatement() throws ParseException:
1014{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001015 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001016 String id = null;
1017 Pair<Identifier,Identifier> pairId = null;
1018 Triple<Identifier,Identifier,Identifier> tripleId = null;
1019 FunctionSignature funcSig = null;
1020 boolean ifExists = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001021 AbstractStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001022}
1023{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001024 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001025 (
Yingyi Bu1c0fff52016-03-25 20:23:30 -07001026 Dataset() pairId = QualifiedName() ifExists = IfExists()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001027 {
Yingyi Buab817482016-08-19 21:29:31 -07001028 stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001029 }
1030 | <INDEX> tripleId = DoubleQualifiedName() ifExists = IfExists()
1031 {
1032 stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
1033 }
1034 | <NODEGROUP> id = Identifier() ifExists = IfExists()
1035 {
1036 stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
1037 }
1038 | <TYPE> pairId = TypeName() ifExists = IfExists()
1039 {
1040 stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
1041 }
1042 | <DATAVERSE> id = Identifier() ifExists = IfExists()
1043 {
1044 stmt = new DataverseDropStatement(new Identifier(id), ifExists);
1045 }
1046 | <FUNCTION> funcSig = FunctionSignature() ifExists = IfExists()
1047 {
1048 stmt = new FunctionDropStatement(funcSig, ifExists);
1049 }
1050 | <FEED> pairId = QualifiedName() ifExists = IfExists()
1051 {
1052 stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
1053 }
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +03001054 | <INGESTION> <POLICY> pairId = QualifiedName() ifExists = IfExists()
1055 {
1056 stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
1057 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001058 )
1059 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001060 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001061 }
1062}
1063
1064boolean IfExists() throws ParseException :
1065{
1066}
1067{
1068 ( LOOKAHEAD(1) <IF> <EXISTS>
1069 {
1070 return true;
1071 }
1072 )?
1073 {
1074 return false;
1075 }
1076}
1077
1078InsertStatement InsertStatement() throws ParseException:
1079{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001080 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001081 Pair<Identifier,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001082 VariableExpr var = null;
1083 Query query = null;
1084 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001085}
1086{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001087 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Yingyi Bucb5bf332017-01-02 22:19:50 -08001088 query = Query(false)
1089 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001090 {
Yingyi Bucb5bf332017-01-02 22:19:50 -08001091 if (returnExpression != null && var == null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001092 var = ExpressionToVariableUtil.getGeneratedVariable(query.getBody(), true);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001093 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08001094 query.setTopLevel(true);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001095 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second, query, getVarCounter(),
1096 var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001097 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001098 }
1099}
1100
1101UpsertStatement UpsertStatement() throws ParseException:
1102{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001103 Token startToken = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001104 Pair<Identifier,Identifier> nameComponents = null;
1105 VariableExpr var = null;
1106 Query query = null;
1107 Expression returnExpression = null;
1108}
1109{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001110 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Yingyi Bucb5bf332017-01-02 22:19:50 -08001111 query = Query(false)
1112 ( <RETURNING> returnExpression = Expression())?
1113 {
1114 if (returnExpression != null && var == null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001115 var = ExpressionToVariableUtil.getGeneratedVariable(query.getBody(), true);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001116 }
1117 query.setTopLevel(true);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001118 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second, query, getVarCounter(),
1119 var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001120 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001121 }
1122}
1123
1124DeleteStatement DeleteStatement() throws ParseException:
1125{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001126 Token startToken = null;
Yingyi Bu20e085b2016-07-06 12:57:27 -07001127 VariableExpr varExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001128 Expression condition = null;
1129 Pair<Identifier, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001130}
1131{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001132 <DELETE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001133 <FROM> nameComponents = QualifiedName()
Yingyi Bu20e085b2016-07-06 12:57:27 -07001134 ((<AS>)? varExpr = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001135 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07001136 {
Yingyi Bu20e085b2016-07-06 12:57:27 -07001137 if(varExpr == null){
1138 varExpr = new VariableExpr();
1139 VarIdentifier var = SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue());
1140 varExpr.setVar(var);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001141 addSourceLocation(varExpr, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001142 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001143 DeleteStatement stmt = new DeleteStatement(varExpr, nameComponents.first, nameComponents.second,
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07001144 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001145 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001146 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001147}
1148
1149UpdateStatement UpdateStatement() throws ParseException:
1150{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001151 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001152 VariableExpr vars;
1153 Expression target;
1154 Expression condition;
1155 UpdateClause uc;
1156 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
1157}
1158{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001159 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001160 <WHERE> condition = Expression()
1161 <LEFTPAREN> (uc = UpdateClause()
1162 {
1163 ucs.add(uc);
1164 }
1165 (<COMMA> uc = UpdateClause()
1166 {
1167 ucs.add(uc);
1168 }
1169 )*) <RIGHTPAREN>
1170 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001171 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001172 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001173 }
1174}
1175
1176UpdateClause UpdateClause() throws ParseException:
1177{
1178 Expression target = null;
1179 Expression value = null ;
1180 InsertStatement is = null;
1181 DeleteStatement ds = null;
1182 UpdateStatement us = null;
1183 Expression condition = null;
1184 UpdateClause ifbranch = null;
1185 UpdateClause elsebranch = null;
1186}
1187{
1188 (<SET> target = Expression() <EQ> value = Expression()
1189 | is = InsertStatement()
1190 | ds = DeleteStatement()
1191 | us = UpdateStatement()
1192 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
1193 <THEN> ifbranch = UpdateClause()
1194 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
1195 {
1196 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
1197 }
1198 )
1199}
1200
1201Statement SetStatement() throws ParseException:
1202{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001203 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001204 String pn = null;
1205 String pv = null;
1206}
1207{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001208 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001209 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001210 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001211 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001212 }
1213}
1214
1215Statement WriteStatement() throws ParseException:
1216{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001217 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001218 String nodeName = null;
1219 String fileName = null;
1220 Query query;
1221 String writerClass = null;
1222 Pair<Identifier,Identifier> nameComponents = null;
1223}
1224{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001225 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07001226 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001227 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001228 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001229 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001230 }
1231}
1232
1233LoadStatement LoadStatement() throws ParseException:
1234{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001235 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001236 Identifier dataverseName = null;
1237 Identifier datasetName = null;
1238 boolean alreadySorted = false;
1239 String adapterName;
1240 Map<String,String> properties;
1241 Pair<Identifier,Identifier> nameComponents = null;
1242}
1243{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001244 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001245 {
1246 dataverseName = nameComponents.first;
1247 datasetName = nameComponents.second;
1248 }
1249 <USING> adapterName = AdapterName() properties = Configuration()
1250 (<PRESORTED>
1251 {
1252 alreadySorted = true;
1253 }
1254 )?
1255 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001256 LoadStatement stmt = new LoadStatement(dataverseName, datasetName, adapterName, properties, alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001257 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001258 }
1259}
1260
1261
1262String AdapterName() throws ParseException :
1263{
1264 String adapterName = null;
1265}
1266{
1267 adapterName = Identifier()
1268 {
1269 return adapterName;
1270 }
1271}
1272
1273Statement CompactStatement() throws ParseException:
1274{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001275 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001276 Pair<Identifier,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001277}
1278{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001279 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001280 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001281 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001282 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001283 }
1284}
1285
Yingyi Buab817482016-08-19 21:29:31 -07001286Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001287{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001288 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001289 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001290}
1291{
1292 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001293 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
1294 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
1295 | <START> { startToken = token; } stmt = StartStatement(startToken)
1296 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07001297 )
1298 {
1299 return stmt;
1300 }
1301}
1302
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001303Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001304{
1305 Pair<Identifier,Identifier> feedNameComponents = null;
1306
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001307 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001308}
1309{
1310 <FEED> feedNameComponents = QualifiedName()
1311 {
1312 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001313 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001314 }
1315}
1316
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001317AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001318{
1319 Pair<Identifier,Identifier> feedNameComponents = null;
1320
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001321 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001322}
1323{
1324 <FEED> feedNameComponents = QualifiedName()
1325 {
1326 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001327 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001328 }
1329}
1330
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001331AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001332{
1333 Pair<Identifier,Identifier> feedNameComponents = null;
1334 Pair<Identifier,Identifier> datasetNameComponents = null;
1335
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001336 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07001337}
1338{
1339 (
1340 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001341 {
1342 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
1343 }
1344 )
Yingyi Buab817482016-08-19 21:29:31 -07001345 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001346 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07001347 }
1348}
1349
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001350AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001351{
1352 Pair<Identifier,Identifier> feedNameComponents = null;
1353 Pair<Identifier,Identifier> datasetNameComponents = null;
1354
1355 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07001356 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001357 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07001358 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08001359 String whereClauseBody = null;
1360 WhereClause whereClause = null;
1361 Token beginPos = null;
1362 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07001363}
1364{
1365 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001366 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08001367 (ApplyFunction(appliedFunctions))?
1368 (policy = GetPolicy())?
1369 (
1370 <WHERE>
1371 {
1372 beginPos = token;
1373 whereClause = new WhereClause();
1374 Expression whereExpr;
1375 }
1376 whereExpr = Expression()
1377 {
1378 whereClause.setWhereExpr(whereExpr);
1379 }
1380 )?
1381 {
1382 if (whereClause != null) {
1383 endPos = token;
1384 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
1385 }
1386 }
Yingyi Buab817482016-08-19 21:29:31 -07001387 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07001388 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08001389 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07001390 }
1391 )
1392 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001393 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07001394 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001395}
1396
1397Map<String,String> Configuration() throws ParseException :
1398{
1399 Map<String,String> configuration = new LinkedHashMap<String,String>();
1400 Pair<String, String> keyValuePair = null;
1401}
1402{
1403 <LEFTPAREN> ( keyValuePair = KeyValuePair()
1404 {
1405 configuration.put(keyValuePair.first, keyValuePair.second);
1406 }
1407 ( <COMMA> keyValuePair = KeyValuePair()
1408 {
1409 configuration.put(keyValuePair.first, keyValuePair.second);
1410 }
1411 )* )? <RIGHTPAREN>
1412 {
1413 return configuration;
1414 }
1415}
1416
1417Pair<String, String> KeyValuePair() throws ParseException:
1418{
1419 String key;
1420 String value;
1421}
1422{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001423 <LEFTPAREN> key = ConstantString() <EQ> value = ConstantString() <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001424 {
1425 return new Pair<String, String>(key, value);
1426 }
1427}
1428
1429Map<String,String> Properties() throws ParseException:
1430{
1431 Map<String,String> properties = new HashMap<String,String>();
1432 Pair<String, String> property;
1433}
1434{
1435 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
1436 {
1437 properties.put(property.first, property.second);
1438 }
1439 ( <COMMA> property = Property()
1440 {
1441 properties.put(property.first, property.second);
1442 }
1443 )* <RIGHTPAREN> )?
1444 {
1445 return properties;
1446 }
1447}
1448
1449Pair<String, String> Property() throws ParseException:
1450{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001451 String key = null;
1452 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001453}
1454{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001455 (key = Identifier() | key = StringLiteral())
1456 <EQ>
1457 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001458 {
1459 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001460 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07001461 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001462 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001463 }
1464 }
1465 )
1466 {
1467 return new Pair<String, String>(key.toUpperCase(), value);
1468 }
1469}
1470
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001471IndexedTypeExpression IndexedTypeExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001472{
1473 TypeExpression typeExpr = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001474 boolean isUnknownable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001475}
1476{
1477 (
1478 typeExpr = TypeReference()
1479 | typeExpr = OrderedListTypeDef()
1480 | typeExpr = UnorderedListTypeDef()
1481 )
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001482 ( <QUES> { isUnknownable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001483 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001484 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001485 }
1486}
1487
1488TypeExpression TypeExpr() throws ParseException:
1489{
1490 TypeExpression typeExpr = null;
1491}
1492{
1493 (
1494 typeExpr = RecordTypeDef()
1495 | typeExpr = TypeReference()
1496 | typeExpr = OrderedListTypeDef()
1497 | typeExpr = UnorderedListTypeDef()
1498 )
1499 {
1500 return typeExpr;
1501 }
1502}
1503
1504RecordTypeDefinition RecordTypeDef() throws ParseException:
1505{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001506 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001507 RecordTypeDefinition recType = new RecordTypeDefinition();
1508 RecordTypeDefinition.RecordKind recordKind = null;
1509}
1510{
1511 ( <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
1512 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; } )?
1513 <LEFTBRACE>
1514 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001515 startToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001516 String hint = getHint(token);
1517 if (hint != null) {
1518 String splits[] = hint.split(" +");
1519 if (splits[0].equals(GEN_FIELDS_HINT)) {
1520 if (splits.length != 5) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001521 throw new SqlppParseException(getSourceLocation(getHintToken(token)),
1522 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07001523 }
1524 if (!splits[1].equals("int")) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001525 throw new SqlppParseException(getSourceLocation(getHintToken(token)),
1526 "The only supported type for gen-fields is int.");
Yingyi Bu391f09e2015-10-29 13:49:39 -07001527 }
1528 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
1529 Integer.parseInt(splits[2]), Integer.parseInt(splits[3]), splits[4]);
1530 recType.setUndeclaredFieldsDataGen(ufdg);
1531 }
1532 }
1533
1534 }
1535 (
1536 RecordField(recType)
1537 ( <COMMA> RecordField(recType) )*
1538 )?
1539 <RIGHTBRACE>
1540 {
1541 if (recordKind == null) {
1542 recordKind = RecordTypeDefinition.RecordKind.OPEN;
1543 }
1544 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001545 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001546 }
1547}
1548
1549void RecordField(RecordTypeDefinition recType) throws ParseException:
1550{
1551 String fieldName;
1552 TypeExpression type = null;
1553 boolean nullable = false;
1554}
1555{
1556 fieldName = Identifier()
1557 {
1558 String hint = getHint(token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001559 IRecordFieldDataGen rfdg = hint != null ? parseFieldDataGen(hint, token.specialToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001560 }
1561 <COLON> type = TypeExpr() (<QUES> { nullable = true; } )?
1562 {
1563 recType.addField(fieldName, type, nullable, rfdg);
1564 }
1565}
1566
1567TypeReferenceExpression TypeReference() throws ParseException:
1568{
Abdullah Alamoudie4b318f2016-09-19 13:31:25 +03001569 Pair<Identifier,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001570}
1571{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001572 id = QualifiedName()
1573 {
1574 if (id.first == null && id.second.getValue().equalsIgnoreCase("int")) {
1575 id.second.setValue("int64");
1576 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001577
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001578 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001579 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001580 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001581}
1582
1583OrderedListTypeDefinition OrderedListTypeDef() throws ParseException:
1584{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001585 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001586 TypeExpression type = null;
1587}
1588{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001589 <LEFTBRACKET> { startToken = token; }
1590 ( type = TypeExpr() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07001591 <RIGHTBRACKET>
1592 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001593 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001594 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001595 }
1596}
1597
Yingyi Bu391f09e2015-10-29 13:49:39 -07001598UnorderedListTypeDefinition UnorderedListTypeDef() throws ParseException:
1599{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001600 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001601 TypeExpression type = null;
1602}
1603{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001604 <LEFTDBLBRACE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001605 ( type = TypeExpr() )
1606 <RIGHTDBLBRACE>
1607 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001608 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001609 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001610 }
1611}
1612
1613FunctionName FunctionName() throws ParseException:
1614{
1615 String first = null;
1616 String second = null;
1617 String third = null;
1618 boolean secondAfterDot = false;
1619}
1620{
Michael Blowd6cf6412016-06-30 02:44:35 -04001621 first = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001622 {
1623 FunctionName result = new FunctionName();
1624 result.hint = getHint(token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001625 result.sourceLoc = getSourceLocation(token);
Michael Blowd6cf6412016-06-30 02:44:35 -04001626 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001627 ( <DOT> second = Identifier()
1628 {
1629 secondAfterDot = true;
1630 }
1631 (<SHARP> third = Identifier())? | <SHARP> second = Identifier() )?
1632 {
1633 if (second == null) {
1634 result.dataverse = defaultDataverse;
1635 result.library = null;
1636 result.function = first;
1637 } else if (third == null) {
1638 if (secondAfterDot) {
1639 result.dataverse = first;
1640 result.library = null;
1641 result.function = second;
1642 } else {
1643 result.dataverse = defaultDataverse;
1644 result.library = first;
1645 result.function = second;
1646 }
1647 } else {
1648 result.dataverse = first;
1649 result.library = second;
1650 result.function = third;
1651 }
1652
1653 if (result.function.equalsIgnoreCase("int")) {
1654 result.function = "int64";
1655 }
1656 return result;
1657 }
1658}
1659
1660Pair<Identifier,Identifier> TypeName() throws ParseException:
1661{
1662 Pair<Identifier,Identifier> name = null;
1663}
1664{
1665 name = QualifiedName()
1666 {
1667 if (name.first == null) {
1668 name.first = new Identifier(defaultDataverse);
1669 }
1670 return name;
1671 }
1672}
1673
1674String Identifier() throws ParseException:
1675{
1676 String lit = null;
1677}
1678{
1679 (<IDENTIFIER>
1680 {
1681 return token.image;
1682 }
1683 | lit = QuotedString()
1684 {
1685 return lit;
1686 }
1687 )
1688}
1689
Yingyi Bu1c0fff52016-03-25 20:23:30 -07001690void Dataset() throws ParseException:
1691{
1692}
1693{
1694 (<DATASET>|<COLLECTION>)
1695}
1696
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001697Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001698{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001699 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001700 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001701}
1702{
1703 fieldList = NestedField()
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001704 ( <COLON> fieldType = IndexedTypeExpr() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001705 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001706 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
1707 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07001708 }
1709}
1710
Yingyi Buc9bfe252016-03-01 00:02:40 -08001711Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001712{
1713 List<String> exprList = new ArrayList<String>();
1714 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001715 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001716 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001717}
1718{
1719 lit = Identifier()
1720 {
Yingyi Bub9169b62016-02-26 21:21:49 -08001721 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001722 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08001723 }
1724 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08001725 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08001726 <LEFTPAREN><RIGHTPAREN>
1727 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001728 if(!lit.equalsIgnoreCase("meta")){
1729 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08001730 }
1731 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001732 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08001733 }
1734 )?
1735 {
1736 if(!meetParens){
1737 exprList.add(lit);
1738 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001739 }
1740 (<DOT>
1741 lit = Identifier()
1742 {
1743 exprList.add(lit);
1744 }
1745 )*
1746 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001747 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001748 }
1749}
1750
Yingyi Bu6d57e492016-06-06 21:24:42 -07001751String ConstantString() throws ParseException:
1752{
1753 String value = null;
1754}
1755{
1756 (value = QuotedString() | value = StringLiteral())
1757 {
1758 return value;
1759 }
1760}
1761
Yingyi Bu391f09e2015-10-29 13:49:39 -07001762String QuotedString() throws ParseException:
1763{
1764}
1765{
1766 <QUOTED_STRING>
1767 {
1768 return removeQuotesAndEscapes(token.image);
1769 }
1770}
1771
Yingyi Bu391f09e2015-10-29 13:49:39 -07001772String StringLiteral() throws ParseException:
1773{
1774}
1775{
1776 <STRING_LITERAL>
1777 {
1778 return removeQuotesAndEscapes(token.image);
1779 }
1780}
1781
1782Pair<Identifier,Identifier> QualifiedName() throws ParseException:
1783{
1784 String first = null;
1785 String second = null;
1786}
1787{
1788 first = Identifier() (<DOT> second = Identifier())?
1789 {
1790 Identifier id1 = null;
1791 Identifier id2 = null;
1792 if (second == null) {
1793 id2 = new Identifier(first);
1794 } else
1795 {
1796 id1 = new Identifier(first);
1797 id2 = new Identifier(second);
1798 }
1799 return new Pair<Identifier,Identifier>(id1, id2);
1800 }
1801}
1802
1803Triple<Identifier,Identifier,Identifier> DoubleQualifiedName() throws ParseException:
1804{
1805 String first = null;
1806 String second = null;
1807 String third = null;
1808}
1809{
1810 first = Identifier() <DOT> second = Identifier() (<DOT> third = Identifier())?
1811 {
1812 Identifier id1 = null;
1813 Identifier id2 = null;
1814 Identifier id3 = null;
1815 if (third == null) {
1816 id2 = new Identifier(first);
1817 id3 = new Identifier(second);
1818 } else {
1819 id1 = new Identifier(first);
1820 id2 = new Identifier(second);
1821 id3 = new Identifier(third);
1822 }
1823 return new Triple<Identifier,Identifier,Identifier>(id1, id2, id3);
1824 }
1825}
1826
1827FunctionDecl FunctionDeclaration() throws ParseException:
1828{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001829 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001830 String functionName;
1831 List<VarIdentifier> paramList = new ArrayList<VarIdentifier>();
1832 Expression funcBody;
1833 createNewScope();
1834}
1835{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001836 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07001837 functionName = Identifier()
1838 paramList = ParameterList()
1839 <LEFTBRACE>
1840 (funcBody = SelectExpression(true) | funcBody = Expression())
1841 <RIGHTBRACE>
1842 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001843 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, paramList.size());
Xikui Wang3de700a2018-03-15 16:32:55 -07001844 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001845 FunctionDecl stmt = new FunctionDecl(signature, paramList, funcBody);
Xikui Wang3de700a2018-03-15 16:32:55 -07001846 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001847 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07001848 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001849}
1850
Till Westmannef3f0272016-07-27 18:34:01 -07001851Query ExplainStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001852{
Till Westmannef3f0272016-07-27 18:34:01 -07001853 Query query;
1854}
1855{
Till Westmann516d1a82016-08-02 14:45:53 -07001856 <EXPLAIN> query = Query(true)
Till Westmannef3f0272016-07-27 18:34:01 -07001857 {
1858 return query;
1859 }
1860}
1861
1862Query Query(boolean explain) throws ParseException:
1863{
1864 Query query = new Query(explain);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001865 Expression expr;
1866}
1867{
1868 (
1869 expr = Expression()
1870 |
1871 expr = SelectExpression(false)
1872 )
1873 {
1874 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001875 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07001876 return query;
1877 }
1878}
1879
1880
Yingyi Bu391f09e2015-10-29 13:49:39 -07001881Expression Expression():
1882{
1883 Expression expr = null;
1884 Expression exprP = null;
1885}
1886{
1887(
1888 LOOKAHEAD(2)
1889 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001890 | expr = QuantifiedExpression()
1891)
1892 {
1893 return (exprP==null) ? expr : exprP;
1894 }
1895}
1896
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001897Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001898{
1899 OperatorExpr op = null;
1900 Expression operand = null;
1901}
1902{
1903 operand = AndExpr()
1904 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07001905 <OR>
1906 {
1907 if (op == null) {
1908 op = new OperatorExpr();
1909 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07001910 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001911 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001912 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07001913 try{
1914 op.addOperator(token.image.toLowerCase());
1915 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001916 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07001917 }
Xikui Wang3de700a2018-03-15 16:32:55 -07001918 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001919
Xikui Wang3de700a2018-03-15 16:32:55 -07001920 operand = AndExpr()
1921 {
1922 op.addOperand(operand);
1923 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001924
1925 )*
1926
1927 {
1928 return op==null? operand: op;
1929 }
1930}
1931
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001932Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001933{
1934 OperatorExpr op = null;
1935 Expression operand = null;
1936}
1937{
Yingyi Bu196db5d2016-07-15 19:07:20 -07001938 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001939 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07001940 <AND>
1941 {
1942 if (op == null) {
1943 op = new OperatorExpr();
1944 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07001945 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001946 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001947 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07001948 try{
1949 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08001950 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001951 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07001952 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001953 }
1954
Yingyi Bu196db5d2016-07-15 19:07:20 -07001955 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001956 {
1957 op.addOperand(operand);
1958 }
1959
1960 )*
1961
1962 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001963 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001964 }
1965}
1966
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001967Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07001968{
1969 Expression inputExpr;
1970 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001971 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07001972}
1973{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001974 (<NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07001975 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001976 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08001977 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001978 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001979 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07001980 } else {
1981 return inputExpr;
1982 }
1983 }
1984}
Yingyi Bu391f09e2015-10-29 13:49:39 -07001985
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001986Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001987{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07001988 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001989 OperatorExpr op = null;
1990 Expression operand = null;
1991 boolean broadcast = false;
1992 IExpressionAnnotation annotation = null;
1993}
1994{
Yingyi Bu6c638342016-09-02 17:54:34 -07001995 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001996
1997 (
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08001998 LOOKAHEAD(2)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN>)
Yingyi Bu391f09e2015-10-29 13:49:39 -07001999 {
2000 String mhint = getHint(token);
2001 if (mhint != null) {
2002 if (mhint.equals(INDEXED_NESTED_LOOP_JOIN_HINT)) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002003 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
2004 } else if (mhint.equals(SKIP_SECONDARY_INDEX_SEARCH_HINT)) {
2005 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Yingyi Buea4ec722016-11-04 01:26:16 -07002006 } else if (mhint.equals(BROADCAST_JOIN_HINT)) {
2007 broadcast = true;
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002008 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002009 }
Yingyi Buea4ec722016-11-04 01:26:16 -07002010
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002011 String operator = token.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07002012 if (operator.equals("<>")){
2013 operator = "!=";
2014 }
2015 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002016 operator = "not_" + operator;
2017 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002018 if (op == null) {
2019 op = new OperatorExpr();
Yingyi Buea4ec722016-11-04 01:26:16 -07002020 op.addOperand(operand, false); // broadcast is always for the right branch
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002021 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002022 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002023 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002024 try{
2025 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002026 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002027 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002028 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002029 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002030
Yingyi Bu6c638342016-09-02 17:54:34 -07002031 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07002032 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002033 op.addOperand(operand, broadcast);
Yingyi Buea4ec722016-11-04 01:26:16 -07002034 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002035 )?
2036
2037 {
2038 if (annotation != null) {
2039 op.addHint(annotation);
2040 }
2041 return op==null? operand: op;
2042 }
2043}
2044
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002045Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07002046{
2047 boolean not = false;
2048 OperatorExpr op = null;
2049 Expression operand = null;
2050 IExpressionAnnotation annotation = null;
2051}
2052{
2053 operand = IsExpr()
2054 (
2055 LOOKAHEAD(2)
2056 (<NOT> { not = true; })? <BETWEEN>
2057 {
2058 String mhint = getHint(token);
2059 if (mhint != null) {
2060 if (mhint.equals(INDEXED_NESTED_LOOP_JOIN_HINT)) {
2061 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
2062 } else if (mhint.equals(SKIP_SECONDARY_INDEX_SEARCH_HINT)) {
2063 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
2064 }
2065 }
2066 String operator = token.image.toLowerCase();
2067 if(not){
2068 operator = "not_" + operator;
2069 }
2070 if (op == null) {
2071 op = new OperatorExpr();
2072 op.addOperand(operand);
2073 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002074 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07002075 }
2076 try{
2077 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002078 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002079 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07002080 }
2081 }
2082
2083 operand = IsExpr()
2084 {
2085 op.addOperand(operand);
2086 }
2087
2088 <AND>
2089 operand = IsExpr()
2090 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08002091 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07002092 op.addOperand(operand);
2093 }
2094 )?
2095
2096 {
2097 if (annotation != null) {
2098 op.addHint(annotation);
2099 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002100 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07002101 }
2102}
2103
Yingyi Budaa549c2016-06-28 22:30:52 -07002104Expression IsExpr() throws ParseException:
2105{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002106 Token notToken = null;
2107 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002108 Expression operand = null;
2109 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002110 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002111}
2112{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002113 operand = LikeExpr()
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002114 ( <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002115 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002116 (
2117 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
2118 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08002119 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07002120 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002121 )
Yingyi Budaa549c2016-06-28 22:30:52 -07002122 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002123 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07002124 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002125 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002126 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002127 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07002128 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002129 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07002130 }
2131 }
2132 )?
2133 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002134 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07002135 }
2136}
2137
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002138Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002139{
2140 boolean not = false;
2141 OperatorExpr op = null;
2142 Expression operand = null;
2143}
2144{
2145 operand = ConcatExpr()
2146 (
2147 LOOKAHEAD(2)
2148 (<NOT> { not = true; })? <LIKE>
2149 {
2150 op = new OperatorExpr();
2151 op.addOperand(operand);
2152 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002153 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002154
2155 String operator = token.image.toLowerCase();
2156 if (not) {
2157 operator = "not_" + operator;
2158 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002159 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002160 op.addOperator(operator);
2161 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002162 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002163 }
2164 }
2165
2166 operand = ConcatExpr()
2167 {
2168 op.addOperand(operand);
2169 }
2170 )?
2171
2172 {
2173 return op == null ? operand : op;
2174 }
2175}
2176
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002177Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002178{
2179 OperatorExpr op = null;
2180 Expression operand = null;
2181}
2182{
2183 operand = AddExpr()
2184 (
2185 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002186 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002187 {
2188 if (op == null) {
2189 op = new OperatorExpr();
2190 op.addOperand(operand);
2191 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002192 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002193 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002194 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002195 }
2196 operand = AddExpr()
2197 {
2198 op.addOperand(operand);
2199 }
2200 )*
2201
2202 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002203 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002204 }
2205}
Yingyi Budaa549c2016-06-28 22:30:52 -07002206
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002207Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002208{
2209 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002210 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002211 Expression operand = null;
2212}
2213{
2214 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07002215 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002216 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002217 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002218 {
2219 if (op == null) {
2220 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002221 op.addOperand(operand);
2222 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002223 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002224 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002225 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002226 }
2227
2228 operand = MultExpr()
2229 {
2230 op.addOperand(operand);
2231 }
2232 )*
2233
2234 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002235 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002236 }
2237}
2238
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002239Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002240{
2241 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002242 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002243 Expression operand = null;
2244}
2245{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002246 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002247
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002248 ( (
2249 <MUL> { opType = OperatorType.MUL; } |
2250 <DIVIDE> { opType = OperatorType.DIVIDE; } |
2251 <DIV> { opType = OperatorType.DIV; } |
2252 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
2253 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002254 {
2255 if (op == null) {
2256 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002257 op.addOperand(operand);
2258 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002259 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002260 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002261 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002262 }
2263 operand = ExponentExpr()
2264 {
2265 op.addOperand(operand);
2266 }
2267 )*
2268
2269 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002270 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002271 }
2272}
2273
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002274Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002275{
2276 OperatorExpr op = null;
2277 Expression operand = null;
2278}
2279{
2280 operand = UnaryExpr()
2281 (<CARET>
2282 {
2283 if (op == null) {
2284 op = new OperatorExpr();
2285 op.addOperand(operand);
2286 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002287 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002288 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002289 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002290 }
2291 operand = UnaryExpr()
2292 {
2293 op.addOperand(operand);
2294 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002295 )?
2296 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002297 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002298 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002299}
2300
2301Expression UnaryExpr() throws ParseException:
2302{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002303 boolean not = false;
2304 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002305 Expression expr = null;
2306}
Yingyi Budaa549c2016-06-28 22:30:52 -07002307{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002308 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002309 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002310 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002311 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002312 exprType = "not_" + exprType;
2313 }
2314 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002315 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002316 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002317 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002318 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002319 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07002320 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002321 }
2322 )?
2323
2324 expr = ValueExpr()
2325 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002326 if (uexpr == null) {
2327 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002328 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002329 uexpr.setExpr(expr);
2330 return uexpr;
2331 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002332 }
2333}
2334
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002335Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002336{
2337 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002338 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002339}
2340{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002341 expr = PrimaryExpr()
2342 (
2343 accessor = FieldAccessor(accessor != null ? accessor : expr)
2344 |
2345 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002346 )*
2347 {
2348 return accessor == null ? expr : accessor;
2349 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002350}
2351
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002352FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002353{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002354 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002355 String ident = null;
2356}
2357{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002358 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002359 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002360 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002361 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002362 }
2363}
2364
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002365AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002366{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002367 Token startToken = null;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002368 boolean isListSliceExpression = false;
2369 AbstractAccessor resultExpression = null;
2370 Expression argumentExpression1 = null;
2371 Expression argumentExpression2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002372}
2373{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002374 <LEFTBRACKET> { startToken = token; }
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002375 ( argumentExpression1 = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002376 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002377 if (argumentExpression1.getKind() == Expression.Kind.LITERAL_EXPRESSION)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002378 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002379 Literal lit = ((LiteralExpr)argumentExpression1).getValue();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002380 if (lit.getLiteralType() != Literal.Type.INTEGER &&
2381 lit.getLiteralType() != Literal.Type.LONG) {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002382 throw new SqlppParseException(argumentExpression1.getSourceLocation(), "Index should be an INTEGER");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002383 }
2384 }
2385 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002386 )
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002387 (<COLON>
2388 {
2389 isListSliceExpression = true;
2390 }
2391 ( argumentExpression2 = Expression()
2392 {
2393 if (argumentExpression2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
2394 Literal lit = ((LiteralExpr)argumentExpression2).getValue();
2395 if (lit.getLiteralType() != Literal.Type.INTEGER &&
2396 lit.getLiteralType() != Literal.Type.LONG) {
2397 throw new SqlppParseException(argumentExpression2.getSourceLocation(), "Index should be an INTEGER");
2398 }
2399 }
2400 })?
2401 )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002402 <RIGHTBRACKET>
2403 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002404 if (!isListSliceExpression) {
2405 resultExpression = new IndexAccessor(inputExpr, argumentExpression1);
2406 } else {
2407 resultExpression = new ListSliceExpression(inputExpr, argumentExpression1, argumentExpression2);
2408 }
2409
2410 return addSourceLocation(resultExpression, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002411 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002412}
2413
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002414Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002415{
2416 Expression expr = null;
2417}
2418{
2419 ( LOOKAHEAD(4)
2420 expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07002421 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002422 | expr = Literal()
2423 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07002424 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002425 | expr = ListConstructor()
2426 | expr = RecordConstructor()
2427 | expr = ParenthesizedExpression()
2428 )
2429 {
2430 return expr;
2431 }
2432}
2433
2434Expression Literal() throws ParseException:
2435{
2436 LiteralExpr lit = new LiteralExpr();
2437 String str = null;
2438}
2439{
2440 ( str = StringLiteral()
2441 {
2442 lit.setValue(new StringLiteral(str));
2443 }
2444 | <INTEGER_LITERAL>
2445 {
Till Westmann68c6a992016-09-30 00:19:12 -07002446 try {
2447 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
2448 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002449 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002450 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002451 }
2452 | <FLOAT_LITERAL>
2453 {
Till Westmann68c6a992016-09-30 00:19:12 -07002454 try {
2455 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
2456 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002457 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002458 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002459 }
2460 | <DOUBLE_LITERAL>
2461 {
Till Westmann68c6a992016-09-30 00:19:12 -07002462 try {
2463 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
2464 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002465 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002466 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002467 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07002468 | <MISSING>
2469 {
2470 lit.setValue(MissingLiteral.INSTANCE);
2471 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002472 | <NULL>
2473 {
2474 lit.setValue(NullLiteral.INSTANCE);
2475 }
2476 | <TRUE>
2477 {
2478 lit.setValue(TrueLiteral.INSTANCE);
2479 }
2480 | <FALSE>
2481 {
2482 lit.setValue(FalseLiteral.INSTANCE);
2483 }
2484 )
2485 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002486 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002487 }
2488}
2489
Yingyi Bu391f09e2015-10-29 13:49:39 -07002490VariableExpr VariableRef() throws ParseException:
2491{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002492 VarIdentifier var = new VarIdentifier();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002493 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002494}
2495{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002496 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
2497 {
Yingyi Buacc12a92016-03-26 17:25:05 -07002498 id = SqlppVariableUtil.toInternalVariableName(id); // Prefix user-defined variables with "$"
Yingyi Bu391f09e2015-10-29 13:49:39 -07002499 Identifier ident = lookupSymbol(id);
2500 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002501 throw new SqlppParseException(getSourceLocation(token),
2502 "Inside limit clauses, it is disallowed to reference a variable having the same name as any variable bound in the same scope as the limit clause.");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002503 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002504 VariableExpr varExp = new VariableExpr();
2505 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07002506 varExp.setVar((VarIdentifier)ident);
2507 } else {
2508 varExp.setVar(var);
Yingyi Buacc12a92016-03-26 17:25:05 -07002509 varExp.setIsNewVar(false);
2510 var.setValue(id);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002511 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002512 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002513 }
2514}
2515
Yingyi Bu391f09e2015-10-29 13:49:39 -07002516VariableExpr Variable() throws ParseException:
2517{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002518 VarIdentifier var = new VarIdentifier();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002519 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002520}
2521{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002522 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
2523 {
Yingyi Buacc12a92016-03-26 17:25:05 -07002524 id = SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
Yingyi Bu391f09e2015-10-29 13:49:39 -07002525 Identifier ident = lookupSymbol(id);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002526 VariableExpr varExp = new VariableExpr();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002527 if(ident != null) { // exist such ident
2528 varExp.setIsNewVar(false);
2529 }
2530 varExp.setVar(var);
Yingyi Bucaea8f02015-11-16 15:12:15 -08002531 var.setValue(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002532 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002533 }
2534}
2535
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002536Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
2537{
2538 VariableExpr var = null;
2539 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
2540}
2541{
2542 var = Variable()
2543 ( LOOKAHEAD(1)
2544 {
2545 VariableExpr fieldVarExpr = null;
2546 String fieldIdentifierStr = null;
2547 }
2548 <LEFTPAREN>
2549 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
2550 {
2551 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
2552 }
2553 (<COMMA>
2554 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
2555 {
2556 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
2557 }
2558 )*
2559 <RIGHTPAREN>
2560 )?
2561 {
2562 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
2563 }
2564}
2565
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07002566VariableExpr ExternalVariableRef() throws ParseException:
2567{
2568 String name = null;
2569}
2570{
2571 (
2572 (
2573 <DOLLAR>
2574 (
2575 <INTEGER_LITERAL> { name = token.image; } |
2576 <IDENTIFIER> { name = token.image; } |
2577 name = QuotedString()
2578 )
2579 )
2580 |
2581 (
2582 <QUES> { name = String.valueOf(++externalVarCounter); }
2583 )
2584 )
2585 {
2586 String idName = SqlppVariableUtil.toExternalVariableName(name);
2587 VarIdentifier id = new VarIdentifier(idName);
2588 VariableExpr varExp = new VariableExpr(id);
2589 return addSourceLocation(varExp, token);
2590 }
2591}
2592
Yingyi Bu391f09e2015-10-29 13:49:39 -07002593Expression ListConstructor() throws ParseException:
2594{
2595 Expression expr = null;
2596}
2597{
2598 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002599 expr = OrderedListConstructor() |
2600 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002601 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002602 {
2603 return expr;
2604 }
2605}
2606
Yingyi Bu391f09e2015-10-29 13:49:39 -07002607ListConstructor OrderedListConstructor() throws ParseException:
2608{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002609 Token startToken = null;
2610 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002611}
2612{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002613 <LEFTBRACKET> { startToken = token; }
2614 exprList = ExpressionList()
2615 <RIGHTBRACKET>
2616 {
2617 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002618 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002619 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002620}
2621
2622ListConstructor UnorderedListConstructor() throws ParseException:
2623{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002624 Token startToken = null;
2625 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002626}
2627{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002628 <LEFTDBLBRACE> { startToken = token; }
2629 exprList = ExpressionList()
2630 <RIGHTDBLBRACE>
2631 {
2632 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002633 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002634 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002635}
2636
2637List<Expression> ExpressionList() throws ParseException:
2638{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002639 Expression expr = null;
2640 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002641}
2642{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002643 (
2644 expr = Expression()
2645 {
2646 exprList.add(expr);
2647 }
2648 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07002649 {
2650 exprList.add(expr);
2651 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002652 )*
2653 )?
2654 {
2655 return exprList;
2656 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002657}
2658
Yingyi Bu391f09e2015-10-29 13:49:39 -07002659RecordConstructor RecordConstructor() throws ParseException:
2660{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002661 Token startToken = null;
2662 FieldBinding fb = null;
2663 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002664}
2665{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002666 <LEFTBRACE> { startToken = token; }
2667 (
2668 fb = FieldBinding() { fbList.add(fb); }
2669 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
2670 )?
2671 <RIGHTBRACE>
2672 {
2673 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002674 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002675 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002676}
2677
2678FieldBinding FieldBinding() throws ParseException:
2679{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002680 Expression left, right;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002681}
2682{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002683 left = Expression() <COLON> right = Expression()
2684 {
2685 return new FieldBinding(left, right);
2686 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002687}
2688
Yingyi Bu391f09e2015-10-29 13:49:39 -07002689Expression FunctionCallExpr() throws ParseException:
2690{
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002691 Expression resultExpr;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002692 CallExpr callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002693 List<Expression> argList = new ArrayList<Expression>();
Yingyi Buf4d09842016-08-26 00:03:52 -07002694 Expression tmp = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002695 int arity = 0;
2696 FunctionName funcName = null;
2697 String hint = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07002698 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002699 boolean distinct = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002700}
2701{
2702 funcName = FunctionName()
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002703 {
2704 hint = funcName.hint;
2705 }
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002706 <LEFTPAREN> (
2707 ( <DISTINCT> { distinct = true; } )?
2708 ( tmp = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002709 {
Yingyi Buf4d09842016-08-26 00:03:52 -07002710 if(star){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002711 if(!funcName.function.equalsIgnoreCase("count")){
2712 throw new SqlppParseException(getSourceLocation(token), "The parameter * can only be used in COUNT().");
Yingyi Buf4d09842016-08-26 00:03:52 -07002713 }
2714 argList.add(new LiteralExpr(new LongIntegerLiteral(1L)));
2715 } else {
2716 argList.add(tmp);
2717 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002718 arity ++;
2719 }
2720 (<COMMA> tmp = Expression()
2721 {
2722 argList.add(tmp);
2723 arity++;
2724 }
2725 )*)? <RIGHTPAREN>
2726 {
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002727 String name = funcName.function;
2728 if (distinct) {
Dmitry Lychagine5ea42d2019-03-15 11:38:09 -07002729 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002730 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002731 // TODO use funcName.library
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002732 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002733 FunctionSignature signature
2734 = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
2735 if (signature == null) {
2736 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
2737 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002738 callExpr = FunctionMapUtil.normalizedListInputFunctions(new CallExpr(signature,argList));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002739 if (hint != null) {
2740 if (hint.startsWith(INDEXED_NESTED_LOOP_JOIN_HINT)) {
2741 callExpr.addHint(IndexedNLJoinExpressionAnnotation.INSTANCE);
2742 } else if (hint.startsWith(SKIP_SECONDARY_INDEX_SEARCH_HINT)) {
2743 callExpr.addHint(SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE);
2744 }
2745 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002746 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002747 resultExpr = callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002748 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002749
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002750 ( <OVER> resultExpr = WindowExpr(callExpr.getFunctionSignature(), callExpr.getExprList(), token) )?
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002751
2752 {
2753 return resultExpr;
2754 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002755}
2756
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002757WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Token startToken) throws ParseException:
2758{
2759 Expression partitionExpr = null;
2760 List<Expression> partitionExprs = new ArrayList<Expression>();
2761 OrderbyClause orderByClause = null;
2762 List<Expression> orderbyList = null;
2763 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
2764 WindowExpression.FrameMode frameMode = null;
2765 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
2766 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
2767 Expression frameStartExpr = null, frameEndExpr = null;
2768 WindowExpression.FrameExclusionKind frameExclusionKind = null;
2769 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
2770 VariableExpr windowVar = null;
2771 List<Pair<Expression, Identifier>> windowFieldList = null;
2772}
2773{
2774 (
2775 windowVarWithFieldList = VariableWithFieldMap() <AS>
2776 {
2777 windowVar = windowVarWithFieldList.first;
2778 windowFieldList = windowVarWithFieldList.second;
2779 }
2780 )?
2781 <LEFTPAREN>
2782 (
2783 <IDENTIFIER> { expectToken(PARTITION); } <BY>
2784 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
2785 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
2786 )?
2787 (
2788 orderByClause = OrderbyClause()
2789 {
2790 orderbyList = orderByClause.getOrderbyList();
2791 orderbyModifierList = orderByClause.getModifierList();
2792 }
2793 (
2794 frameMode = WindowFrameMode()
2795 (
2796 frameStart = WindowFrameBoundary() |
2797 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
2798 )
2799 ( frameExclusionKind = WindowFrameExclusion() )?
2800 {
2801 frameStartKind = frameStart.first;
2802 frameStartExpr = frameStart.second;
2803 if (frameEnd == null) {
2804 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
2805 } else {
2806 frameEndKind = frameEnd.first;
2807 frameEndExpr = frameEnd.second;
2808 }
2809 if (frameExclusionKind == null) {
2810 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
2811 }
2812 }
2813 )?
2814 )?
2815 <RIGHTPAREN>
2816 {
2817 WindowExpression winExp = new WindowExpression(signature, argList, partitionExprs, orderbyList, orderbyModifierList,
2818 frameMode, frameStartKind, frameStartExpr, frameEndKind, frameEndExpr, frameExclusionKind, windowVar,
2819 windowFieldList);
2820 return addSourceLocation(winExp, startToken);
2821 }
2822}
2823
2824WindowExpression.FrameMode WindowFrameMode() throws ParseException:
2825{
2826}
2827{
2828 <IDENTIFIER>
2829 {
2830 if (isToken(RANGE)) {
2831 return WindowExpression.FrameMode.RANGE;
2832 } else if (isToken(ROWS)) {
2833 return WindowExpression.FrameMode.ROWS;
2834 } else if (isToken(GROUPS)) {
2835 return WindowExpression.FrameMode.GROUPS;
2836 } else {
2837 throw createUnexpectedTokenError();
2838 }
2839 }
2840}
2841
2842Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
2843{
2844 boolean current = false;
2845 Expression expr = null;
2846}
2847{
2848 (
2849 LOOKAHEAD({ laIdentifier(CURRENT) }) <IDENTIFIER> { current = true; }
2850 | LOOKAHEAD({ laIdentifier(UNBOUNDED) }) <IDENTIFIER>
2851 | expr = Expression()
2852 )
2853 <IDENTIFIER>
2854 {
2855 WindowExpression.FrameBoundaryKind kind;
2856 if (current && isToken(ROW)) {
2857 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
2858 } else if (!current && isToken(PRECEDING)) {
2859 kind = expr == null
2860 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
2861 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
2862 } else if (!current && isToken(FOLLOWING)) {
2863 kind = expr == null
2864 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
2865 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
2866 } else {
2867 throw createUnexpectedTokenError();
2868 }
2869 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
2870 }
2871}
2872
2873WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
2874{
2875 boolean current = false, no = false;
2876}
2877{
2878 <IDENTIFIER>
2879 {
2880 expectToken(EXCLUDE);
2881 }
2882 (
2883 <GROUP>
2884 {
2885 return WindowExpression.FrameExclusionKind.GROUP;
2886 }
2887 |
2888 (
2889 <IDENTIFIER>
2890 {
2891 if (isToken(TIES)) {
2892 return WindowExpression.FrameExclusionKind.TIES;
2893 } else if (isToken(CURRENT)) {
2894 current = true;
2895 } else if (isToken(NO)) {
2896 no = true;
2897 } else {
2898 throw createUnexpectedTokenError();
2899 }
2900 }
2901 <IDENTIFIER>
2902 {
2903 if (current && isToken(ROW)) {
2904 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
2905 } else if (no && isToken(OTHERS)) {
2906 return WindowExpression.FrameExclusionKind.NO_OTHERS;
2907 } else {
2908 throw createUnexpectedTokenError();
2909 }
2910 }
2911 )
2912 )
2913}
2914
Yingyi Bu391f09e2015-10-29 13:49:39 -07002915Expression ParenthesizedExpression() throws ParseException:
2916{
2917 Expression expr;
2918}
2919{
2920 (
2921 LOOKAHEAD(2)
2922 <LEFTPAREN> expr = Expression() <RIGHTPAREN>
2923 |
2924 expr = Subquery()
2925 )
2926 {
2927 return expr;
2928 }
2929}
2930
Yingyi Buc8c067c2016-07-25 23:37:19 -07002931Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002932{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002933 Token startToken = null;
2934 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07002935 List<Expression> whenExprs = new ArrayList<Expression>();
2936 List<Expression> thenExprs = new ArrayList<Expression>();
2937 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07002938 Expression whenExpr = null;
2939 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002940}
2941{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002942 <CASE> { startToken = token; }
2943 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07002944 (
2945 <WHEN> whenExpr = Expression()
2946 {
2947 whenExprs.add(whenExpr);
2948 }
2949 <THEN> thenExpr = Expression()
2950 {
2951 thenExprs.add(thenExpr);
2952 }
2953 )*
2954 (<ELSE> elseExpr = Expression() )?
2955 <END>
2956 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002957 if (conditionExpr == null) {
2958 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002959 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002960 }
2961 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002962 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07002963 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002964}
2965
Yingyi Buab817482016-08-19 21:29:31 -07002966SelectExpression SelectExpression(boolean subquery) throws ParseException:
2967{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002968 List<LetClause> letClauses = new ArrayList<LetClause>();
2969 SelectSetOperation selectSetOperation;
2970 OrderbyClause orderbyClause = null;
2971 LimitClause limitClause = null;
2972 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07002973}
2974{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002975 ( letClauses = LetClause() )?
2976 selectSetOperation = SelectSetOperation()
2977 (orderbyClause = OrderbyClause() {})?
2978 (limitClause = LimitClause() {})?
2979 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002980 SelectExpression selectExpr =
2981 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
2982 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
2983 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002984 }
2985}
2986
Yingyi Buab817482016-08-19 21:29:31 -07002987SelectSetOperation SelectSetOperation() throws ParseException:
2988{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002989 SetOperationInput setOperationInputLeft;
2990 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
2991}
2992{
2993 {
2994 SelectBlock selectBlockLeft = null;
2995 SelectExpression subqueryLeft = null;
2996 Expression expr = null;
2997 }
2998 selectBlockLeft = SelectBlock()
2999 {
3000 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
3001 }
3002 (
3003 {
3004 SetOpType opType = SetOpType.UNION;
3005 boolean setSemantics = true;
3006 SelectBlock selectBlockRight = null;
3007 SelectExpression subqueryRight = null;
3008 }
3009 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
3010 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
3011 {
3012 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
3013 }
3014 )*
3015 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003016 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
3017 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
3018 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003019 }
3020}
3021
Yingyi Buab817482016-08-19 21:29:31 -07003022SelectExpression Subquery() throws ParseException:
3023{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003024 SelectExpression selectExpr = null;
3025}
3026{
3027 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
3028 {
3029 return selectExpr;
3030 }
3031}
3032
Yingyi Buab817482016-08-19 21:29:31 -07003033SelectBlock SelectBlock() throws ParseException:
3034{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003035 SelectClause selectClause = null;
3036 FromClause fromClause = null;
3037 List<LetClause> fromLetClauses = null;
3038 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003039 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003040 GroupbyClause groupbyClause = null;
3041 List<LetClause> gbyLetClauses = null;
3042 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003043 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003044 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003045}
3046{
3047 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003048 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003049 (
3050 LOOKAHEAD(1)
3051 fromClause = FromClause()
3052 (
3053 LOOKAHEAD(1)
3054 fromLetClauses = LetClause()
3055 )?
3056 )?
3057 (whereClause = WhereClause())?
3058 (
3059 groupbyClause = GroupbyClause()
3060 (
3061 LOOKAHEAD(1)
3062 gbyLetClauses = LetClause()
3063 )?
3064 (havingClause = HavingClause())?
3065 )?
3066 |
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003067 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003068 (
3069 LOOKAHEAD(1)
3070 fromLetClauses = LetClause()
3071 )?
3072 (whereClause = WhereClause())?
3073 (
3074 groupbyClause = GroupbyClause()
3075 (
3076 gbyLetClauses = LetClause()
3077 )?
3078 (havingClause = HavingClause())?
3079 )?
3080 selectClause = SelectClause()
3081 )
3082 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003083 if (fromLetClauses != null) {
3084 fromLetWhereClauses.addAll(fromLetClauses);
3085 }
3086 if (whereClause != null) {
3087 fromLetWhereClauses.add(whereClause);
3088 }
3089 if (gbyLetClauses != null) {
3090 gbyLetHavingClauses.addAll(gbyLetClauses);
3091 }
3092 if (havingClause != null) {
3093 gbyLetHavingClauses.add(havingClause);
3094 }
3095 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
3096 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003097 selectBlock.setSourceLocation(startSrcLoc);
3098 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003099 }
3100}
3101
Yingyi Buab817482016-08-19 21:29:31 -07003102SelectClause SelectClause() throws ParseException:
3103{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003104 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003105 SelectRegular selectRegular = null;
3106 SelectElement selectElement = null;
3107 boolean distinct = false;
3108}
3109{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003110 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003111 (
3112 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04003113 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07003114 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07003115 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003116 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003117 SourceLocation sourceLoc = getSourceLocation(startToken);
3118 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07003119 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003120 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003121 List<Projection> projections = new ArrayList<Projection>();
3122 projections.add(projection);
3123 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003124 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003125 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003126 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
3127 selectClause.setSourceLocation(sourceLoc);
3128 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003129 }
3130}
3131
Yingyi Buab817482016-08-19 21:29:31 -07003132SelectRegular SelectRegular() throws ParseException:
3133{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003134 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003135 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003136 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003137}
3138{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003139 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003140 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003141 projections.add(projection);
3142 startSrcLoc = projection.getSourceLocation();
3143 }
3144 ( LOOKAHEAD(2) <COMMA> projection = Projection()
3145 {
3146 projections.add(projection);
3147 }
3148 )*
3149 {
3150 SelectRegular selectRegular = new SelectRegular(projections);
3151 selectRegular.setSourceLocation(startSrcLoc);
3152 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003153 }
3154}
3155
Yingyi Buab817482016-08-19 21:29:31 -07003156SelectElement SelectElement() throws ParseException:
3157{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003158 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003159 Expression expr = null;
3160 String name = null;
3161}
3162{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003163 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003164 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003165 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003166 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003167 }
3168}
3169
Yingyi Buab817482016-08-19 21:29:31 -07003170Projection Projection() throws ParseException :
3171{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003172 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003173 Expression expr = null;
3174 Identifier identifier = null;
3175 String name = null;
3176 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003177 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003178}
3179{
3180 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003181 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
3182 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003183 | expr = Expression() ((<AS>)? name = Identifier())?
3184 {
3185 if (name == null) {
3186 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
3187 if (generatedColumnIdentifier != null) {
3188 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
3189 }
3190 }
3191 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003192 )
3193 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003194 Projection projection = new Projection(expr, name, star, varStar);
3195 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
3196 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003197 }
3198}
3199
3200FromClause FromClause() throws ParseException :
3201{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003202 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003203 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
3204 extendCurrentScope();
3205}
3206{
3207 {
3208 FromTerm fromTerm = null;
3209 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003210 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003211 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
3212 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003213 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003214 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003215 }
3216}
3217
3218FromTerm FromTerm() throws ParseException :
3219{
3220 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003221 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003222 VariableExpr posVar = null;
3223 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
3224}
3225{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003226 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003227 (
3228 {JoinType joinType = JoinType.INNER; }
3229 (joinType = JoinType())?
3230 {
3231 AbstractBinaryCorrelateClause correlateClause = null;
3232 }
3233 (correlateClause = JoinClause(joinType)
Yingyi Bu6d999ed2016-07-13 11:59:06 -07003234 | correlateClause = UnnestClause(joinType)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003235 )
3236 {
3237 correlateClauses.add(correlateClause);
3238 }
3239 )*
3240 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003241 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003242 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003243 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003244 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
3245 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
3246 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003247 }
3248}
3249
3250JoinClause JoinClause(JoinType joinType) throws ParseException :
3251{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003252 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003253 Expression rightExpr = null;
3254 VariableExpr rightVar = null;
3255 VariableExpr posVar = null;
3256 Expression conditionExpr = null;
3257}
3258{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003259 <JOIN> { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())? <ON> conditionExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003260 {
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003261 if(rightVar==null){
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003262 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003263 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003264 JoinClause joinClause = new JoinClause(joinType, rightExpr, rightVar, posVar, conditionExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003265 return addSourceLocation(joinClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003266 }
3267}
3268
Yingyi Bu391f09e2015-10-29 13:49:39 -07003269UnnestClause UnnestClause(JoinType joinType) throws ParseException :
3270{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003271 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003272 Expression rightExpr;
3273 VariableExpr rightVar;
3274 VariableExpr posVar = null;
3275}
3276{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003277 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable()) (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003278 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003279 if (rightVar == null) {
3280 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003281 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003282 UnnestClause unnestClause = new UnnestClause(joinType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003283 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003284 }
3285}
3286
Yingyi Bu391f09e2015-10-29 13:49:39 -07003287JoinType JoinType() throws ParseException :
3288{
3289 JoinType joinType = JoinType.INNER;
3290}
3291{
3292 (<INNER>|<LEFT> (<OUTER>)? {joinType = JoinType.LEFTOUTER; })
3293 {
3294 return joinType;
3295 }
3296}
3297
3298List<LetClause> LetClause() throws ParseException:
3299{
3300 List<LetClause> letList = new ArrayList<LetClause>();
3301 LetClause letClause;
3302}
3303{
3304 (
3305 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
3306 |
3307 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
3308 )
3309 {
3310 return letList;
3311 }
3312}
3313
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003314WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07003315{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003316 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003317 Expression whereExpr;
3318}
3319{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003320 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003321 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003322 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003323 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003324 }
3325}
3326
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003327OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07003328{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003329 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003330 OrderbyClause oc = new OrderbyClause();
3331 Expression orderbyExpr;
3332 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003333 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003334 int numOfOrderby = 0;
3335}
3336{
3337 <ORDER>
3338 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003339 startToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003340 String hint = getHint(token);
3341 if (hint != null) {
3342 if (hint.startsWith(INMEMORY_HINT)) {
3343 String splits[] = hint.split(" +");
3344 int numFrames = Integer.parseInt(splits[1]);
3345 int numTuples = Integer.parseInt(splits[2]);
3346 oc.setNumFrames(numFrames);
3347 oc.setNumTuples(numTuples);
3348 }
Ali Alsuliman80225e22018-10-15 14:17:07 -07003349 if (hint.startsWith(RANGE_HINT)) {
3350 try {
3351 oc.setRangeMap(RangeMapBuilder.parseHint(createNewParser(hint.substring(RANGE_HINT.length()))));
3352 } catch (CompilationException e) {
3353 throw new SqlppParseException(getSourceLocation(getHintToken(token)), e.getMessage());
3354 }
3355 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003356 }
3357 }
3358 <BY> orderbyExpr = Expression()
3359 {
3360 orderbyList.add(orderbyExpr);
3361 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
3362 }
3363 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
3364 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
3365 {
3366 modifierList.add(modif);
3367 }
3368
3369 (LOOKAHEAD(2) <COMMA> orderbyExpr = Expression()
3370 {
3371 orderbyList.add(orderbyExpr);
3372 modif = OrderbyClause.OrderModifier.ASC;
3373 }
3374 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
3375 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
3376 {
3377 modifierList.add(modif);
3378 }
3379 )*
3380
3381 {
3382 oc.setModifierList(modifierList);
3383 oc.setOrderbyList(orderbyList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003384 return addSourceLocation(oc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003385 }
3386}
3387
3388GroupbyClause GroupbyClause()throws ParseException :
3389{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003390 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003391 GroupbyClause gbc = new GroupbyClause();
3392 List<GbyVariableExpressionPair> vePairList = new ArrayList<GbyVariableExpressionPair>();
3393 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003394 Expression expr = null;
3395 VariableExpr decorVar = null;
3396 Expression decorExpr = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003397 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07003398 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003399 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003400}
3401{
3402 {
3403 Scope newScope = extendCurrentScopeNoPush(true);
3404 // extendCurrentScope(true);
3405 }
3406 <GROUP>
3407 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003408 startToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003409 String hint = getHint(token);
3410 if (hint != null && hint.equals(HASH_GROUP_BY_HINT)) {
3411 gbc.setHashGroupByHint(true);
3412 }
3413 }
3414 <BY> (
3415 expr = Expression()
3416 (LOOKAHEAD(1) (<AS>)?
3417 var = Variable()
Yingyi Bucaea8f02015-11-16 15:12:15 -08003418 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003419 {
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003420 if(var==null){
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003421 var = ExpressionToVariableUtil.getGeneratedVariable(expr, false);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003422 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003423 GbyVariableExpressionPair pair1 = new GbyVariableExpressionPair(var, expr);
3424 vePairList.add(pair1);
3425 }
3426 ( LOOKAHEAD(1) <COMMA>
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003427 {
3428 var = null;
3429 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003430 expr = Expression()
3431 (LOOKAHEAD(1) (<AS>)?
3432 var = Variable()
Yingyi Bucaea8f02015-11-16 15:12:15 -08003433 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003434 {
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003435 if(var==null){
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003436 var = ExpressionToVariableUtil.getGeneratedVariable(expr, false);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003437 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003438 GbyVariableExpressionPair pair2 = new GbyVariableExpressionPair(var, expr);
3439 vePairList.add(pair2);
3440 }
3441 )*
3442 )
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003443 (<GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
3444 {
3445 groupVar = groupVarWithFieldList.first;
3446 groupFieldList = groupVarWithFieldList.second;
3447 }
Yingyi Buacc12a92016-03-26 17:25:05 -07003448 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003449 {
3450 gbc.setGbyPairList(vePairList);
3451 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07003452 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07003453 gbc.setGroupVar(groupVar);
3454 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003455 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003456 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003457 }
3458}
3459
3460HavingClause HavingClause() throws ParseException:
3461{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003462 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003463 Expression filterExpr = null;
3464}
3465{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003466 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003467 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003468 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003469 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003470 }
3471}
3472
3473LimitClause LimitClause() throws ParseException:
3474{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003475 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003476 LimitClause lc = new LimitClause();
3477 Expression expr;
3478 pushForbiddenScope(getCurrentScope());
3479}
3480{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003481 <LIMIT> { startToken = token; } expr = Expression() { lc.setLimitExpr(expr); }
3482 (<OFFSET> expr = Expression() { lc.setOffset(expr); })?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003483
3484 {
3485 popForbiddenScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003486 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003487 }
3488}
3489
3490QuantifiedExpression QuantifiedExpression()throws ParseException:
3491{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003492 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003493 QuantifiedExpression qc = new QuantifiedExpression();
3494 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
3495 Expression satisfiesExpr;
3496 VariableExpr var;
3497 Expression inExpr;
3498 QuantifiedPair pair;
3499}
3500{
3501 {
3502 createNewScope();
3503 }
3504
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003505 ( ((<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); })
3506 | (<EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); }))
Yingyi Bu391f09e2015-10-29 13:49:39 -07003507 var = Variable() <IN> inExpr = Expression()
3508 {
3509 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003510 quantifiedList.add(pair);
3511 }
3512 (
3513 <COMMA> var = Variable() <IN> inExpr = Expression()
3514 {
3515 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003516 quantifiedList.add(pair);
3517 }
3518 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07003519 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003520 {
3521 qc.setSatisfiesExpr(satisfiesExpr);
3522 qc.setQuantifiedList(quantifiedList);
3523 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003524 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003525 }
3526}
3527
3528LetClause LetElement() throws ParseException:
3529{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003530 VariableExpr varExp;
3531 Expression beExp;
3532 extendCurrentScope();
3533}
3534{
3535 varExp = Variable() <EQ> beExp = Expression()
3536 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003537 LetClause lc = new LetClause(varExp, beExp);
3538 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003539 return lc;
3540 }
3541}
3542
3543LetClause WithElement() throws ParseException:
3544{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003545 VariableExpr varExp;
3546 Expression beExp;
3547 extendCurrentScope();
3548}
3549{
3550 varExp = Variable() <AS> beExp = Expression()
3551 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003552 LetClause lc = new LetClause(varExp, beExp);
3553 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003554 return lc;
3555 }
3556}
3557
3558TOKEN_MGR_DECLS:
3559{
3560 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07003561 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003562
3563 public void pushState() {
3564 lexerStateStack.push( curLexState );
3565 }
3566
3567 public void popState(String token) {
3568 if (lexerStateStack.size() > 0) {
3569 SwitchTo( lexerStateStack.pop() );
3570 } else {
3571 int errorLine = input_stream.getEndLine();
3572 int errorColumn = input_stream.getEndColumn();
3573 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
3574 + "\" but state stack is empty.";
3575 throw new TokenMgrError(msg, -1);
3576 }
3577 }
3578}
3579
3580<DEFAULT,IN_DBL_BRACE>
3581TOKEN [IGNORE_CASE]:
3582{
3583 <ALL : "all">
3584 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07003585 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003586 | <APPLY : "apply">
3587 | <AS : "as">
3588 | <ASC : "asc">
3589 | <AT : "at">
3590 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003591 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003592 | <BTREE : "btree">
3593 | <BY : "by">
3594 | <CASE : "case">
3595 | <CLOSED : "closed">
3596 | <CREATE : "create">
3597 | <COMPACTION : "compaction">
3598 | <COMPACT : "compact">
3599 | <CONNECT : "connect">
3600 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07003601 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07003602 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07003603 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003604 | <DECLARE : "declare">
3605 | <DEFINITION : "definition">
3606 | <DELETE : "delete">
3607 | <DESC : "desc">
3608 | <DISCONNECT : "disconnect">
3609 | <DISTINCT : "distinct">
3610 | <DROP : "drop">
3611 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07003612 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003613 | <ELSE : "else">
3614 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07003615 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003616 | <EVERY : "every">
3617 | <EXCEPT : "except">
3618 | <EXISTS : "exists">
3619 | <EXTERNAL : "external">
3620 | <FEED : "feed">
3621 | <FILTER : "filter">
3622 | <FLATTEN : "flatten">
3623 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003624 | <FROM : "from">
3625 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08003626 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003627 | <FUNCTION : "function">
3628 | <GROUP : "group">
3629 | <HAVING : "having">
3630 | <HINTS : "hints">
3631 | <IF : "if">
3632 | <INTO : "into">
3633 | <IN : "in">
3634 | <INDEX : "index">
3635 | <INGESTION : "ingestion">
3636 | <INNER : "inner">
3637 | <INSERT : "insert">
3638 | <INTERNAL : "internal">
3639 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07003640 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003641 | <JOIN : "join">
3642 | <KEYWORD : "keyword">
3643 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07003644 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003645 | <LEFT : "left">
3646 | <LETTING : "letting">
3647 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003648 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003649 | <LIMIT : "limit">
3650 | <LOAD : "load">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003651 | <NODEGROUP : "nodegroup">
3652 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07003653 | <NOT : "not">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003654 | <OFFSET : "offset">
3655 | <ON : "on">
3656 | <OPEN : "open">
3657 | <OR : "or">
3658 | <ORDER : "order">
3659 | <OUTER : "outer">
3660 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003661 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003662 | <PATH : "path">
3663 | <POLICY : "policy">
3664 | <PRESORTED : "pre-sorted">
3665 | <PRIMARY : "primary">
3666 | <RAW : "raw">
3667 | <REFRESH : "refresh">
3668 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08003669 | <RETURNING : "returning">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003670 | <RTREE : "rtree">
3671 | <RUN : "run">
3672 | <SATISFIES : "satisfies">
3673 | <SECONDARY : "secondary">
3674 | <SELECT : "select">
3675 | <SET : "set">
3676 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08003677 | <START : "start">
3678 | <STOP : "stop">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03003679 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07003680 | <THEN : "then">
3681 | <TYPE : "type">
3682 | <TO : "to">
3683 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08003684 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003685 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07003686 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08003687 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07003688 | <USE : "use">
3689 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003690 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08003691 | <VALUED : "valued">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003692 | <WHEN : "when">
3693 | <WHERE : "where">
3694 | <WITH : "with">
3695 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003696}
3697
3698<DEFAULT,IN_DBL_BRACE>
3699TOKEN :
3700{
3701 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003702 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003703 | <DIVIDE : "/">
3704 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003705 | <MINUS : "-">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003706 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003707 | <MUL : "*">
3708 | <PLUS : "+">
3709
3710 | <LEFTPAREN : "(">
3711 | <RIGHTPAREN : ")">
3712 | <LEFTBRACKET : "[">
3713 | <RIGHTBRACKET : "]">
3714
3715 | <ATT : "@">
3716 | <COLON : ":">
3717 | <COMMA : ",">
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003718 | <DOLLAR: "$">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003719 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003720 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003721 | <QUES : "?">
3722 | <SEMICOLON : ";">
3723 | <SHARP : "#">
3724
3725 | <LT : "<">
3726 | <GT : ">">
3727 | <LE : "<=">
3728 | <GE : ">=">
3729 | <EQ : "=">
3730 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07003731 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003732 | <SIMILAR : "~=">
3733}
3734
3735<DEFAULT,IN_DBL_BRACE>
3736TOKEN :
3737{
3738 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
3739}
3740
3741<DEFAULT>
3742TOKEN :
3743{
3744 <RIGHTBRACE : "}"> { popState("}"); }
3745}
3746
3747<DEFAULT,IN_DBL_BRACE>
3748TOKEN :
3749{
3750 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
3751}
3752
3753<IN_DBL_BRACE>
3754TOKEN :
3755{
3756 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
3757}
3758
3759<DEFAULT,IN_DBL_BRACE>
3760TOKEN :
3761{
3762 <INTEGER_LITERAL : (<DIGIT>)+ >
3763}
3764
3765<DEFAULT,IN_DBL_BRACE>
Yingyi Bue311a632016-06-07 18:23:16 -07003766TOKEN [IGNORE_CASE]:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003767{
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003768 <MISSING : "missing">
3769 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003770 | <TRUE : "true">
3771 | <FALSE : "false">
3772}
3773
3774<DEFAULT,IN_DBL_BRACE>
3775TOKEN :
3776{
3777 <#DIGIT : ["0" - "9"]>
3778}
3779
3780<DEFAULT,IN_DBL_BRACE>
3781TOKEN:
3782{
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07003783 < DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
3784 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
3785 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003786 >
Yingyi Bue4d919e2016-10-30 10:47:03 -07003787 | < FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
3788 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
3789 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003790 >
3791 | <DIGITS : (<DIGIT>)+ >
3792}
3793
3794<DEFAULT,IN_DBL_BRACE>
3795TOKEN :
3796{
3797 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08003798 | <#IDENTIFIER_SPECIALCHARS_START : ["_"]>
3799 | <#IDENTIFIER_SPECIALCHARS_REST : ["$"]>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003800}
3801
3802<DEFAULT,IN_DBL_BRACE>
3803TOKEN :
3804{
3805 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07003806 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003807 <EscapeQuot>
3808 | <EscapeBslash>
3809 | <EscapeSlash>
3810 | <EscapeBspace>
3811 | <EscapeFormf>
3812 | <EscapeNl>
3813 | <EscapeCr>
3814 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07003815 | ~["`","\\"])* "`">
3816 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003817 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003818 | <EscapeBslash>
3819 | <EscapeSlash>
3820 | <EscapeBspace>
3821 | <EscapeFormf>
3822 | <EscapeNl>
3823 | <EscapeCr>
3824 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07003825 | ~["\"","\\"])* "\"")
3826 | ("\'"(
3827 <EscapeApos>
3828 | <EscapeBslash>
3829 | <EscapeSlash>
3830 | <EscapeBspace>
3831 | <EscapeFormf>
3832 | <EscapeNl>
3833 | <EscapeCr>
3834 | <EscapeTab>
3835 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003836 | < #EscapeQuot: "\\\"" >
3837 | < #EscapeApos: "\\\'" >
3838 | < #EscapeBslash: "\\\\" >
3839 | < #EscapeSlash: "\\/" >
3840 | < #EscapeBspace: "\\b" >
3841 | < #EscapeFormf: "\\f" >
3842 | < #EscapeNl: "\\n" >
3843 | < #EscapeCr: "\\r" >
3844 | < #EscapeTab: "\\t" >
3845}
3846
3847<DEFAULT,IN_DBL_BRACE>
3848TOKEN :
3849{
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08003850 <IDENTIFIER : ( <LETTER> | <IDENTIFIER_SPECIALCHARS_START> )
3851 ( <LETTER> | <DIGIT> | <IDENTIFIER_SPECIALCHARS_START> | <IDENTIFIER_SPECIALCHARS_REST> )*>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003852}
3853
3854<DEFAULT,IN_DBL_BRACE>
3855SKIP:
3856{
3857 " "
3858 | "\t"
3859 | "\r"
3860 | "\n"
3861}
3862
3863<DEFAULT,IN_DBL_BRACE>
3864SKIP:
3865{
3866 <"//" (~["\n"])* "\n">
3867}
3868
3869<DEFAULT,IN_DBL_BRACE>
3870SKIP:
3871{
3872 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
3873}
3874
3875<DEFAULT,IN_DBL_BRACE>
3876SKIP:
3877{
Yingyi Bu93846a72016-09-13 16:30:39 -07003878 <"--" (~["\n"])* "\n">
3879}
3880
3881
3882<DEFAULT,IN_DBL_BRACE>
3883SKIP:
3884{
3885 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
3886}
3887
3888<DEFAULT,IN_DBL_BRACE>
3889SKIP:
3890{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003891 <"/*"> { pushState(); } : INSIDE_COMMENT
3892}
3893
3894<INSIDE_COMMENT>
3895SPECIAL_TOKEN:
3896{
3897 <"+"(" ")*(~["*"])*>
3898}
3899
3900<INSIDE_COMMENT>
3901SKIP:
3902{
3903 <"/*"> { pushState(); }
3904}
3905
3906<INSIDE_COMMENT>
3907SKIP:
3908{
3909 <"*/"> { popState("*/"); }
3910 | <~[]>
3911}