blob: 6f63441d3939acfbc2639b3042361c2667631aa7 [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";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700194 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800195 private static final String FOLLOWING = "FOLLOWING";
196 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700197 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700198 private static final String LAST = "LAST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800199 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700200 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800201 private static final String OTHERS = "OTHERS";
202 private static final String PARTITION = "PARTITION";
203 private static final String PRECEDING = "PRECEDING";
204 private static final String RANGE = "RANGE";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700205 private static final String RESPECT = "RESPECT";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800206 private static final String ROW = "ROW";
207 private static final String ROWS = "ROWS";
208 private static final String TIES = "TIES";
209 private static final String UNBOUNDED = "UNBOUNDED";
210
Yingyi Bu391f09e2015-10-29 13:49:39 -0700211 // optimizer hints
212 private static final String AUTO_HINT = "auto";
213 private static final String BROADCAST_JOIN_HINT = "bcast";
214 private static final String COMPOSE_VAL_FILES_HINT = "compose-val-files";
215 private static final String DATE_BETWEEN_YEARS_HINT = "date-between-years";
216 private static final String DATETIME_ADD_RAND_HOURS_HINT = "datetime-add-rand-hours";
217 private static final String DATETIME_BETWEEN_YEARS_HINT = "datetime-between-years";
218 private static final String HASH_GROUP_BY_HINT = "hash";
219 private static final String INDEXED_NESTED_LOOP_JOIN_HINT = "indexnl";
220 private static final String INMEMORY_HINT = "inmem";
221 private static final String INSERT_RAND_INT_HINT = "insert-rand-int";
222 private static final String INTERVAL_HINT = "interval";
223 private static final String LIST_HINT = "list";
224 private static final String LIST_VAL_FILE_HINT = "list-val-file";
225 private static final String RANGE_HINT = "range";
226 private static final String SKIP_SECONDARY_INDEX_SEARCH_HINT = "skip-index";
227 private static final String VAL_FILE_HINT = "val-files";
228 private static final String VAL_FILE_SAME_INDEX_HINT = "val-file-same-idx";
Yingyi Bu391f09e2015-10-29 13:49:39 -0700229 private static final String GEN_FIELDS_HINT = "gen-fields";
230
231 // data generator hints
232 private static final String DGEN_HINT = "dgen";
233
Till Westmann7199a562016-09-17 16:07:32 -0700234 // error configuration
235 protected static final boolean REPORT_EXPECTED_TOKENS = false;
236
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700237 private int externalVarCounter;
238
Yingyi Bu391f09e2015-10-29 13:49:39 -0700239 private static class IndexParams {
240 public IndexType type;
241 public int gramLength;
242
243 public IndexParams(IndexType type, int gramLength) {
244 this.type = type;
245 this.gramLength = gramLength;
246 }
247 };
248
249 private static class FunctionName {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700250 public String dataverse;
251 public String library;
252 public String function;
253 public String hint;
254 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700255 }
256
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700257 private String getHint(Token t) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700258 if (t.specialToken == null) {
259 return null;
260 }
261 String s = t.specialToken.image;
262 int n = s.length();
263 if (n < 2) {
264 return null;
265 }
266 return s.substring(1).trim();
267 }
268
Ali Alsuliman80225e22018-10-15 14:17:07 -0700269 private static IParser createNewParser(String statement) {
270 return new SQLPPParser(statement);
271 }
272
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700273 private Token getHintToken(Token t) {
274 return t.specialToken;
275 }
276
277 private IRecordFieldDataGen parseFieldDataGen(String hint, Token hintToken) throws ParseException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700278 IRecordFieldDataGen rfdg = null;
279 String splits[] = hint.split(" +");
280 if (splits[0].equals(VAL_FILE_HINT)) {
281 File[] valFiles = new File[splits.length - 1];
282 for (int k=1; k<splits.length; k++) {
283 valFiles[k-1] = new File(splits[k]);
284 }
285 rfdg = new FieldValFileDataGen(valFiles);
286 } else if (splits[0].equals(VAL_FILE_SAME_INDEX_HINT)) {
287 rfdg = new FieldValFileSameIndexDataGen(new File(splits[1]), splits[2]);
288 } else if (splits[0].equals(LIST_VAL_FILE_HINT)) {
289 rfdg = new ListValFileDataGen(new File(splits[1]), Integer.parseInt(splits[2]), Integer.parseInt(splits[3]));
290 } else if (splits[0].equals(LIST_HINT)) {
291 rfdg = new ListDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
292 } else if (splits[0].equals(INTERVAL_HINT)) {
293 FieldIntervalDataGen.ValueType vt;
294 if (splits[1].equals("int")) {
295 vt = FieldIntervalDataGen.ValueType.INT;
296 } else if (splits[1].equals("long")) {
297 vt = FieldIntervalDataGen.ValueType.LONG;
298 } else if (splits[1].equals("float")) {
299 vt = FieldIntervalDataGen.ValueType.FLOAT;
300 } else if (splits[1].equals("double")) {
301 vt = FieldIntervalDataGen.ValueType.DOUBLE;
302 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700303 throw new SqlppParseException(getSourceLocation(hintToken), "Unknown type for interval data gen: " + splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700304 }
305 rfdg = new FieldIntervalDataGen(vt, splits[2], splits[3]);
306 } else if (splits[0].equals(INSERT_RAND_INT_HINT)) {
307 rfdg = new InsertRandIntDataGen(splits[1], splits[2]);
308 } else if (splits[0].equals(DATE_BETWEEN_YEARS_HINT)) {
309 rfdg = new DateBetweenYearsDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
310 } else if (splits[0].equals(DATETIME_BETWEEN_YEARS_HINT)) {
311 rfdg = new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
312 } else if (splits[0].equals(DATETIME_ADD_RAND_HOURS_HINT)) {
313 rfdg = new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
314 } else if (splits[0].equals(AUTO_HINT)) {
315 rfdg = new AutoDataGen(splits[1]);
316 }
317 return rfdg;
318 }
319
Till Westmann7199a562016-09-17 16:07:32 -0700320 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700321 this(new StringReader(s));
322 super.setInput(s);
323 }
324
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800325 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700326 File file = new File(args[0]);
327 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
328 SQLPPParser parser = new SQLPPParser(fis);
329 List<Statement> st = parser.parse();
330 //st.accept(new SQLPPPrintVisitor(), 0);
331 }
332
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800333 public List<Statement> parse() throws CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700334 try {
335 return Statement();
336 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700337 // 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 -0700338 // 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 -0700339 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700340 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700341 } catch (SqlppParseException e) {
Ali Alsulimanfe901892019-03-19 01:40:35 -0700342 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(), LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700343 } catch (ParseException e) {
Ali Alsulimanfe901892019-03-19 01:40:35 -0700344 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700345 }
346 }
Till Westmann7199a562016-09-17 16:07:32 -0700347
348 protected String getMessage(ParseException pe) {
349 Token currentToken = pe.currentToken;
350 if (currentToken == null) {
351 return pe.getMessage();
352 }
353 int[][] expectedTokenSequences = pe.expectedTokenSequences;
354 String[] tokenImage = pe.tokenImage;
355 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
356 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
357 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
358 Token tok = currentToken.next;
359 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700360 StringBuilder message = new StringBuilder(128);
361 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700362 for (int i = 0; i < maxSize; i++) {
363 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700364 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700365 }
366 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700367 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700368 break;
369 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700370 final String fixedTokenImage = tokenImage[tok.kind];
371 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700372 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700373 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700374 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700375 tok = tok.next;
376 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700377 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700378 if (REPORT_EXPECTED_TOKENS) {
379 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700380 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700381 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700382 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700383 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700384 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700385 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700386 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700387 }
388
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700389 protected static SourceLocation getSourceLocation(Token token) {
390 return token != null ? new SourceLocation(token.beginLine, token.beginColumn) : null;
391 }
392
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700393 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
394 expr.setSourceLocation(getSourceLocation(token));
395 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700396 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800397
398 private boolean isToken(String image) {
399 return token.image.equalsIgnoreCase(image);
400 }
401
402 private void expectToken(String image) throws SqlppParseException {
403 if (!isToken(image)) {
404 throw createUnexpectedTokenError();
405 }
406 }
407
408 private SqlppParseException createUnexpectedTokenError() {
Ali Alsuliman587b7902019-01-21 14:33:50 -0800409 return new SqlppParseException(getSourceLocation(token), "Unexpected token: " + LogRedactionUtil.userData(token.image));
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800410 }
411
412 private boolean laToken(int idx, int kind, String image) {
413 Token t = getToken(idx);
414 return t.kind == kind && t.image.equalsIgnoreCase(image);
415 }
416
417 private boolean laIdentifier(int idx, String image) {
418 return laToken(idx, IDENTIFIER, image);
419 }
420
421 private boolean laIdentifier(String image) {
422 return laIdentifier(1, image);
423 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700424}
425
426PARSER_END(SQLPPParser)
427
428
429List<Statement> Statement() throws ParseException:
430{
431 scopeStack.push(RootScopeFactory.createRootScope(this));
432 List<Statement> decls = new ArrayList<Statement>();
433 Statement stmt = null;
434}
435{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300436 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700437 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300438 {
439 decls.add(stmt);
440 }
441 )?
442 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700443 )*
444 <EOF>
445 {
446 return decls;
447 }
448}
449
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700450Statement ExplainStatement() throws ParseException:
451{
452 Statement stmt = null;
453 Token explainToken = null;
454}
455{
456 ( <EXPLAIN> { explainToken = token; } )?
457 stmt = SingleStatement()
458 {
459 if (explainToken != null) {
460 if (stmt.getKind() == Statement.Kind.QUERY) {
461 ((Query)stmt).setExplain(true);
462 } else {
463 throw new SqlppParseException(getSourceLocation(explainToken),
464 "EXPLAIN is not supported for this kind of statement");
465 }
466 }
467 return stmt;
468 }
469}
470
Yingyi Bu391f09e2015-10-29 13:49:39 -0700471Statement SingleStatement() throws ParseException:
472{
473 Statement stmt = null;
474}
475{
476 (
477 stmt = DataverseDeclaration()
478 | stmt = FunctionDeclaration()
479 | stmt = CreateStatement()
480 | stmt = LoadStatement()
481 | stmt = DropStatement()
482 | stmt = WriteStatement()
483 | stmt = SetStatement()
484 | stmt = InsertStatement()
485 | stmt = DeleteStatement()
486 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800487 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700488 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700489 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700490 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700491 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700492 )
493 {
494 return stmt;
495 }
496}
497
498DataverseDecl DataverseDeclaration() throws ParseException:
499{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700500 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700501 String dvName = null;
502}
503{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700504 <USE> { startToken = token; } dvName = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700505 {
506 defaultDataverse = dvName;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700507 DataverseDecl dvDecl = new DataverseDecl(new Identifier(dvName));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700508 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700509 }
510}
511
512Statement CreateStatement() throws ParseException:
513{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700514 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700515 String hint = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700516 Token hintToken = null;
517 boolean hintDGen = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700518 Statement stmt = null;
519}
520{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700521 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700522 (
523 {
524 hint = getHint(token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700525 if (hint != null) {
526 hintToken = getHintToken(token);
527 hintDGen = hint.startsWith(DGEN_HINT);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700528 }
529 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700530 stmt = TypeSpecification(startToken, hint, hintDGen, hintToken)
531 | stmt = NodegroupSpecification(startToken)
532 | stmt = DatasetSpecification(startToken)
533 | stmt = IndexSpecification(startToken)
534 | stmt = DataverseSpecification(startToken)
535 | stmt = FunctionSpecification(startToken)
536 | stmt = FeedSpecification(startToken)
537 | stmt = FeedPolicySpecification(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700538 )
539 {
540 return stmt;
541 }
542}
543
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700544TypeDecl TypeSpecification(Token startStmtToken, String hint, boolean dgen, Token hintToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700545{
546 Pair<Identifier,Identifier> nameComponents = null;
547 boolean ifNotExists = false;
548 TypeExpression typeExpr = null;
549}
550{
551 <TYPE> nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700552 <AS> typeExpr = RecordTypeDef()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700553 {
554 long numValues = -1;
555 String filename = null;
556 if (dgen) {
557 String splits[] = hint.split(" +");
558 if (splits.length != 3) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700559 throw new SqlppParseException(getSourceLocation(hintToken), "Expecting /*+ dgen <filename> <numberOfItems> */");
Yingyi Bu391f09e2015-10-29 13:49:39 -0700560 }
561 filename = splits[1];
562 numValues = Long.parseLong(splits[2]);
563 }
564 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700565 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, 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 -0700570NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700571{
572 String name = null;
573 String tmp = null;
574 boolean ifNotExists = false;
575 List<Identifier>ncNames = null;
576}
577{
578 <NODEGROUP> name = Identifier()
579 ifNotExists = IfNotExists() <ON> tmp = Identifier()
580 {
581 ncNames = new ArrayList<Identifier>();
582 ncNames.add(new Identifier(tmp));
583 }
584 ( <COMMA> tmp = Identifier()
585 {
586 ncNames.add(new Identifier(tmp));
587 }
588 )*
589 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700590 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700591 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700592 }
593}
594
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700595DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700596{
597 Pair<Identifier,Identifier> nameComponents = null;
598 boolean ifNotExists = false;
Your Namedace5f22016-01-12 14:02:48 -0800599 Pair<Identifier,Identifier> typeComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700600 String adapterName = null;
601 Map<String,String> properties = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700602 FunctionSignature appliedFunction = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800603 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700604 String nodeGroupName = null;
605 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700606 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700607 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800608 Pair<Integer, List<String>> filterField = null;
Yingyi Bub9169b62016-02-26 21:21:49 -0800609 Pair<Identifier,Identifier> metaTypeComponents = new Pair<Identifier, Identifier>(null, null);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800610 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700611}
612{
613 (
Yingyi Bu1c0fff52016-03-25 20:23:30 -0700614 <EXTERNAL> Dataset() nameComponents = QualifiedName()
Your Namedace5f22016-01-12 14:02:48 -0800615 <LEFTPAREN> typeComponents = TypeName() <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -0700616 ifNotExists = IfNotExists()
617 <USING> adapterName = AdapterName() properties = Configuration()
Till Westmannf3aa19f2017-12-01 17:42:35 -0800618 ( <ON> nodeGroupName = Identifier() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700619 ( <HINTS> hints = Properties() )?
Till Westmannf3aa19f2017-12-01 17:42:35 -0800620 ( <WITH> withRecord = RecordConstructor() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700621 {
622 ExternalDetailsDecl edd = new ExternalDetailsDecl();
623 edd.setAdapter(adapterName);
624 edd.setProperties(properties);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800625 try{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700626 stmt = new DatasetDecl(nameComponents.first,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700627 nameComponents.second,
Your Namedace5f22016-01-12 14:02:48 -0800628 typeComponents.first,
629 typeComponents.second,
Yingyi Bub9169b62016-02-26 21:21:49 -0800630 metaTypeComponents.first,
631 metaTypeComponents.second,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700632 nodeGroupName != null? new Identifier(nodeGroupName): null,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700633 hints,
634 DatasetType.EXTERNAL,
635 edd,
Till Westmannf3aa19f2017-12-01 17:42:35 -0800636 withRecord,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700637 ifNotExists);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800638 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700639 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Till Westmannf3aa19f2017-12-01 17:42:35 -0800640 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700641 }
642
Murtadha Hubail2c04ae02017-11-21 15:58:01 +0300643 | ( <INTERNAL> )?
Yingyi Bu1c0fff52016-03-25 20:23:30 -0700644 Dataset() nameComponents = QualifiedName()
Your Namedace5f22016-01-12 14:02:48 -0800645 <LEFTPAREN> typeComponents = TypeName() <RIGHTPAREN>
Yingyi Bub9169b62016-02-26 21:21:49 -0800646 (
647 { String name; }
648 <WITH>
649 name = Identifier()
650 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700651 if (!name.equalsIgnoreCase("meta")){
652 throw new SqlppParseException(getSourceLocation(startStmtToken),
653 "We can only support one additional associated field called \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -0800654 }
655 }
656 <LEFTPAREN> metaTypeComponents = TypeName() <RIGHTPAREN>
657 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700658 ifNotExists = IfNotExists()
659 primaryKeyFields = PrimaryKey()
660 (<AUTOGENERATED> { autogenerated = true; } )?
661 (<ON> nodeGroupName = Identifier() )?
662 ( <HINTS> hints = Properties() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700663 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
Till Westmannf3aa19f2017-12-01 17:42:35 -0800664 ( <WITH> withRecord = RecordConstructor() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700665 {
Yingyi Buc9bfe252016-03-01 00:02:40 -0800666 if(filterField!=null && filterField.first!=0){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700667 throw new SqlppParseException(getSourceLocation(startStmtToken),
668 "A filter field can only be a field in the main record of the dataset.");
Yingyi Buc9bfe252016-03-01 00:02:40 -0800669 }
670 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second,
671 primaryKeyFields.first,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700672 autogenerated,
Murtadha Hubail2c04ae02017-11-21 15:58:01 +0300673 filterField == null? null : filterField.second);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800674 try{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700675 stmt = new DatasetDecl(nameComponents.first,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700676 nameComponents.second,
Your Namedace5f22016-01-12 14:02:48 -0800677 typeComponents.first,
678 typeComponents.second,
Yingyi Bub9169b62016-02-26 21:21:49 -0800679 metaTypeComponents.first,
680 metaTypeComponents.second,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700681 nodeGroupName != null ? new Identifier(nodeGroupName) : null,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700682 hints,
683 DatasetType.INTERNAL,
684 idd,
Till Westmannf3aa19f2017-12-01 17:42:35 -0800685 withRecord,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700686 ifNotExists);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800687 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700688 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Till Westmannf3aa19f2017-12-01 17:42:35 -0800689 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700690 }
691 )
692 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700693 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700694 }
695}
696
697RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
698{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700699 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700700 Pair<Identifier,Identifier> nameComponents = null;
701 String datasetName = null;
702}
703{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700704 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700705 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700706 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
707 stmt.setDataverseName(nameComponents.first);
708 stmt.setDatasetName(nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700709 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700710 }
711}
712
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700713CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700714{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700715 CreateIndexStatement stmt = new CreateIndexStatement();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700716 String indexName = null;
717 boolean ifNotExists = false;
718 Pair<Identifier,Identifier> nameComponents = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700719 Pair<Integer, Pair<List<String>, IndexedTypeExpression>> fieldPair = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700720 IndexParams indexType = null;
721 boolean enforced = false;
Ali Alsuliman8351d252017-09-24 00:43:15 -0700722 boolean isPrimaryIdx = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700723}
724{
Ali Alsuliman8351d252017-09-24 00:43:15 -0700725 (
726 (<INDEX> indexName = Identifier()
727 ifNotExists = IfNotExists()
728 <ON> nameComponents = QualifiedName()
729 <LEFTPAREN> ( fieldPair = OpenField()
730 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700731 stmt.addFieldExprPair(fieldPair.second);
732 stmt.addFieldIndexIndicator(fieldPair.first);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700733 }
Ali Alsuliman8351d252017-09-24 00:43:15 -0700734 ) (<COMMA> fieldPair = OpenField()
735 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700736 stmt.addFieldExprPair(fieldPair.second);
737 stmt.addFieldIndexIndicator(fieldPair.first);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700738 }
739 )* <RIGHTPAREN> ( <TYPE> indexType = IndexType() )? ( <ENFORCED> { enforced = true; } )?)
740 |
741 (<PRIMARY> <INDEX> {isPrimaryIdx = true;}
742 (
743 (indexName = Identifier())? ifNotExists = IfNotExists()
744 )
745 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
746 )
747 )
748 {
749 if (isPrimaryIdx && indexName == null) {
750 indexName = "primary_idx_" + nameComponents.second;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700751 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700752 stmt.setIndexName(new Identifier(indexName));
753 stmt.setIfNotExists(ifNotExists);
754 stmt.setDataverseName(nameComponents.first);
755 stmt.setDatasetName(nameComponents.second);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700756 if (indexType != null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700757 stmt.setIndexType(indexType.type);
758 stmt.setGramLength(indexType.gramLength);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700759 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700760 stmt.setEnforced(enforced);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700761 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700762 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700763}
764
765String CompactionPolicy() throws ParseException :
766{
767 String compactionPolicy = null;
768}
769{
770 compactionPolicy = Identifier()
771 {
772 return compactionPolicy;
773 }
774}
775
776String FilterField() throws ParseException :
777{
778 String filterField = null;
779}
780{
781 filterField = Identifier()
782 {
783 return filterField;
784 }
785}
786
787IndexParams IndexType() throws ParseException:
788{
789 IndexType type = null;
790 int gramLength = 0;
791}
792{
793 (<BTREE>
794 {
795 type = IndexType.BTREE;
796 }
797 | <RTREE>
798 {
799 type = IndexType.RTREE;
800 }
801 | <KEYWORD>
802 {
803 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
804 }
Taewoo Kimc49405a2017-01-04 00:30:43 -0800805 |<FULLTEXT>
806 {
807 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
808 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700809 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
810 {
811 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
812 gramLength = Integer.valueOf(token.image);
813 }
814 <RIGHTPAREN>)
815 {
816 return new IndexParams(type, gramLength);
817 }
818}
819
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700820CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -0700821{
822 String dvName = null;
823 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700824}
825{
826 <DATAVERSE> dvName = Identifier()
827 ifNotExists = IfNotExists()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700828 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700829 CreateDataverseStatement stmt = new CreateDataverseStatement(new Identifier(dvName), null, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700830 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700831 }
832}
833
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700834CreateFunctionStatement FunctionSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700835{
836 FunctionSignature signature;
837 boolean ifNotExists = false;
838 List<VarIdentifier> paramList = new ArrayList<VarIdentifier>();
839 String functionBody;
840 VarIdentifier var = null;
841 Expression functionBodyExpr;
842 Token beginPos;
843 Token endPos;
844 FunctionName fctName = null;
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -0800845 String currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700846
847 createNewScope();
848}
849{
850 <FUNCTION> fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -0800851 {
852 defaultDataverse = fctName.dataverse;
853 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700854 ifNotExists = IfNotExists()
855 paramList = ParameterList()
856 <LEFTBRACE>
857 {
858 beginPos = token;
859 }
Xikui Wang3de700a2018-03-15 16:32:55 -0700860 (functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression())
861 <RIGHTBRACE>
Yingyi Bu391f09e2015-10-29 13:49:39 -0700862 {
863 endPos = token;
864 functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine, endPos.beginColumn);
865 // TODO use fctName.library
866 signature = new FunctionSignature(fctName.dataverse, fctName.function, paramList.size());
867 getCurrentScope().addFunctionDescriptor(signature, false);
868 removeCurrentScope();
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -0800869 defaultDataverse = currentDataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700870 CreateFunctionStatement stmt = new CreateFunctionStatement(signature, paramList, functionBody, functionBodyExpr, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700871 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700872 }
873}
874
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700875CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700876{
877 Pair<Identifier,Identifier> nameComponents = null;
878 boolean ifNotExists = false;
879 String adapterName = null;
880 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700881 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700882 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800883 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700884}
885{
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800886 <FEED> nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800887 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800888 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800889 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700890 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700891 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800892 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700893 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800894 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800895 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700896}
897
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700898CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700899{
Michael Blowd6cf6412016-06-30 02:44:35 -0400900 String policyName = null;
901 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700902 String sourcePolicyFile = null;
903 String definition = null;
904 boolean ifNotExists = false;
905 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700906 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700907}
908{
909 (
910 <INGESTION> <POLICY> policyName = Identifier() ifNotExists = IfNotExists()
Yingyi Bu6d57e492016-06-06 21:24:42 -0700911 <FROM>
912 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700913 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700914 stmt = new CreateFeedPolicyStatement(policyName,
Yingyi Bu391f09e2015-10-29 13:49:39 -0700915 basePolicyName, properties, definition, ifNotExists);
916 }
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300917 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -0700918 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700919 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700920 }
Yingyi Bu6d57e492016-06-06 21:24:42 -0700921 )
Yingyi Bu391f09e2015-10-29 13:49:39 -0700922 )
923 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700924 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700925 }
926}
927
Yingyi Bu391f09e2015-10-29 13:49:39 -0700928List<VarIdentifier> ParameterList() throws ParseException:
929{
930 List<VarIdentifier> paramList = new ArrayList<VarIdentifier>();
931 VarIdentifier var = null;
932}
933{
934 <LEFTPAREN> (<IDENTIFIER>
935 {
Yingyi Buacc12a92016-03-26 17:25:05 -0700936 var = SqlppVariableUtil.toInternalVariableIdentifier(token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700937 paramList.add(var);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700938 }
939 (<COMMA> <IDENTIFIER>
940 {
Yingyi Buacc12a92016-03-26 17:25:05 -0700941 var = SqlppVariableUtil.toInternalVariableIdentifier(token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700942 paramList.add(var);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700943 }
944 )*)? <RIGHTPAREN>
945 {
946 return paramList;
947 }
948}
949
950boolean IfNotExists() throws ParseException:
951{
952}
953{
Yingyi Budaa549c2016-06-28 22:30:52 -0700954 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -0700955 {
956 return true;
957 }
958 )?
959 {
960 return false;
961 }
962}
963
Xikui Wang9d63f622017-05-18 17:50:44 -0700964void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700965{
966 FunctionName functioName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -0700967 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700968}
969{
970 <APPLY> <FUNCTION> functioName = FunctionName()
971 {
Xikui Wang261dc6d2017-03-29 21:23:15 -0700972 fqFunctionName = functioName.library == null ? functioName.function : functioName.library + "#" + functioName.function;
973 funcSigs.add(new FunctionSignature(functioName.dataverse, fqFunctionName, 1));
974 }
975 (
976 <COMMA> functioName = FunctionName()
977 {
978 fqFunctionName = functioName.library == null ? functioName.function : functioName.library + "#" + functioName.function;
979 funcSigs.add(new FunctionSignature(functioName.dataverse, fqFunctionName, 1));
980 }
981 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -0700982}
983
984String GetPolicy() throws ParseException:
985{
986 String policy = null;
987}
988{
989 <USING> <POLICY> policy = Identifier()
990 {
991 return policy;
992 }
993
994}
995
996FunctionSignature FunctionSignature() throws ParseException:
997{
998 FunctionName fctName = null;
999 int arity = 0;
1000}
1001{
1002 fctName = FunctionName() <ATT> <INTEGER_LITERAL>
1003 {
1004 arity = new Integer(token.image);
1005 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001006 throw new SqlppParseException(getSourceLocation(token), "Invalid arity:" + arity);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001007 }
1008
1009 // TODO use fctName.library
1010 String fqFunctionName = fctName.library == null ? fctName.function : fctName.library + "#" + fctName.function;
1011 return new FunctionSignature(fctName.dataverse, fqFunctionName, arity);
1012 }
1013}
1014
Yingyi Buc9bfe252016-03-01 00:02:40 -08001015Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001016{
Yingyi Buc9bfe252016-03-01 00:02:40 -08001017 Pair<Integer, List<String>> tmp = null;
1018 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001019 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
1020}
1021{
1022 <PRIMARY> <KEY> tmp = NestedField()
1023 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001024 keyFieldSourceIndicators.add(tmp.first);
1025 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001026 }
1027 ( <COMMA> tmp = NestedField()
1028 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001029 keyFieldSourceIndicators.add(tmp.first);
1030 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001031 }
1032 )*
1033 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001034 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001035 }
1036}
1037
1038Statement DropStatement() throws ParseException:
1039{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001040 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001041 String id = null;
1042 Pair<Identifier,Identifier> pairId = null;
1043 Triple<Identifier,Identifier,Identifier> tripleId = null;
1044 FunctionSignature funcSig = null;
1045 boolean ifExists = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001046 AbstractStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001047}
1048{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001049 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001050 (
Yingyi Bu1c0fff52016-03-25 20:23:30 -07001051 Dataset() pairId = QualifiedName() ifExists = IfExists()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001052 {
Yingyi Buab817482016-08-19 21:29:31 -07001053 stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001054 }
1055 | <INDEX> tripleId = DoubleQualifiedName() ifExists = IfExists()
1056 {
1057 stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
1058 }
1059 | <NODEGROUP> id = Identifier() ifExists = IfExists()
1060 {
1061 stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
1062 }
1063 | <TYPE> pairId = TypeName() ifExists = IfExists()
1064 {
1065 stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
1066 }
1067 | <DATAVERSE> id = Identifier() ifExists = IfExists()
1068 {
1069 stmt = new DataverseDropStatement(new Identifier(id), ifExists);
1070 }
1071 | <FUNCTION> funcSig = FunctionSignature() ifExists = IfExists()
1072 {
1073 stmt = new FunctionDropStatement(funcSig, ifExists);
1074 }
1075 | <FEED> pairId = QualifiedName() ifExists = IfExists()
1076 {
1077 stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
1078 }
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +03001079 | <INGESTION> <POLICY> pairId = QualifiedName() ifExists = IfExists()
1080 {
1081 stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
1082 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001083 )
1084 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001085 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001086 }
1087}
1088
1089boolean IfExists() throws ParseException :
1090{
1091}
1092{
1093 ( LOOKAHEAD(1) <IF> <EXISTS>
1094 {
1095 return true;
1096 }
1097 )?
1098 {
1099 return false;
1100 }
1101}
1102
1103InsertStatement InsertStatement() throws ParseException:
1104{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001105 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001106 Pair<Identifier,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001107 VariableExpr var = null;
1108 Query query = null;
1109 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001110}
1111{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001112 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001113 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001114 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001115 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001116 if (var == null && returnExpression != null) {
1117 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1118 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001119 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08001120 query.setTopLevel(true);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001121 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second, query, getVarCounter(),
1122 var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001123 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001124 }
1125}
1126
1127UpsertStatement UpsertStatement() throws ParseException:
1128{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001129 Token startToken = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001130 Pair<Identifier,Identifier> nameComponents = null;
1131 VariableExpr var = null;
1132 Query query = null;
1133 Expression returnExpression = null;
1134}
1135{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001136 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001137 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001138 ( <RETURNING> returnExpression = Expression())?
1139 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001140 if (var == null && returnExpression != null) {
1141 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1142 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001143 }
1144 query.setTopLevel(true);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001145 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second, query, getVarCounter(),
1146 var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001147 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001148 }
1149}
1150
1151DeleteStatement DeleteStatement() throws ParseException:
1152{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001153 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001154 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001155 Expression condition = null;
1156 Pair<Identifier, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001157}
1158{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001159 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001160 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001161 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07001162 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001163 if (var == null) {
1164 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1165 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001166 }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001167 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second,
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07001168 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001169 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001170 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001171}
1172
1173UpdateStatement UpdateStatement() throws ParseException:
1174{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001175 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001176 VariableExpr vars;
1177 Expression target;
1178 Expression condition;
1179 UpdateClause uc;
1180 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
1181}
1182{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001183 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001184 <WHERE> condition = Expression()
1185 <LEFTPAREN> (uc = UpdateClause()
1186 {
1187 ucs.add(uc);
1188 }
1189 (<COMMA> uc = UpdateClause()
1190 {
1191 ucs.add(uc);
1192 }
1193 )*) <RIGHTPAREN>
1194 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001195 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001196 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001197 }
1198}
1199
1200UpdateClause UpdateClause() throws ParseException:
1201{
1202 Expression target = null;
1203 Expression value = null ;
1204 InsertStatement is = null;
1205 DeleteStatement ds = null;
1206 UpdateStatement us = null;
1207 Expression condition = null;
1208 UpdateClause ifbranch = null;
1209 UpdateClause elsebranch = null;
1210}
1211{
1212 (<SET> target = Expression() <EQ> value = Expression()
1213 | is = InsertStatement()
1214 | ds = DeleteStatement()
1215 | us = UpdateStatement()
1216 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
1217 <THEN> ifbranch = UpdateClause()
1218 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
1219 {
1220 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
1221 }
1222 )
1223}
1224
1225Statement SetStatement() throws ParseException:
1226{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001227 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001228 String pn = null;
1229 String pv = null;
1230}
1231{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001232 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001233 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001234 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001235 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001236 }
1237}
1238
1239Statement WriteStatement() throws ParseException:
1240{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001241 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001242 String nodeName = null;
1243 String fileName = null;
1244 Query query;
1245 String writerClass = null;
1246 Pair<Identifier,Identifier> nameComponents = null;
1247}
1248{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001249 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07001250 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001251 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001252 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001253 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001254 }
1255}
1256
1257LoadStatement LoadStatement() throws ParseException:
1258{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001259 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001260 Identifier dataverseName = null;
1261 Identifier datasetName = null;
1262 boolean alreadySorted = false;
1263 String adapterName;
1264 Map<String,String> properties;
1265 Pair<Identifier,Identifier> nameComponents = null;
1266}
1267{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001268 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001269 {
1270 dataverseName = nameComponents.first;
1271 datasetName = nameComponents.second;
1272 }
1273 <USING> adapterName = AdapterName() properties = Configuration()
1274 (<PRESORTED>
1275 {
1276 alreadySorted = true;
1277 }
1278 )?
1279 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001280 LoadStatement stmt = new LoadStatement(dataverseName, datasetName, adapterName, properties, alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001281 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001282 }
1283}
1284
1285
1286String AdapterName() throws ParseException :
1287{
1288 String adapterName = null;
1289}
1290{
1291 adapterName = Identifier()
1292 {
1293 return adapterName;
1294 }
1295}
1296
1297Statement CompactStatement() throws ParseException:
1298{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001299 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001300 Pair<Identifier,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001301}
1302{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001303 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001304 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001305 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001306 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001307 }
1308}
1309
Yingyi Buab817482016-08-19 21:29:31 -07001310Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001311{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001312 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001313 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001314}
1315{
1316 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001317 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
1318 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
1319 | <START> { startToken = token; } stmt = StartStatement(startToken)
1320 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07001321 )
1322 {
1323 return stmt;
1324 }
1325}
1326
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001327Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001328{
1329 Pair<Identifier,Identifier> feedNameComponents = null;
1330
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001331 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001332}
1333{
1334 <FEED> feedNameComponents = QualifiedName()
1335 {
1336 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001337 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001338 }
1339}
1340
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001341AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001342{
1343 Pair<Identifier,Identifier> feedNameComponents = null;
1344
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001345 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001346}
1347{
1348 <FEED> feedNameComponents = QualifiedName()
1349 {
1350 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001351 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001352 }
1353}
1354
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001355AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001356{
1357 Pair<Identifier,Identifier> feedNameComponents = null;
1358 Pair<Identifier,Identifier> datasetNameComponents = null;
1359
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001360 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07001361}
1362{
1363 (
1364 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001365 {
1366 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
1367 }
1368 )
Yingyi Buab817482016-08-19 21:29:31 -07001369 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001370 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07001371 }
1372}
1373
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001374AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001375{
1376 Pair<Identifier,Identifier> feedNameComponents = null;
1377 Pair<Identifier,Identifier> datasetNameComponents = null;
1378
1379 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07001380 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001381 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07001382 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08001383 String whereClauseBody = null;
1384 WhereClause whereClause = null;
1385 Token beginPos = null;
1386 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07001387}
1388{
1389 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001390 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08001391 (ApplyFunction(appliedFunctions))?
1392 (policy = GetPolicy())?
1393 (
1394 <WHERE>
1395 {
1396 beginPos = token;
1397 whereClause = new WhereClause();
1398 Expression whereExpr;
1399 }
1400 whereExpr = Expression()
1401 {
1402 whereClause.setWhereExpr(whereExpr);
1403 }
1404 )?
1405 {
1406 if (whereClause != null) {
1407 endPos = token;
1408 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
1409 }
1410 }
Yingyi Buab817482016-08-19 21:29:31 -07001411 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07001412 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08001413 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07001414 }
1415 )
1416 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001417 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07001418 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001419}
1420
1421Map<String,String> Configuration() throws ParseException :
1422{
1423 Map<String,String> configuration = new LinkedHashMap<String,String>();
1424 Pair<String, String> keyValuePair = null;
1425}
1426{
1427 <LEFTPAREN> ( keyValuePair = KeyValuePair()
1428 {
1429 configuration.put(keyValuePair.first, keyValuePair.second);
1430 }
1431 ( <COMMA> keyValuePair = KeyValuePair()
1432 {
1433 configuration.put(keyValuePair.first, keyValuePair.second);
1434 }
1435 )* )? <RIGHTPAREN>
1436 {
1437 return configuration;
1438 }
1439}
1440
1441Pair<String, String> KeyValuePair() throws ParseException:
1442{
1443 String key;
1444 String value;
1445}
1446{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001447 <LEFTPAREN> key = ConstantString() <EQ> value = ConstantString() <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001448 {
1449 return new Pair<String, String>(key, value);
1450 }
1451}
1452
1453Map<String,String> Properties() throws ParseException:
1454{
1455 Map<String,String> properties = new HashMap<String,String>();
1456 Pair<String, String> property;
1457}
1458{
1459 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
1460 {
1461 properties.put(property.first, property.second);
1462 }
1463 ( <COMMA> property = Property()
1464 {
1465 properties.put(property.first, property.second);
1466 }
1467 )* <RIGHTPAREN> )?
1468 {
1469 return properties;
1470 }
1471}
1472
1473Pair<String, String> Property() throws ParseException:
1474{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001475 String key = null;
1476 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001477}
1478{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001479 (key = Identifier() | key = StringLiteral())
1480 <EQ>
1481 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001482 {
1483 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001484 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07001485 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001486 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001487 }
1488 }
1489 )
1490 {
1491 return new Pair<String, String>(key.toUpperCase(), value);
1492 }
1493}
1494
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001495IndexedTypeExpression IndexedTypeExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001496{
1497 TypeExpression typeExpr = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001498 boolean isUnknownable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001499}
1500{
1501 (
1502 typeExpr = TypeReference()
1503 | typeExpr = OrderedListTypeDef()
1504 | typeExpr = UnorderedListTypeDef()
1505 )
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001506 ( <QUES> { isUnknownable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001507 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001508 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001509 }
1510}
1511
1512TypeExpression TypeExpr() throws ParseException:
1513{
1514 TypeExpression typeExpr = null;
1515}
1516{
1517 (
1518 typeExpr = RecordTypeDef()
1519 | typeExpr = TypeReference()
1520 | typeExpr = OrderedListTypeDef()
1521 | typeExpr = UnorderedListTypeDef()
1522 )
1523 {
1524 return typeExpr;
1525 }
1526}
1527
1528RecordTypeDefinition RecordTypeDef() throws ParseException:
1529{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001530 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001531 RecordTypeDefinition recType = new RecordTypeDefinition();
1532 RecordTypeDefinition.RecordKind recordKind = null;
1533}
1534{
1535 ( <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
1536 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; } )?
1537 <LEFTBRACE>
1538 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001539 startToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001540 String hint = getHint(token);
1541 if (hint != null) {
1542 String splits[] = hint.split(" +");
1543 if (splits[0].equals(GEN_FIELDS_HINT)) {
1544 if (splits.length != 5) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001545 throw new SqlppParseException(getSourceLocation(getHintToken(token)),
1546 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07001547 }
1548 if (!splits[1].equals("int")) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001549 throw new SqlppParseException(getSourceLocation(getHintToken(token)),
1550 "The only supported type for gen-fields is int.");
Yingyi Bu391f09e2015-10-29 13:49:39 -07001551 }
1552 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
1553 Integer.parseInt(splits[2]), Integer.parseInt(splits[3]), splits[4]);
1554 recType.setUndeclaredFieldsDataGen(ufdg);
1555 }
1556 }
1557
1558 }
1559 (
1560 RecordField(recType)
1561 ( <COMMA> RecordField(recType) )*
1562 )?
1563 <RIGHTBRACE>
1564 {
1565 if (recordKind == null) {
1566 recordKind = RecordTypeDefinition.RecordKind.OPEN;
1567 }
1568 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001569 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001570 }
1571}
1572
1573void RecordField(RecordTypeDefinition recType) throws ParseException:
1574{
1575 String fieldName;
1576 TypeExpression type = null;
1577 boolean nullable = false;
1578}
1579{
1580 fieldName = Identifier()
1581 {
1582 String hint = getHint(token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001583 IRecordFieldDataGen rfdg = hint != null ? parseFieldDataGen(hint, token.specialToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001584 }
1585 <COLON> type = TypeExpr() (<QUES> { nullable = true; } )?
1586 {
1587 recType.addField(fieldName, type, nullable, rfdg);
1588 }
1589}
1590
1591TypeReferenceExpression TypeReference() throws ParseException:
1592{
Abdullah Alamoudie4b318f2016-09-19 13:31:25 +03001593 Pair<Identifier,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001594}
1595{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001596 id = QualifiedName()
1597 {
1598 if (id.first == null && id.second.getValue().equalsIgnoreCase("int")) {
1599 id.second.setValue("int64");
1600 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001601
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001602 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001603 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001604 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001605}
1606
1607OrderedListTypeDefinition OrderedListTypeDef() throws ParseException:
1608{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001609 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001610 TypeExpression type = null;
1611}
1612{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001613 <LEFTBRACKET> { startToken = token; }
1614 ( type = TypeExpr() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07001615 <RIGHTBRACKET>
1616 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001617 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001618 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001619 }
1620}
1621
Yingyi Bu391f09e2015-10-29 13:49:39 -07001622UnorderedListTypeDefinition UnorderedListTypeDef() throws ParseException:
1623{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001624 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001625 TypeExpression type = null;
1626}
1627{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001628 <LEFTDBLBRACE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001629 ( type = TypeExpr() )
1630 <RIGHTDBLBRACE>
1631 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001632 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001633 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001634 }
1635}
1636
1637FunctionName FunctionName() throws ParseException:
1638{
1639 String first = null;
1640 String second = null;
1641 String third = null;
1642 boolean secondAfterDot = false;
1643}
1644{
Michael Blowd6cf6412016-06-30 02:44:35 -04001645 first = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001646 {
1647 FunctionName result = new FunctionName();
1648 result.hint = getHint(token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001649 result.sourceLoc = getSourceLocation(token);
Michael Blowd6cf6412016-06-30 02:44:35 -04001650 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001651 ( <DOT> second = Identifier()
1652 {
1653 secondAfterDot = true;
1654 }
1655 (<SHARP> third = Identifier())? | <SHARP> second = Identifier() )?
1656 {
1657 if (second == null) {
1658 result.dataverse = defaultDataverse;
1659 result.library = null;
1660 result.function = first;
1661 } else if (third == null) {
1662 if (secondAfterDot) {
1663 result.dataverse = first;
1664 result.library = null;
1665 result.function = second;
1666 } else {
1667 result.dataverse = defaultDataverse;
1668 result.library = first;
1669 result.function = second;
1670 }
1671 } else {
1672 result.dataverse = first;
1673 result.library = second;
1674 result.function = third;
1675 }
1676
1677 if (result.function.equalsIgnoreCase("int")) {
1678 result.function = "int64";
1679 }
1680 return result;
1681 }
1682}
1683
1684Pair<Identifier,Identifier> TypeName() throws ParseException:
1685{
1686 Pair<Identifier,Identifier> name = null;
1687}
1688{
1689 name = QualifiedName()
1690 {
1691 if (name.first == null) {
1692 name.first = new Identifier(defaultDataverse);
1693 }
1694 return name;
1695 }
1696}
1697
1698String Identifier() throws ParseException:
1699{
1700 String lit = null;
1701}
1702{
1703 (<IDENTIFIER>
1704 {
1705 return token.image;
1706 }
1707 | lit = QuotedString()
1708 {
1709 return lit;
1710 }
1711 )
1712}
1713
Yingyi Bu1c0fff52016-03-25 20:23:30 -07001714void Dataset() throws ParseException:
1715{
1716}
1717{
1718 (<DATASET>|<COLLECTION>)
1719}
1720
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001721Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001722{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001723 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001724 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001725}
1726{
1727 fieldList = NestedField()
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001728 ( <COLON> fieldType = IndexedTypeExpr() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001729 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001730 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
1731 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07001732 }
1733}
1734
Yingyi Buc9bfe252016-03-01 00:02:40 -08001735Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001736{
1737 List<String> exprList = new ArrayList<String>();
1738 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001739 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001740 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001741}
1742{
1743 lit = Identifier()
1744 {
Yingyi Bub9169b62016-02-26 21:21:49 -08001745 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001746 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08001747 }
1748 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08001749 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08001750 <LEFTPAREN><RIGHTPAREN>
1751 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001752 if(!lit.equalsIgnoreCase("meta")){
1753 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08001754 }
1755 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001756 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08001757 }
1758 )?
1759 {
1760 if(!meetParens){
1761 exprList.add(lit);
1762 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001763 }
1764 (<DOT>
1765 lit = Identifier()
1766 {
1767 exprList.add(lit);
1768 }
1769 )*
1770 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001771 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001772 }
1773}
1774
Yingyi Bu6d57e492016-06-06 21:24:42 -07001775String ConstantString() throws ParseException:
1776{
1777 String value = null;
1778}
1779{
1780 (value = QuotedString() | value = StringLiteral())
1781 {
1782 return value;
1783 }
1784}
1785
Yingyi Bu391f09e2015-10-29 13:49:39 -07001786String QuotedString() throws ParseException:
1787{
1788}
1789{
1790 <QUOTED_STRING>
1791 {
1792 return removeQuotesAndEscapes(token.image);
1793 }
1794}
1795
Yingyi Bu391f09e2015-10-29 13:49:39 -07001796String StringLiteral() throws ParseException:
1797{
1798}
1799{
1800 <STRING_LITERAL>
1801 {
1802 return removeQuotesAndEscapes(token.image);
1803 }
1804}
1805
1806Pair<Identifier,Identifier> QualifiedName() throws ParseException:
1807{
1808 String first = null;
1809 String second = null;
1810}
1811{
1812 first = Identifier() (<DOT> second = Identifier())?
1813 {
1814 Identifier id1 = null;
1815 Identifier id2 = null;
1816 if (second == null) {
1817 id2 = new Identifier(first);
1818 } else
1819 {
1820 id1 = new Identifier(first);
1821 id2 = new Identifier(second);
1822 }
1823 return new Pair<Identifier,Identifier>(id1, id2);
1824 }
1825}
1826
1827Triple<Identifier,Identifier,Identifier> DoubleQualifiedName() throws ParseException:
1828{
1829 String first = null;
1830 String second = null;
1831 String third = null;
1832}
1833{
1834 first = Identifier() <DOT> second = Identifier() (<DOT> third = Identifier())?
1835 {
1836 Identifier id1 = null;
1837 Identifier id2 = null;
1838 Identifier id3 = null;
1839 if (third == null) {
1840 id2 = new Identifier(first);
1841 id3 = new Identifier(second);
1842 } else {
1843 id1 = new Identifier(first);
1844 id2 = new Identifier(second);
1845 id3 = new Identifier(third);
1846 }
1847 return new Triple<Identifier,Identifier,Identifier>(id1, id2, id3);
1848 }
1849}
1850
1851FunctionDecl FunctionDeclaration() throws ParseException:
1852{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001853 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001854 String functionName;
1855 List<VarIdentifier> paramList = new ArrayList<VarIdentifier>();
1856 Expression funcBody;
1857 createNewScope();
1858}
1859{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001860 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07001861 functionName = Identifier()
1862 paramList = ParameterList()
1863 <LEFTBRACE>
1864 (funcBody = SelectExpression(true) | funcBody = Expression())
1865 <RIGHTBRACE>
1866 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001867 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, paramList.size());
Xikui Wang3de700a2018-03-15 16:32:55 -07001868 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001869 FunctionDecl stmt = new FunctionDecl(signature, paramList, funcBody);
Xikui Wang3de700a2018-03-15 16:32:55 -07001870 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001871 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07001872 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001873}
1874
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001875Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001876{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001877 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001878 Expression expr;
1879}
1880{
1881 (
1882 expr = Expression()
1883 |
1884 expr = SelectExpression(false)
1885 )
1886 {
1887 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001888 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07001889 return query;
1890 }
1891}
1892
1893
Yingyi Bu391f09e2015-10-29 13:49:39 -07001894Expression Expression():
1895{
1896 Expression expr = null;
1897 Expression exprP = null;
1898}
1899{
1900(
1901 LOOKAHEAD(2)
1902 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001903 | expr = QuantifiedExpression()
1904)
1905 {
1906 return (exprP==null) ? expr : exprP;
1907 }
1908}
1909
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001910Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001911{
1912 OperatorExpr op = null;
1913 Expression operand = null;
1914}
1915{
1916 operand = AndExpr()
1917 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07001918 <OR>
1919 {
1920 if (op == null) {
1921 op = new OperatorExpr();
1922 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07001923 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001924 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001925 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07001926 try{
1927 op.addOperator(token.image.toLowerCase());
1928 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001929 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07001930 }
Xikui Wang3de700a2018-03-15 16:32:55 -07001931 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001932
Xikui Wang3de700a2018-03-15 16:32:55 -07001933 operand = AndExpr()
1934 {
1935 op.addOperand(operand);
1936 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001937
1938 )*
1939
1940 {
1941 return op==null? operand: op;
1942 }
1943}
1944
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001945Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001946{
1947 OperatorExpr op = null;
1948 Expression operand = null;
1949}
1950{
Yingyi Bu196db5d2016-07-15 19:07:20 -07001951 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001952 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07001953 <AND>
1954 {
1955 if (op == null) {
1956 op = new OperatorExpr();
1957 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07001958 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001959 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001960 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07001961 try{
1962 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08001963 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001964 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07001965 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001966 }
1967
Yingyi Bu196db5d2016-07-15 19:07:20 -07001968 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001969 {
1970 op.addOperand(operand);
1971 }
1972
1973 )*
1974
1975 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001976 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001977 }
1978}
1979
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001980Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07001981{
1982 Expression inputExpr;
1983 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001984 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07001985}
1986{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07001987 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07001988 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001989 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08001990 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001991 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001992 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07001993 } else {
1994 return inputExpr;
1995 }
1996 }
1997}
Yingyi Bu391f09e2015-10-29 13:49:39 -07001998
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001999Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002000{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002001 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002002 OperatorExpr op = null;
2003 Expression operand = null;
2004 boolean broadcast = false;
2005 IExpressionAnnotation annotation = null;
2006}
2007{
Yingyi Bu6c638342016-09-02 17:54:34 -07002008 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002009
2010 (
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002011 LOOKAHEAD(2)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN>)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002012 {
2013 String mhint = getHint(token);
2014 if (mhint != null) {
2015 if (mhint.equals(INDEXED_NESTED_LOOP_JOIN_HINT)) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002016 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
2017 } else if (mhint.equals(SKIP_SECONDARY_INDEX_SEARCH_HINT)) {
2018 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Yingyi Buea4ec722016-11-04 01:26:16 -07002019 } else if (mhint.equals(BROADCAST_JOIN_HINT)) {
2020 broadcast = true;
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002021 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002022 }
Yingyi Buea4ec722016-11-04 01:26:16 -07002023
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002024 String operator = token.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07002025 if (operator.equals("<>")){
2026 operator = "!=";
2027 }
2028 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002029 operator = "not_" + operator;
2030 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002031 if (op == null) {
2032 op = new OperatorExpr();
Yingyi Buea4ec722016-11-04 01:26:16 -07002033 op.addOperand(operand, false); // broadcast is always for the right branch
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002034 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002035 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002036 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002037 try{
2038 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002039 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002040 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002041 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002042 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002043
Yingyi Bu6c638342016-09-02 17:54:34 -07002044 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07002045 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002046 op.addOperand(operand, broadcast);
Yingyi Buea4ec722016-11-04 01:26:16 -07002047 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002048 )?
2049
2050 {
2051 if (annotation != null) {
2052 op.addHint(annotation);
2053 }
2054 return op==null? operand: op;
2055 }
2056}
2057
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002058Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07002059{
2060 boolean not = false;
2061 OperatorExpr op = null;
2062 Expression operand = null;
2063 IExpressionAnnotation annotation = null;
2064}
2065{
2066 operand = IsExpr()
2067 (
2068 LOOKAHEAD(2)
2069 (<NOT> { not = true; })? <BETWEEN>
2070 {
2071 String mhint = getHint(token);
2072 if (mhint != null) {
2073 if (mhint.equals(INDEXED_NESTED_LOOP_JOIN_HINT)) {
2074 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
2075 } else if (mhint.equals(SKIP_SECONDARY_INDEX_SEARCH_HINT)) {
2076 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
2077 }
2078 }
2079 String operator = token.image.toLowerCase();
2080 if(not){
2081 operator = "not_" + operator;
2082 }
2083 if (op == null) {
2084 op = new OperatorExpr();
2085 op.addOperand(operand);
2086 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002087 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07002088 }
2089 try{
2090 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002091 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002092 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07002093 }
2094 }
2095
2096 operand = IsExpr()
2097 {
2098 op.addOperand(operand);
2099 }
2100
2101 <AND>
2102 operand = IsExpr()
2103 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08002104 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07002105 op.addOperand(operand);
2106 }
2107 )?
2108
2109 {
2110 if (annotation != null) {
2111 op.addHint(annotation);
2112 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002113 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07002114 }
2115}
2116
Yingyi Budaa549c2016-06-28 22:30:52 -07002117Expression IsExpr() throws ParseException:
2118{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002119 Token notToken = null;
2120 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002121 Expression operand = null;
2122 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002123 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002124}
2125{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002126 operand = LikeExpr()
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002127 ( <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002128 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002129 (
2130 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
2131 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08002132 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07002133 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002134 )
Yingyi Budaa549c2016-06-28 22:30:52 -07002135 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002136 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07002137 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002138 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002139 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002140 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07002141 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002142 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07002143 }
2144 }
2145 )?
2146 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002147 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07002148 }
2149}
2150
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002151Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002152{
2153 boolean not = false;
2154 OperatorExpr op = null;
2155 Expression operand = null;
2156}
2157{
2158 operand = ConcatExpr()
2159 (
2160 LOOKAHEAD(2)
2161 (<NOT> { not = true; })? <LIKE>
2162 {
2163 op = new OperatorExpr();
2164 op.addOperand(operand);
2165 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002166 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002167
2168 String operator = token.image.toLowerCase();
2169 if (not) {
2170 operator = "not_" + operator;
2171 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002172 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002173 op.addOperator(operator);
2174 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002175 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002176 }
2177 }
2178
2179 operand = ConcatExpr()
2180 {
2181 op.addOperand(operand);
2182 }
2183 )?
2184
2185 {
2186 return op == null ? operand : op;
2187 }
2188}
2189
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002190Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002191{
2192 OperatorExpr op = null;
2193 Expression operand = null;
2194}
2195{
2196 operand = AddExpr()
2197 (
2198 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002199 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002200 {
2201 if (op == null) {
2202 op = new OperatorExpr();
2203 op.addOperand(operand);
2204 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002205 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002206 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002207 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002208 }
2209 operand = AddExpr()
2210 {
2211 op.addOperand(operand);
2212 }
2213 )*
2214
2215 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002216 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002217 }
2218}
Yingyi Budaa549c2016-06-28 22:30:52 -07002219
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002220Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002221{
2222 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002223 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002224 Expression operand = null;
2225}
2226{
2227 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07002228 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002229 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002230 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002231 {
2232 if (op == null) {
2233 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002234 op.addOperand(operand);
2235 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002236 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002237 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002238 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002239 }
2240
2241 operand = MultExpr()
2242 {
2243 op.addOperand(operand);
2244 }
2245 )*
2246
2247 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002248 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002249 }
2250}
2251
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002252Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002253{
2254 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002255 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002256 Expression operand = null;
2257}
2258{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002259 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002260
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002261 ( (
2262 <MUL> { opType = OperatorType.MUL; } |
2263 <DIVIDE> { opType = OperatorType.DIVIDE; } |
2264 <DIV> { opType = OperatorType.DIV; } |
2265 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
2266 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002267 {
2268 if (op == null) {
2269 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002270 op.addOperand(operand);
2271 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002272 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002273 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002274 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002275 }
2276 operand = ExponentExpr()
2277 {
2278 op.addOperand(operand);
2279 }
2280 )*
2281
2282 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002283 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002284 }
2285}
2286
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002287Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002288{
2289 OperatorExpr op = null;
2290 Expression operand = null;
2291}
2292{
2293 operand = UnaryExpr()
2294 (<CARET>
2295 {
2296 if (op == null) {
2297 op = new OperatorExpr();
2298 op.addOperand(operand);
2299 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002300 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002301 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002302 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002303 }
2304 operand = UnaryExpr()
2305 {
2306 op.addOperand(operand);
2307 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002308 )?
2309 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002310 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002311 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002312}
2313
2314Expression UnaryExpr() throws ParseException:
2315{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002316 boolean not = false;
2317 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002318 Expression expr = null;
2319}
Yingyi Budaa549c2016-06-28 22:30:52 -07002320{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002321 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002322 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002323 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002324 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002325 exprType = "not_" + exprType;
2326 }
2327 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002328 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002329 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002330 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002331 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002332 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07002333 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002334 }
2335 )?
2336
2337 expr = ValueExpr()
2338 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002339 if (uexpr == null) {
2340 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002341 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002342 uexpr.setExpr(expr);
2343 return uexpr;
2344 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002345 }
2346}
2347
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002348Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002349{
2350 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002351 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002352}
2353{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002354 expr = PrimaryExpr()
2355 (
2356 accessor = FieldAccessor(accessor != null ? accessor : expr)
2357 |
2358 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002359 )*
2360 {
2361 return accessor == null ? expr : accessor;
2362 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002363}
2364
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002365FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002366{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002367 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002368 String ident = null;
2369}
2370{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002371 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002372 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002373 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002374 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002375 }
2376}
2377
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002378AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002379{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002380 Token startToken = null;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002381 boolean isListSliceExpression = false;
2382 AbstractAccessor resultExpression = null;
2383 Expression argumentExpression1 = null;
2384 Expression argumentExpression2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002385}
2386{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002387 <LEFTBRACKET> { startToken = token; }
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002388 ( argumentExpression1 = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002389 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002390 if (argumentExpression1.getKind() == Expression.Kind.LITERAL_EXPRESSION)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002391 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002392 Literal lit = ((LiteralExpr)argumentExpression1).getValue();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002393 if (lit.getLiteralType() != Literal.Type.INTEGER &&
2394 lit.getLiteralType() != Literal.Type.LONG) {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002395 throw new SqlppParseException(argumentExpression1.getSourceLocation(), "Index should be an INTEGER");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002396 }
2397 }
2398 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002399 )
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002400 (<COLON>
2401 {
2402 isListSliceExpression = true;
2403 }
2404 ( argumentExpression2 = Expression()
2405 {
2406 if (argumentExpression2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
2407 Literal lit = ((LiteralExpr)argumentExpression2).getValue();
2408 if (lit.getLiteralType() != Literal.Type.INTEGER &&
2409 lit.getLiteralType() != Literal.Type.LONG) {
2410 throw new SqlppParseException(argumentExpression2.getSourceLocation(), "Index should be an INTEGER");
2411 }
2412 }
2413 })?
2414 )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002415 <RIGHTBRACKET>
2416 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002417 if (!isListSliceExpression) {
2418 resultExpression = new IndexAccessor(inputExpr, argumentExpression1);
2419 } else {
2420 resultExpression = new ListSliceExpression(inputExpr, argumentExpression1, argumentExpression2);
2421 }
2422
2423 return addSourceLocation(resultExpression, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002424 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002425}
2426
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002427Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002428{
2429 Expression expr = null;
2430}
2431{
2432 ( LOOKAHEAD(4)
2433 expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07002434 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002435 | expr = Literal()
2436 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07002437 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002438 | expr = ListConstructor()
2439 | expr = RecordConstructor()
2440 | expr = ParenthesizedExpression()
2441 )
2442 {
2443 return expr;
2444 }
2445}
2446
2447Expression Literal() throws ParseException:
2448{
2449 LiteralExpr lit = new LiteralExpr();
2450 String str = null;
2451}
2452{
2453 ( str = StringLiteral()
2454 {
2455 lit.setValue(new StringLiteral(str));
2456 }
2457 | <INTEGER_LITERAL>
2458 {
Till Westmann68c6a992016-09-30 00:19:12 -07002459 try {
2460 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
2461 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002462 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002463 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002464 }
2465 | <FLOAT_LITERAL>
2466 {
Till Westmann68c6a992016-09-30 00:19:12 -07002467 try {
2468 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
2469 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002470 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002471 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002472 }
2473 | <DOUBLE_LITERAL>
2474 {
Till Westmann68c6a992016-09-30 00:19:12 -07002475 try {
2476 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
2477 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002478 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002479 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002480 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07002481 | <MISSING>
2482 {
2483 lit.setValue(MissingLiteral.INSTANCE);
2484 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002485 | <NULL>
2486 {
2487 lit.setValue(NullLiteral.INSTANCE);
2488 }
2489 | <TRUE>
2490 {
2491 lit.setValue(TrueLiteral.INSTANCE);
2492 }
2493 | <FALSE>
2494 {
2495 lit.setValue(FalseLiteral.INSTANCE);
2496 }
2497 )
2498 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002499 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002500 }
2501}
2502
Yingyi Bu391f09e2015-10-29 13:49:39 -07002503VariableExpr VariableRef() throws ParseException:
2504{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002505 VarIdentifier var = new VarIdentifier();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002506 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002507}
2508{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002509 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
2510 {
Yingyi Buacc12a92016-03-26 17:25:05 -07002511 id = SqlppVariableUtil.toInternalVariableName(id); // Prefix user-defined variables with "$"
Yingyi Bu391f09e2015-10-29 13:49:39 -07002512 Identifier ident = lookupSymbol(id);
2513 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002514 throw new SqlppParseException(getSourceLocation(token),
2515 "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 -07002516 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002517 VariableExpr varExp = new VariableExpr();
2518 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07002519 varExp.setVar((VarIdentifier)ident);
2520 } else {
2521 varExp.setVar(var);
Yingyi Buacc12a92016-03-26 17:25:05 -07002522 varExp.setIsNewVar(false);
2523 var.setValue(id);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002524 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002525 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002526 }
2527}
2528
Yingyi Bu391f09e2015-10-29 13:49:39 -07002529VariableExpr Variable() throws ParseException:
2530{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002531 VarIdentifier var = new VarIdentifier();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002532 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002533}
2534{
Yingyi Bu391f09e2015-10-29 13:49:39 -07002535 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
2536 {
Yingyi Buacc12a92016-03-26 17:25:05 -07002537 id = SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
Yingyi Bu391f09e2015-10-29 13:49:39 -07002538 Identifier ident = lookupSymbol(id);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002539 VariableExpr varExp = new VariableExpr();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002540 if(ident != null) { // exist such ident
2541 varExp.setIsNewVar(false);
2542 }
2543 varExp.setVar(var);
Yingyi Bucaea8f02015-11-16 15:12:15 -08002544 var.setValue(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002545 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002546 }
2547}
2548
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002549Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
2550{
2551 VariableExpr var = null;
2552 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
2553}
2554{
2555 var = Variable()
2556 ( LOOKAHEAD(1)
2557 {
2558 VariableExpr fieldVarExpr = null;
2559 String fieldIdentifierStr = null;
2560 }
2561 <LEFTPAREN>
2562 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
2563 {
2564 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
2565 }
2566 (<COMMA>
2567 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
2568 {
2569 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
2570 }
2571 )*
2572 <RIGHTPAREN>
2573 )?
2574 {
2575 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
2576 }
2577}
2578
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07002579VariableExpr ExternalVariableRef() throws ParseException:
2580{
2581 String name = null;
2582}
2583{
2584 (
2585 (
2586 <DOLLAR>
2587 (
2588 <INTEGER_LITERAL> { name = token.image; } |
2589 <IDENTIFIER> { name = token.image; } |
2590 name = QuotedString()
2591 )
2592 )
2593 |
2594 (
2595 <QUES> { name = String.valueOf(++externalVarCounter); }
2596 )
2597 )
2598 {
2599 String idName = SqlppVariableUtil.toExternalVariableName(name);
2600 VarIdentifier id = new VarIdentifier(idName);
2601 VariableExpr varExp = new VariableExpr(id);
2602 return addSourceLocation(varExp, token);
2603 }
2604}
2605
Yingyi Bu391f09e2015-10-29 13:49:39 -07002606Expression ListConstructor() throws ParseException:
2607{
2608 Expression expr = null;
2609}
2610{
2611 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002612 expr = OrderedListConstructor() |
2613 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002614 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002615 {
2616 return expr;
2617 }
2618}
2619
Yingyi Bu391f09e2015-10-29 13:49:39 -07002620ListConstructor OrderedListConstructor() throws ParseException:
2621{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002622 Token startToken = null;
2623 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002624}
2625{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002626 <LEFTBRACKET> { startToken = token; }
2627 exprList = ExpressionList()
2628 <RIGHTBRACKET>
2629 {
2630 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002631 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002632 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002633}
2634
2635ListConstructor UnorderedListConstructor() throws ParseException:
2636{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002637 Token startToken = null;
2638 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002639}
2640{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002641 <LEFTDBLBRACE> { startToken = token; }
2642 exprList = ExpressionList()
2643 <RIGHTDBLBRACE>
2644 {
2645 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002646 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002647 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002648}
2649
2650List<Expression> ExpressionList() throws ParseException:
2651{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002652 Expression expr = null;
2653 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002654}
2655{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002656 (
2657 expr = Expression()
2658 {
2659 exprList.add(expr);
2660 }
2661 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07002662 {
2663 exprList.add(expr);
2664 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002665 )*
2666 )?
2667 {
2668 return exprList;
2669 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002670}
2671
Yingyi Bu391f09e2015-10-29 13:49:39 -07002672RecordConstructor RecordConstructor() throws ParseException:
2673{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002674 Token startToken = null;
2675 FieldBinding fb = null;
2676 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002677}
2678{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002679 <LEFTBRACE> { startToken = token; }
2680 (
2681 fb = FieldBinding() { fbList.add(fb); }
2682 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
2683 )?
2684 <RIGHTBRACE>
2685 {
2686 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002687 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002688 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002689}
2690
2691FieldBinding FieldBinding() throws ParseException:
2692{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002693 Expression left, right;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002694}
2695{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002696 left = Expression() <COLON> right = Expression()
2697 {
2698 return new FieldBinding(left, right);
2699 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002700}
2701
Yingyi Bu391f09e2015-10-29 13:49:39 -07002702Expression FunctionCallExpr() throws ParseException:
2703{
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002704 Expression resultExpr;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002705 CallExpr callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002706 List<Expression> argList = new ArrayList<Expression>();
Yingyi Buf4d09842016-08-26 00:03:52 -07002707 Expression tmp = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002708 int arity = 0;
2709 FunctionName funcName = null;
2710 String hint = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07002711 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002712 boolean distinct = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002713}
2714{
2715 funcName = FunctionName()
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002716 {
2717 hint = funcName.hint;
2718 }
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002719 <LEFTPAREN> (
2720 ( <DISTINCT> { distinct = true; } )?
2721 ( tmp = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002722 {
Yingyi Buf4d09842016-08-26 00:03:52 -07002723 if(star){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002724 if(!funcName.function.equalsIgnoreCase("count")){
2725 throw new SqlppParseException(getSourceLocation(token), "The parameter * can only be used in COUNT().");
Yingyi Buf4d09842016-08-26 00:03:52 -07002726 }
2727 argList.add(new LiteralExpr(new LongIntegerLiteral(1L)));
2728 } else {
2729 argList.add(tmp);
2730 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002731 arity ++;
2732 }
2733 (<COMMA> tmp = Expression()
2734 {
2735 argList.add(tmp);
2736 arity++;
2737 }
2738 )*)? <RIGHTPAREN>
2739 {
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002740 String name = funcName.function;
2741 if (distinct) {
Dmitry Lychagine5ea42d2019-03-15 11:38:09 -07002742 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002743 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002744 // TODO use funcName.library
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07002745 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002746 FunctionSignature signature
2747 = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
2748 if (signature == null) {
2749 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
2750 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002751 callExpr = FunctionMapUtil.normalizedListInputFunctions(new CallExpr(signature,argList));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002752 if (hint != null) {
2753 if (hint.startsWith(INDEXED_NESTED_LOOP_JOIN_HINT)) {
2754 callExpr.addHint(IndexedNLJoinExpressionAnnotation.INSTANCE);
2755 } else if (hint.startsWith(SKIP_SECONDARY_INDEX_SEARCH_HINT)) {
2756 callExpr.addHint(SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE);
2757 }
2758 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002759 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002760 resultExpr = callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002761 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002762
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002763 ( LOOKAHEAD(5) resultExpr = WindowExpr(callExpr.getFunctionSignature(), callExpr.getExprList()) )?
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07002764
2765 {
2766 return resultExpr;
2767 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002768}
2769
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002770WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList) throws ParseException:
2771{
2772 Boolean fromLast = null, ignoreNulls = null;
2773}
2774{
2775 (
2776 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
2777 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
2778 <FROM> <IDENTIFIER>
2779 {
2780 if (isToken(FIRST)) {
2781 fromLast = false;
2782 } else if (isToken(LAST)) {
2783 fromLast = true;
2784 } else {
2785 throw createUnexpectedTokenError();
2786 }
2787 }
2788 )?
2789 (
2790 // ( RESPECT | IGNORE ) NULLS OVER
2791 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
2792 <IDENTIFIER>
2793 {
2794 if (isToken(RESPECT)) {
2795 ignoreNulls = false;
2796 } else if (isToken(IGNORE)) {
2797 ignoreNulls = true;
2798 } else {
2799 throw createUnexpectedTokenError();
2800 }
2801 }
2802 <IDENTIFIER>
2803 {
2804 if (!isToken(NULLS)) {
2805 throw createUnexpectedTokenError();
2806 }
2807 }
2808 )?
2809 <OVER>
2810 {
2811 return OverClause(signature, argList, token, fromLast, ignoreNulls);
2812 }
2813}
2814
2815WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Token startToken, Boolean fromLast,
2816 Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002817{
2818 Expression partitionExpr = null;
2819 List<Expression> partitionExprs = new ArrayList<Expression>();
2820 OrderbyClause orderByClause = null;
2821 List<Expression> orderbyList = null;
2822 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
2823 WindowExpression.FrameMode frameMode = null;
2824 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
2825 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
2826 Expression frameStartExpr = null, frameEndExpr = null;
2827 WindowExpression.FrameExclusionKind frameExclusionKind = null;
2828 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
2829 VariableExpr windowVar = null;
2830 List<Pair<Expression, Identifier>> windowFieldList = null;
2831}
2832{
2833 (
2834 windowVarWithFieldList = VariableWithFieldMap() <AS>
2835 {
2836 windowVar = windowVarWithFieldList.first;
2837 windowFieldList = windowVarWithFieldList.second;
2838 }
2839 )?
2840 <LEFTPAREN>
2841 (
2842 <IDENTIFIER> { expectToken(PARTITION); } <BY>
2843 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
2844 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
2845 )?
2846 (
2847 orderByClause = OrderbyClause()
2848 {
2849 orderbyList = orderByClause.getOrderbyList();
2850 orderbyModifierList = orderByClause.getModifierList();
2851 }
2852 (
2853 frameMode = WindowFrameMode()
2854 (
2855 frameStart = WindowFrameBoundary() |
2856 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
2857 )
2858 ( frameExclusionKind = WindowFrameExclusion() )?
2859 {
2860 frameStartKind = frameStart.first;
2861 frameStartExpr = frameStart.second;
2862 if (frameEnd == null) {
2863 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
2864 } else {
2865 frameEndKind = frameEnd.first;
2866 frameEndExpr = frameEnd.second;
2867 }
2868 if (frameExclusionKind == null) {
2869 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
2870 }
2871 }
2872 )?
2873 )?
2874 <RIGHTPAREN>
2875 {
2876 WindowExpression winExp = new WindowExpression(signature, argList, partitionExprs, orderbyList, orderbyModifierList,
2877 frameMode, frameStartKind, frameStartExpr, frameEndKind, frameEndExpr, frameExclusionKind, windowVar,
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002878 windowFieldList, ignoreNulls, fromLast);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002879 return addSourceLocation(winExp, startToken);
2880 }
2881}
2882
2883WindowExpression.FrameMode WindowFrameMode() throws ParseException:
2884{
2885}
2886{
2887 <IDENTIFIER>
2888 {
2889 if (isToken(RANGE)) {
2890 return WindowExpression.FrameMode.RANGE;
2891 } else if (isToken(ROWS)) {
2892 return WindowExpression.FrameMode.ROWS;
2893 } else if (isToken(GROUPS)) {
2894 return WindowExpression.FrameMode.GROUPS;
2895 } else {
2896 throw createUnexpectedTokenError();
2897 }
2898 }
2899}
2900
2901Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
2902{
2903 boolean current = false;
2904 Expression expr = null;
2905}
2906{
2907 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002908 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08002909 | expr = Expression()
2910 )
2911 <IDENTIFIER>
2912 {
2913 WindowExpression.FrameBoundaryKind kind;
2914 if (current && isToken(ROW)) {
2915 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
2916 } else if (!current && isToken(PRECEDING)) {
2917 kind = expr == null
2918 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
2919 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
2920 } else if (!current && isToken(FOLLOWING)) {
2921 kind = expr == null
2922 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
2923 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
2924 } else {
2925 throw createUnexpectedTokenError();
2926 }
2927 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
2928 }
2929}
2930
2931WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
2932{
2933 boolean current = false, no = false;
2934}
2935{
2936 <IDENTIFIER>
2937 {
2938 expectToken(EXCLUDE);
2939 }
2940 (
2941 <GROUP>
2942 {
2943 return WindowExpression.FrameExclusionKind.GROUP;
2944 }
2945 |
2946 (
2947 <IDENTIFIER>
2948 {
2949 if (isToken(TIES)) {
2950 return WindowExpression.FrameExclusionKind.TIES;
2951 } else if (isToken(CURRENT)) {
2952 current = true;
2953 } else if (isToken(NO)) {
2954 no = true;
2955 } else {
2956 throw createUnexpectedTokenError();
2957 }
2958 }
2959 <IDENTIFIER>
2960 {
2961 if (current && isToken(ROW)) {
2962 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
2963 } else if (no && isToken(OTHERS)) {
2964 return WindowExpression.FrameExclusionKind.NO_OTHERS;
2965 } else {
2966 throw createUnexpectedTokenError();
2967 }
2968 }
2969 )
2970 )
2971}
2972
Yingyi Bu391f09e2015-10-29 13:49:39 -07002973Expression ParenthesizedExpression() throws ParseException:
2974{
2975 Expression expr;
2976}
2977{
2978 (
2979 LOOKAHEAD(2)
2980 <LEFTPAREN> expr = Expression() <RIGHTPAREN>
2981 |
2982 expr = Subquery()
2983 )
2984 {
2985 return expr;
2986 }
2987}
2988
Yingyi Buc8c067c2016-07-25 23:37:19 -07002989Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002990{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002991 Token startToken = null;
2992 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07002993 List<Expression> whenExprs = new ArrayList<Expression>();
2994 List<Expression> thenExprs = new ArrayList<Expression>();
2995 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07002996 Expression whenExpr = null;
2997 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002998}
2999{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003000 <CASE> { startToken = token; }
3001 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07003002 (
3003 <WHEN> whenExpr = Expression()
3004 {
3005 whenExprs.add(whenExpr);
3006 }
3007 <THEN> thenExpr = Expression()
3008 {
3009 thenExprs.add(thenExpr);
3010 }
3011 )*
3012 (<ELSE> elseExpr = Expression() )?
3013 <END>
3014 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003015 if (conditionExpr == null) {
3016 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003017 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003018 }
3019 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003020 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07003021 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003022}
3023
Yingyi Buab817482016-08-19 21:29:31 -07003024SelectExpression SelectExpression(boolean subquery) throws ParseException:
3025{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003026 List<LetClause> letClauses = new ArrayList<LetClause>();
3027 SelectSetOperation selectSetOperation;
3028 OrderbyClause orderbyClause = null;
3029 LimitClause limitClause = null;
3030 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07003031}
3032{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003033 ( letClauses = LetClause() )?
3034 selectSetOperation = SelectSetOperation()
3035 (orderbyClause = OrderbyClause() {})?
3036 (limitClause = LimitClause() {})?
3037 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003038 SelectExpression selectExpr =
3039 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
3040 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
3041 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003042 }
3043}
3044
Yingyi Buab817482016-08-19 21:29:31 -07003045SelectSetOperation SelectSetOperation() throws ParseException:
3046{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003047 SetOperationInput setOperationInputLeft;
3048 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
3049}
3050{
3051 {
3052 SelectBlock selectBlockLeft = null;
3053 SelectExpression subqueryLeft = null;
3054 Expression expr = null;
3055 }
3056 selectBlockLeft = SelectBlock()
3057 {
3058 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
3059 }
3060 (
3061 {
3062 SetOpType opType = SetOpType.UNION;
3063 boolean setSemantics = true;
3064 SelectBlock selectBlockRight = null;
3065 SelectExpression subqueryRight = null;
3066 }
3067 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
3068 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
3069 {
3070 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
3071 }
3072 )*
3073 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003074 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
3075 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
3076 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003077 }
3078}
3079
Yingyi Buab817482016-08-19 21:29:31 -07003080SelectExpression Subquery() throws ParseException:
3081{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003082 SelectExpression selectExpr = null;
3083}
3084{
3085 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
3086 {
3087 return selectExpr;
3088 }
3089}
3090
Yingyi Buab817482016-08-19 21:29:31 -07003091SelectBlock SelectBlock() throws ParseException:
3092{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003093 SelectClause selectClause = null;
3094 FromClause fromClause = null;
3095 List<LetClause> fromLetClauses = null;
3096 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003097 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003098 GroupbyClause groupbyClause = null;
3099 List<LetClause> gbyLetClauses = null;
3100 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003101 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003102 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003103}
3104{
3105 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003106 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003107 (
3108 LOOKAHEAD(1)
3109 fromClause = FromClause()
3110 (
3111 LOOKAHEAD(1)
3112 fromLetClauses = LetClause()
3113 )?
3114 )?
3115 (whereClause = WhereClause())?
3116 (
3117 groupbyClause = GroupbyClause()
3118 (
3119 LOOKAHEAD(1)
3120 gbyLetClauses = LetClause()
3121 )?
3122 (havingClause = HavingClause())?
3123 )?
3124 |
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003125 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003126 (
3127 LOOKAHEAD(1)
3128 fromLetClauses = LetClause()
3129 )?
3130 (whereClause = WhereClause())?
3131 (
3132 groupbyClause = GroupbyClause()
3133 (
3134 gbyLetClauses = LetClause()
3135 )?
3136 (havingClause = HavingClause())?
3137 )?
3138 selectClause = SelectClause()
3139 )
3140 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003141 if (fromLetClauses != null) {
3142 fromLetWhereClauses.addAll(fromLetClauses);
3143 }
3144 if (whereClause != null) {
3145 fromLetWhereClauses.add(whereClause);
3146 }
3147 if (gbyLetClauses != null) {
3148 gbyLetHavingClauses.addAll(gbyLetClauses);
3149 }
3150 if (havingClause != null) {
3151 gbyLetHavingClauses.add(havingClause);
3152 }
3153 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
3154 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003155 selectBlock.setSourceLocation(startSrcLoc);
3156 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003157 }
3158}
3159
Yingyi Buab817482016-08-19 21:29:31 -07003160SelectClause SelectClause() throws ParseException:
3161{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003162 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003163 SelectRegular selectRegular = null;
3164 SelectElement selectElement = null;
3165 boolean distinct = false;
3166}
3167{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003168 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003169 (
3170 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04003171 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07003172 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07003173 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003174 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003175 SourceLocation sourceLoc = getSourceLocation(startToken);
3176 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07003177 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003178 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003179 List<Projection> projections = new ArrayList<Projection>();
3180 projections.add(projection);
3181 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003182 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003183 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003184 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
3185 selectClause.setSourceLocation(sourceLoc);
3186 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003187 }
3188}
3189
Yingyi Buab817482016-08-19 21:29:31 -07003190SelectRegular SelectRegular() throws ParseException:
3191{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003192 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003193 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003194 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003195}
3196{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003197 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003198 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003199 projections.add(projection);
3200 startSrcLoc = projection.getSourceLocation();
3201 }
3202 ( LOOKAHEAD(2) <COMMA> projection = Projection()
3203 {
3204 projections.add(projection);
3205 }
3206 )*
3207 {
3208 SelectRegular selectRegular = new SelectRegular(projections);
3209 selectRegular.setSourceLocation(startSrcLoc);
3210 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003211 }
3212}
3213
Yingyi Buab817482016-08-19 21:29:31 -07003214SelectElement SelectElement() throws ParseException:
3215{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003216 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003217 Expression expr = null;
3218 String name = null;
3219}
3220{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003221 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003222 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003223 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003224 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003225 }
3226}
3227
Yingyi Buab817482016-08-19 21:29:31 -07003228Projection Projection() throws ParseException :
3229{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003230 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003231 Expression expr = null;
3232 Identifier identifier = null;
3233 String name = null;
3234 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003235 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003236}
3237{
3238 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003239 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
3240 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003241 | expr = Expression() ((<AS>)? name = Identifier())?
3242 {
3243 if (name == null) {
3244 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
3245 if (generatedColumnIdentifier != null) {
3246 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
3247 }
3248 }
3249 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003250 )
3251 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003252 Projection projection = new Projection(expr, name, star, varStar);
3253 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
3254 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003255 }
3256}
3257
3258FromClause FromClause() throws ParseException :
3259{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003260 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003261 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
3262 extendCurrentScope();
3263}
3264{
3265 {
3266 FromTerm fromTerm = null;
3267 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003268 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003269 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
3270 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003271 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003272 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003273 }
3274}
3275
3276FromTerm FromTerm() throws ParseException :
3277{
3278 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003279 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003280 VariableExpr posVar = null;
3281 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
3282}
3283{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003284 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003285 (
3286 {JoinType joinType = JoinType.INNER; }
3287 (joinType = JoinType())?
3288 {
3289 AbstractBinaryCorrelateClause correlateClause = null;
3290 }
3291 (correlateClause = JoinClause(joinType)
Yingyi Bu6d999ed2016-07-13 11:59:06 -07003292 | correlateClause = UnnestClause(joinType)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003293 )
3294 {
3295 correlateClauses.add(correlateClause);
3296 }
3297 )*
3298 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003299 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003300 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003301 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003302 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
3303 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
3304 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003305 }
3306}
3307
3308JoinClause JoinClause(JoinType joinType) throws ParseException :
3309{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003310 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003311 Expression rightExpr = null;
3312 VariableExpr rightVar = null;
3313 VariableExpr posVar = null;
3314 Expression conditionExpr = null;
3315}
3316{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003317 <JOIN> { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())? <ON> conditionExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003318 {
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003319 if(rightVar==null){
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003320 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003321 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003322 JoinClause joinClause = new JoinClause(joinType, rightExpr, rightVar, posVar, conditionExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003323 return addSourceLocation(joinClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003324 }
3325}
3326
Yingyi Bu391f09e2015-10-29 13:49:39 -07003327UnnestClause UnnestClause(JoinType joinType) throws ParseException :
3328{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003329 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003330 Expression rightExpr;
3331 VariableExpr rightVar;
3332 VariableExpr posVar = null;
3333}
3334{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003335 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable()) (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003336 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003337 if (rightVar == null) {
3338 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003339 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003340 UnnestClause unnestClause = new UnnestClause(joinType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003341 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003342 }
3343}
3344
Yingyi Bu391f09e2015-10-29 13:49:39 -07003345JoinType JoinType() throws ParseException :
3346{
3347 JoinType joinType = JoinType.INNER;
3348}
3349{
3350 (<INNER>|<LEFT> (<OUTER>)? {joinType = JoinType.LEFTOUTER; })
3351 {
3352 return joinType;
3353 }
3354}
3355
3356List<LetClause> LetClause() throws ParseException:
3357{
3358 List<LetClause> letList = new ArrayList<LetClause>();
3359 LetClause letClause;
3360}
3361{
3362 (
3363 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
3364 |
3365 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
3366 )
3367 {
3368 return letList;
3369 }
3370}
3371
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003372WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07003373{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003374 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003375 Expression whereExpr;
3376}
3377{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003378 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003379 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003380 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003381 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003382 }
3383}
3384
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003385OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07003386{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003387 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003388 OrderbyClause oc = new OrderbyClause();
3389 Expression orderbyExpr;
3390 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003391 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003392 int numOfOrderby = 0;
3393}
3394{
3395 <ORDER>
3396 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003397 startToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003398 String hint = getHint(token);
3399 if (hint != null) {
3400 if (hint.startsWith(INMEMORY_HINT)) {
3401 String splits[] = hint.split(" +");
3402 int numFrames = Integer.parseInt(splits[1]);
3403 int numTuples = Integer.parseInt(splits[2]);
3404 oc.setNumFrames(numFrames);
3405 oc.setNumTuples(numTuples);
3406 }
Ali Alsuliman80225e22018-10-15 14:17:07 -07003407 if (hint.startsWith(RANGE_HINT)) {
3408 try {
3409 oc.setRangeMap(RangeMapBuilder.parseHint(createNewParser(hint.substring(RANGE_HINT.length()))));
3410 } catch (CompilationException e) {
3411 throw new SqlppParseException(getSourceLocation(getHintToken(token)), e.getMessage());
3412 }
3413 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003414 }
3415 }
3416 <BY> orderbyExpr = Expression()
3417 {
3418 orderbyList.add(orderbyExpr);
3419 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
3420 }
3421 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
3422 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
3423 {
3424 modifierList.add(modif);
3425 }
3426
3427 (LOOKAHEAD(2) <COMMA> orderbyExpr = Expression()
3428 {
3429 orderbyList.add(orderbyExpr);
3430 modif = OrderbyClause.OrderModifier.ASC;
3431 }
3432 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
3433 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
3434 {
3435 modifierList.add(modif);
3436 }
3437 )*
3438
3439 {
3440 oc.setModifierList(modifierList);
3441 oc.setOrderbyList(orderbyList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003442 return addSourceLocation(oc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003443 }
3444}
3445
3446GroupbyClause GroupbyClause()throws ParseException :
3447{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003448 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003449 GroupbyClause gbc = new GroupbyClause();
3450 List<GbyVariableExpressionPair> vePairList = new ArrayList<GbyVariableExpressionPair>();
3451 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003452 Expression expr = null;
3453 VariableExpr decorVar = null;
3454 Expression decorExpr = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003455 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07003456 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003457 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003458}
3459{
3460 {
3461 Scope newScope = extendCurrentScopeNoPush(true);
3462 // extendCurrentScope(true);
3463 }
3464 <GROUP>
3465 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003466 startToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003467 String hint = getHint(token);
3468 if (hint != null && hint.equals(HASH_GROUP_BY_HINT)) {
3469 gbc.setHashGroupByHint(true);
3470 }
3471 }
3472 <BY> (
3473 expr = Expression()
3474 (LOOKAHEAD(1) (<AS>)?
3475 var = Variable()
Yingyi Bucaea8f02015-11-16 15:12:15 -08003476 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003477 {
3478 GbyVariableExpressionPair pair1 = new GbyVariableExpressionPair(var, expr);
3479 vePairList.add(pair1);
3480 }
3481 ( LOOKAHEAD(1) <COMMA>
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003482 {
3483 var = null;
3484 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003485 expr = Expression()
3486 (LOOKAHEAD(1) (<AS>)?
3487 var = Variable()
Yingyi Bucaea8f02015-11-16 15:12:15 -08003488 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003489 {
3490 GbyVariableExpressionPair pair2 = new GbyVariableExpressionPair(var, expr);
3491 vePairList.add(pair2);
3492 }
3493 )*
3494 )
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003495 (<GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
3496 {
3497 groupVar = groupVarWithFieldList.first;
3498 groupFieldList = groupVarWithFieldList.second;
3499 }
Yingyi Buacc12a92016-03-26 17:25:05 -07003500 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003501 {
3502 gbc.setGbyPairList(vePairList);
3503 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07003504 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07003505 gbc.setGroupVar(groupVar);
3506 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003507 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003508 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003509 }
3510}
3511
3512HavingClause HavingClause() throws ParseException:
3513{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003514 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003515 Expression filterExpr = null;
3516}
3517{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003518 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003519 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003520 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003521 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003522 }
3523}
3524
3525LimitClause LimitClause() throws ParseException:
3526{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003527 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003528 LimitClause lc = new LimitClause();
3529 Expression expr;
3530 pushForbiddenScope(getCurrentScope());
3531}
3532{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003533 <LIMIT> { startToken = token; } expr = Expression() { lc.setLimitExpr(expr); }
3534 (<OFFSET> expr = Expression() { lc.setOffset(expr); })?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003535
3536 {
3537 popForbiddenScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003538 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003539 }
3540}
3541
3542QuantifiedExpression QuantifiedExpression()throws ParseException:
3543{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003544 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003545 QuantifiedExpression qc = new QuantifiedExpression();
3546 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
3547 Expression satisfiesExpr;
3548 VariableExpr var;
3549 Expression inExpr;
3550 QuantifiedPair pair;
3551}
3552{
3553 {
3554 createNewScope();
3555 }
3556
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003557 ( ((<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); })
3558 | (<EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); }))
Yingyi Bu391f09e2015-10-29 13:49:39 -07003559 var = Variable() <IN> inExpr = Expression()
3560 {
3561 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003562 quantifiedList.add(pair);
3563 }
3564 (
3565 <COMMA> var = Variable() <IN> inExpr = Expression()
3566 {
3567 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003568 quantifiedList.add(pair);
3569 }
3570 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07003571 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003572 {
3573 qc.setSatisfiesExpr(satisfiesExpr);
3574 qc.setQuantifiedList(quantifiedList);
3575 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003576 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003577 }
3578}
3579
3580LetClause LetElement() throws ParseException:
3581{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003582 VariableExpr varExp;
3583 Expression beExp;
3584 extendCurrentScope();
3585}
3586{
3587 varExp = Variable() <EQ> beExp = Expression()
3588 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003589 LetClause lc = new LetClause(varExp, beExp);
3590 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003591 return lc;
3592 }
3593}
3594
3595LetClause WithElement() throws ParseException:
3596{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003597 VariableExpr varExp;
3598 Expression beExp;
3599 extendCurrentScope();
3600}
3601{
3602 varExp = Variable() <AS> beExp = Expression()
3603 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003604 LetClause lc = new LetClause(varExp, beExp);
3605 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003606 return lc;
3607 }
3608}
3609
3610TOKEN_MGR_DECLS:
3611{
3612 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07003613 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003614
3615 public void pushState() {
3616 lexerStateStack.push( curLexState );
3617 }
3618
3619 public void popState(String token) {
3620 if (lexerStateStack.size() > 0) {
3621 SwitchTo( lexerStateStack.pop() );
3622 } else {
3623 int errorLine = input_stream.getEndLine();
3624 int errorColumn = input_stream.getEndColumn();
3625 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
3626 + "\" but state stack is empty.";
3627 throw new TokenMgrError(msg, -1);
3628 }
3629 }
3630}
3631
3632<DEFAULT,IN_DBL_BRACE>
3633TOKEN [IGNORE_CASE]:
3634{
3635 <ALL : "all">
3636 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07003637 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003638 | <APPLY : "apply">
3639 | <AS : "as">
3640 | <ASC : "asc">
3641 | <AT : "at">
3642 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003643 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003644 | <BTREE : "btree">
3645 | <BY : "by">
3646 | <CASE : "case">
3647 | <CLOSED : "closed">
3648 | <CREATE : "create">
3649 | <COMPACTION : "compaction">
3650 | <COMPACT : "compact">
3651 | <CONNECT : "connect">
3652 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07003653 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07003654 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07003655 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003656 | <DECLARE : "declare">
3657 | <DEFINITION : "definition">
3658 | <DELETE : "delete">
3659 | <DESC : "desc">
3660 | <DISCONNECT : "disconnect">
3661 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03003662 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003663 | <DROP : "drop">
3664 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07003665 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003666 | <ELSE : "else">
3667 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07003668 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003669 | <EVERY : "every">
3670 | <EXCEPT : "except">
3671 | <EXISTS : "exists">
3672 | <EXTERNAL : "external">
3673 | <FEED : "feed">
3674 | <FILTER : "filter">
3675 | <FLATTEN : "flatten">
3676 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003677 | <FROM : "from">
3678 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08003679 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003680 | <FUNCTION : "function">
3681 | <GROUP : "group">
3682 | <HAVING : "having">
3683 | <HINTS : "hints">
3684 | <IF : "if">
3685 | <INTO : "into">
3686 | <IN : "in">
3687 | <INDEX : "index">
3688 | <INGESTION : "ingestion">
3689 | <INNER : "inner">
3690 | <INSERT : "insert">
3691 | <INTERNAL : "internal">
3692 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07003693 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003694 | <JOIN : "join">
3695 | <KEYWORD : "keyword">
3696 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07003697 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003698 | <LEFT : "left">
3699 | <LETTING : "letting">
3700 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003701 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003702 | <LIMIT : "limit">
3703 | <LOAD : "load">
Murtadha Hubail29f63912019-04-21 14:23:57 +03003704 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003705 | <NODEGROUP : "nodegroup">
3706 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07003707 | <NOT : "not">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003708 | <OFFSET : "offset">
3709 | <ON : "on">
3710 | <OPEN : "open">
3711 | <OR : "or">
3712 | <ORDER : "order">
3713 | <OUTER : "outer">
3714 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003715 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003716 | <PATH : "path">
3717 | <POLICY : "policy">
3718 | <PRESORTED : "pre-sorted">
3719 | <PRIMARY : "primary">
3720 | <RAW : "raw">
3721 | <REFRESH : "refresh">
3722 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08003723 | <RETURNING : "returning">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003724 | <RTREE : "rtree">
3725 | <RUN : "run">
3726 | <SATISFIES : "satisfies">
3727 | <SECONDARY : "secondary">
3728 | <SELECT : "select">
3729 | <SET : "set">
3730 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08003731 | <START : "start">
3732 | <STOP : "stop">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03003733 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07003734 | <THEN : "then">
3735 | <TYPE : "type">
3736 | <TO : "to">
3737 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08003738 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003739 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07003740 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08003741 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07003742 | <USE : "use">
3743 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003744 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08003745 | <VALUED : "valued">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003746 | <WHEN : "when">
3747 | <WHERE : "where">
3748 | <WITH : "with">
3749 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003750}
3751
3752<DEFAULT,IN_DBL_BRACE>
3753TOKEN :
3754{
3755 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003756 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003757 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003758 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003759 | <MUL : "*">
3760 | <PLUS : "+">
3761
3762 | <LEFTPAREN : "(">
3763 | <RIGHTPAREN : ")">
3764 | <LEFTBRACKET : "[">
3765 | <RIGHTBRACKET : "]">
3766
3767 | <ATT : "@">
3768 | <COLON : ":">
3769 | <COMMA : ",">
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003770 | <DOLLAR: "$">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003771 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003772 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003773 | <QUES : "?">
3774 | <SEMICOLON : ";">
3775 | <SHARP : "#">
3776
3777 | <LT : "<">
3778 | <GT : ">">
3779 | <LE : "<=">
3780 | <GE : ">=">
3781 | <EQ : "=">
3782 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07003783 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003784 | <SIMILAR : "~=">
3785}
3786
3787<DEFAULT,IN_DBL_BRACE>
3788TOKEN :
3789{
3790 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
3791}
3792
3793<DEFAULT>
3794TOKEN :
3795{
3796 <RIGHTBRACE : "}"> { popState("}"); }
3797}
3798
3799<DEFAULT,IN_DBL_BRACE>
3800TOKEN :
3801{
3802 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
3803}
3804
3805<IN_DBL_BRACE>
3806TOKEN :
3807{
3808 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
3809}
3810
3811<DEFAULT,IN_DBL_BRACE>
3812TOKEN :
3813{
3814 <INTEGER_LITERAL : (<DIGIT>)+ >
3815}
3816
3817<DEFAULT,IN_DBL_BRACE>
Yingyi Bue311a632016-06-07 18:23:16 -07003818TOKEN [IGNORE_CASE]:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003819{
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003820 <MISSING : "missing">
3821 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07003822 | <TRUE : "true">
3823 | <FALSE : "false">
3824}
3825
3826<DEFAULT,IN_DBL_BRACE>
3827TOKEN :
3828{
3829 <#DIGIT : ["0" - "9"]>
3830}
3831
3832<DEFAULT,IN_DBL_BRACE>
3833TOKEN:
3834{
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07003835 < DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
3836 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
3837 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003838 >
Yingyi Bue4d919e2016-10-30 10:47:03 -07003839 | < FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
3840 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
3841 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003842 >
3843 | <DIGITS : (<DIGIT>)+ >
3844}
3845
3846<DEFAULT,IN_DBL_BRACE>
3847TOKEN :
3848{
3849 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08003850 | <#IDENTIFIER_SPECIALCHARS_START : ["_"]>
3851 | <#IDENTIFIER_SPECIALCHARS_REST : ["$"]>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003852}
3853
3854<DEFAULT,IN_DBL_BRACE>
3855TOKEN :
3856{
3857 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07003858 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003859 <EscapeQuot>
3860 | <EscapeBslash>
3861 | <EscapeSlash>
3862 | <EscapeBspace>
3863 | <EscapeFormf>
3864 | <EscapeNl>
3865 | <EscapeCr>
3866 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07003867 | ~["`","\\"])* "`">
3868 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003869 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003870 | <EscapeBslash>
3871 | <EscapeSlash>
3872 | <EscapeBspace>
3873 | <EscapeFormf>
3874 | <EscapeNl>
3875 | <EscapeCr>
3876 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07003877 | ~["\"","\\"])* "\"")
3878 | ("\'"(
3879 <EscapeApos>
3880 | <EscapeBslash>
3881 | <EscapeSlash>
3882 | <EscapeBspace>
3883 | <EscapeFormf>
3884 | <EscapeNl>
3885 | <EscapeCr>
3886 | <EscapeTab>
3887 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003888 | < #EscapeQuot: "\\\"" >
3889 | < #EscapeApos: "\\\'" >
3890 | < #EscapeBslash: "\\\\" >
3891 | < #EscapeSlash: "\\/" >
3892 | < #EscapeBspace: "\\b" >
3893 | < #EscapeFormf: "\\f" >
3894 | < #EscapeNl: "\\n" >
3895 | < #EscapeCr: "\\r" >
3896 | < #EscapeTab: "\\t" >
3897}
3898
3899<DEFAULT,IN_DBL_BRACE>
3900TOKEN :
3901{
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08003902 <IDENTIFIER : ( <LETTER> | <IDENTIFIER_SPECIALCHARS_START> )
3903 ( <LETTER> | <DIGIT> | <IDENTIFIER_SPECIALCHARS_START> | <IDENTIFIER_SPECIALCHARS_REST> )*>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003904}
3905
3906<DEFAULT,IN_DBL_BRACE>
3907SKIP:
3908{
3909 " "
3910 | "\t"
3911 | "\r"
3912 | "\n"
3913}
3914
3915<DEFAULT,IN_DBL_BRACE>
3916SKIP:
3917{
3918 <"//" (~["\n"])* "\n">
3919}
3920
3921<DEFAULT,IN_DBL_BRACE>
3922SKIP:
3923{
3924 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
3925}
3926
3927<DEFAULT,IN_DBL_BRACE>
3928SKIP:
3929{
Yingyi Bu93846a72016-09-13 16:30:39 -07003930 <"--" (~["\n"])* "\n">
3931}
3932
3933
3934<DEFAULT,IN_DBL_BRACE>
3935SKIP:
3936{
3937 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
3938}
3939
3940<DEFAULT,IN_DBL_BRACE>
3941SKIP:
3942{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003943 <"/*"> { pushState(); } : INSIDE_COMMENT
3944}
3945
3946<INSIDE_COMMENT>
3947SPECIAL_TOKEN:
3948{
3949 <"+"(" ")*(~["*"])*>
3950}
3951
3952<INSIDE_COMMENT>
3953SKIP:
3954{
3955 <"/*"> { pushState(); }
3956}
3957
3958<INSIDE_COMMENT>
3959SKIP:
3960{
3961 <"*/"> { popState("*/"); }
3962 | <~[]>
3963}