blob: 17c94bd5ef256a58d0cf012855445a31705b9a07 [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 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -070020 COMMON_TOKEN_ACTION = true;
21 STATIC = false;
22 TOKEN_EXTENDS = "org.apache.asterix.lang.sqlpp.parser.SqlppToken";
Yingyi Bu391f09e2015-10-29 13:49:39 -070023}
24
Yingyi Bu391f09e2015-10-29 13:49:39 -070025PARSER_BEGIN(SQLPPParser)
26
27package org.apache.asterix.lang.sqlpp.parser;
28
29// For SQL++ ParserTokenManager
Till Westmanne9b2adf2016-10-15 12:39:01 -070030import java.util.ArrayDeque;
Yingyi Bu391f09e2015-10-29 13:49:39 -070031
32import java.io.BufferedReader;
33import java.io.File;
34import java.io.FileInputStream;
35import java.io.FileNotFoundException;
36import java.io.IOException;
37import java.io.InputStreamReader;
38import java.io.Reader;
39import java.io.StringReader;
40import java.util.ArrayList;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070041import java.util.Arrays;
42import java.util.Collection;
Yingyi Budaa549c2016-06-28 22:30:52 -070043import java.util.Collections;
Yingyi Bu391f09e2015-10-29 13:49:39 -070044import java.util.HashMap;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070045import java.util.HashSet;
Yingyi Bu391f09e2015-10-29 13:49:39 -070046import java.util.Iterator;
47import java.util.LinkedHashMap;
48import java.util.List;
49import java.util.Map;
50
51import org.apache.asterix.common.annotations.AutoDataGen;
52import org.apache.asterix.common.annotations.DateBetweenYearsDataGen;
53import org.apache.asterix.common.annotations.DatetimeAddRandHoursDataGen;
54import org.apache.asterix.common.annotations.DatetimeBetweenYearsDataGen;
55import org.apache.asterix.common.annotations.FieldIntervalDataGen;
56import org.apache.asterix.common.annotations.FieldValFileDataGen;
57import org.apache.asterix.common.annotations.FieldValFileSameIndexDataGen;
58import org.apache.asterix.common.annotations.IRecordFieldDataGen;
59import org.apache.asterix.common.annotations.InsertRandIntDataGen;
60import org.apache.asterix.common.annotations.ListDataGen;
61import org.apache.asterix.common.annotations.ListValFileDataGen;
62import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
63import org.apache.asterix.common.annotations.TypeDataGen;
64import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
65import org.apache.asterix.common.config.DatasetConfig.DatasetType;
66import org.apache.asterix.common.config.DatasetConfig.IndexType;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080067import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070068import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070069import org.apache.asterix.common.exceptions.WarningCollector;
70import org.apache.asterix.common.exceptions.WarningUtil;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080071import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070072import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin54c06012019-11-13 15:54:20 -080073import org.apache.asterix.common.metadata.DataverseName;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080074import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070075import org.apache.asterix.lang.common.base.AbstractLangExpression;
76import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070077import org.apache.asterix.lang.common.base.Expression;
78import org.apache.asterix.lang.common.base.Literal;
79import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070080import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070081import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070082import org.apache.asterix.lang.common.base.Statement;
83import org.apache.asterix.lang.common.clause.GroupbyClause;
84import org.apache.asterix.lang.common.clause.LetClause;
85import org.apache.asterix.lang.common.clause.LimitClause;
86import org.apache.asterix.lang.common.clause.OrderbyClause;
87import org.apache.asterix.lang.common.clause.UpdateClause;
88import org.apache.asterix.lang.common.clause.WhereClause;
89import org.apache.asterix.lang.common.context.RootScopeFactory;
90import org.apache.asterix.lang.common.context.Scope;
91import org.apache.asterix.lang.common.expression.AbstractAccessor;
92import org.apache.asterix.lang.common.expression.CallExpr;
93import org.apache.asterix.lang.common.expression.FieldAccessor;
94import org.apache.asterix.lang.common.expression.FieldBinding;
95import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
96import org.apache.asterix.lang.common.expression.IfExpr;
97import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -070098import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -070099import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +0300100import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700101import org.apache.asterix.lang.common.expression.LiteralExpr;
102import org.apache.asterix.lang.common.expression.OperatorExpr;
103import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
104import org.apache.asterix.lang.common.expression.QuantifiedExpression;
105import org.apache.asterix.lang.common.expression.RecordConstructor;
106import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
107import org.apache.asterix.lang.common.expression.TypeExpression;
108import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
109import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700110import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
111import org.apache.asterix.lang.common.expression.VariableExpr;
112import org.apache.asterix.lang.common.literal.DoubleLiteral;
113import org.apache.asterix.lang.common.literal.FalseLiteral;
114import org.apache.asterix.lang.common.literal.FloatLiteral;
115import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700116import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700117import org.apache.asterix.lang.common.literal.NullLiteral;
118import org.apache.asterix.lang.common.literal.StringLiteral;
119import org.apache.asterix.lang.common.literal.TrueLiteral;
120import org.apache.asterix.lang.common.parser.ScopeChecker;
121import org.apache.asterix.lang.common.statement.CompactStatement;
122import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800123import org.apache.asterix.lang.common.statement.StartFeedStatement;
124import org.apache.asterix.lang.common.statement.StopFeedStatement;
Ian Maxon38fe9402020-01-29 19:27:40 -0800125import org.apache.asterix.lang.common.statement.CreateAdapterStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700126import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
127import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
128import org.apache.asterix.lang.common.statement.CreateFeedStatement;
129import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
130import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800131import org.apache.asterix.lang.common.statement.CreateSynonymStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700132import org.apache.asterix.lang.common.statement.DatasetDecl;
133import org.apache.asterix.lang.common.statement.DataverseDecl;
134import org.apache.asterix.lang.common.statement.DataverseDropStatement;
135import org.apache.asterix.lang.common.statement.DeleteStatement;
136import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700137import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700138import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
139import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300140import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700141import org.apache.asterix.lang.common.statement.FunctionDecl;
142import org.apache.asterix.lang.common.statement.FunctionDropStatement;
143import org.apache.asterix.lang.common.statement.IndexDropStatement;
144import org.apache.asterix.lang.common.statement.InsertStatement;
145import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
146import org.apache.asterix.lang.common.statement.LoadStatement;
147import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
148import org.apache.asterix.lang.common.statement.NodegroupDecl;
149import org.apache.asterix.lang.common.statement.Query;
150import org.apache.asterix.lang.common.statement.RefreshExternalDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700151import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800152import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700153import org.apache.asterix.lang.common.statement.TypeDecl;
154import org.apache.asterix.lang.common.statement.TypeDropStatement;
155import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800156import org.apache.asterix.lang.common.statement.UpsertStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700157import org.apache.asterix.lang.common.statement.WriteStatement;
158import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800159import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700160import org.apache.asterix.lang.common.struct.QuantifiedPair;
161import org.apache.asterix.lang.common.struct.VarIdentifier;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700162import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700163import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
164import org.apache.asterix.lang.sqlpp.clause.FromClause;
165import org.apache.asterix.lang.sqlpp.clause.FromTerm;
166import org.apache.asterix.lang.sqlpp.clause.HavingClause;
167import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700168import org.apache.asterix.lang.sqlpp.clause.Projection;
169import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
170import org.apache.asterix.lang.sqlpp.clause.SelectClause;
171import org.apache.asterix.lang.sqlpp.clause.SelectElement;
172import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
173import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
174import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800175import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700176import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700177import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700178import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700179import org.apache.asterix.lang.sqlpp.optype.JoinType;
180import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700181import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser;
182import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingElement;
183import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSet;
184import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSets;
185import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.RollupCube;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700186import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700187import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
188import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700189import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700190import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700191import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Ian Maxon38fe9402020-01-29 19:27:40 -0800192import org.apache.asterix.external.dataset.adapter.AdapterIdentifier;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800193import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700194import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700195import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700196import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700197import org.apache.hyracks.algebricks.common.utils.Pair;
198import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800199import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700200import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700201import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
202import org.apache.hyracks.algebricks.core.algebra.expressions.IndexedNLJoinExpressionAnnotation;
203import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700204import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700205import org.apache.hyracks.api.exceptions.Warning;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800206import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700207import org.apache.hyracks.util.StringUtil;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700208
Yingyi Bucaea8f02015-11-16 15:12:15 -0800209class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700210
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800211 // tokens parsed as identifiers
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700212 private static final String CUBE = "CUBE";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800213 private static final String CURRENT = "CURRENT";
214 private static final String EXCLUDE = "EXCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700215 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800216 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700217 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800218 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700219 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700220 private static final String LAST = "LAST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800221 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700222 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800223 private static final String OTHERS = "OTHERS";
224 private static final String PARTITION = "PARTITION";
225 private static final String PRECEDING = "PRECEDING";
226 private static final String RANGE = "RANGE";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700227 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700228 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800229 private static final String ROW = "ROW";
230 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700231 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800232 private static final String TIES = "TIES";
233 private static final String UNBOUNDED = "UNBOUNDED";
Ian Maxon38fe9402020-01-29 19:27:40 -0800234 private static final String ACTION = "ACTION";
235 private static final String LANGUAGE = "LANGUAGE";
236 private static final String CALL = "CALL";
237 private static final String DETERMINISTIC = "DETERMINISTIC";
238 private static final String RETURNS = "RETURNS";
239 private static final String INLINE = "INLINE";
240
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800241
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700242 private static final String INT_TYPE_NAME = "int";
243
Till Westmann7199a562016-09-17 16:07:32 -0700244 // error configuration
245 protected static final boolean REPORT_EXPECTED_TOKENS = false;
246
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700247 private int externalVarCounter;
248
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800249 private DataverseName defaultDataverse;
250
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700251 private SqlppGroupingSetsParser groupingSetsParser;
252
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700253 private final WarningCollector warningCollector = new WarningCollector();
254
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700255 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
256
Yingyi Bu391f09e2015-10-29 13:49:39 -0700257 private static class IndexParams {
258 public IndexType type;
259 public int gramLength;
260
261 public IndexParams(IndexType type, int gramLength) {
262 this.type = type;
263 this.gramLength = gramLength;
264 }
265 };
266
267 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800268 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700269 public String library;
270 public String function;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700271 public SqlppHint hint;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700272 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700273 }
274
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700275 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
276 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
277 switch (hintToken.hint) {
278 case VAL_FILE_HINT:
279 File[] valFiles = new File[splits.length];
280 for (int k=0; k<splits.length; k++) {
281 valFiles[k] = new File(splits[k]);
282 }
283 return new FieldValFileDataGen(valFiles);
284 case VAL_FILE_SAME_INDEX_HINT:
285 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
286 case LIST_VAL_FILE_HINT:
287 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
288 case LIST_HINT:
289 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
290 case INTERVAL_HINT:
291 FieldIntervalDataGen.ValueType vt;
292 switch (splits[0]) {
293 case "int":
294 vt = FieldIntervalDataGen.ValueType.INT;
295 break;
296 case "long":
297 vt = FieldIntervalDataGen.ValueType.LONG;
298 break;
299 case "float":
300 vt = FieldIntervalDataGen.ValueType.FLOAT;
301 break;
302 case "double":
303 vt = FieldIntervalDataGen.ValueType.DOUBLE;
304 break;
305 default:
306 throw new SqlppParseException(getSourceLocation(hintToken),
307 "Unknown type for interval data gen: " + splits[0]);
308 }
309 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
310 case INSERT_RAND_INT_HINT:
311 return new InsertRandIntDataGen(splits[0], splits[1]);
312 case DATE_BETWEEN_YEARS_HINT:
313 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
314 case DATETIME_BETWEEN_YEARS_HINT:
315 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
316 case DATETIME_ADD_RAND_HOURS_HINT:
317 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
318 case AUTO_HINT:
319 return new AutoDataGen(splits[0]);
320 default:
321 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700322 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700323 }
324
Till Westmann7199a562016-09-17 16:07:32 -0700325 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700326 this(new StringReader(s));
327 super.setInput(s);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700328 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700329 }
330
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800331 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700332 File file = new File(args[0]);
333 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
334 SQLPPParser parser = new SQLPPParser(fis);
335 List<Statement> st = parser.parse();
336 //st.accept(new SQLPPPrintVisitor(), 0);
337 }
338
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800339 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800340 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700341 return parseImpl(new ParseFunction<List<Statement>>() {
342 @Override
343 public List<Statement> parse() throws ParseException {
344 return SQLPPParser.this.Statement();
345 }
346 });
347 }
348
349 private Expression parseExpression() throws CompilationException {
350 return parseImpl(new ParseFunction<Expression>() {
351 @Override
352 public Expression parse() throws ParseException {
353 return SQLPPParser.this.Expression();
354 }
355 });
356 }
357
358 private static Expression parseExpression(String text) throws CompilationException {
359 return new SQLPPParser(text).parseExpression();
360 }
361
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800362 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800363 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800364 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800365 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800366 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800367 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800368 DataverseName dataverse = defaultDataverse;
369 defaultDataverse = signature.getDataverseName();
370 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800371 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
372 for (String paramName : paramNames) {
373 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
374 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800375 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
376 removeCurrentScope();
377 defaultDataverse = dataverse;
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800378 return new FunctionDecl(signature, paramVars, functionBodyExpr);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800379 }
380 });
381 }
382
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700383 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700384 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700385 hintCollector.clear();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700386 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700387 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700388 } catch (SqlppParseException e) {
389 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(), LogRedactionUtil.userData(getMessage(e)));
390 } catch (ParseException e) {
391 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700392 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700393 // 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 -0700394 // 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 -0700395 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700396 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700397 } finally {
398 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700399 }
400 }
Till Westmann7199a562016-09-17 16:07:32 -0700401
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700402 @FunctionalInterface
403 private interface ParseFunction<T> {
404 T parse() throws ParseException;
405 }
406
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700407 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700408 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
409 warningCollector.getWarnings(outWarnings, maxWarnings);
410 }
411
412 @Override
413 public long getTotalWarningsCount() {
414 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700415 }
416
Till Westmann7199a562016-09-17 16:07:32 -0700417 protected String getMessage(ParseException pe) {
418 Token currentToken = pe.currentToken;
419 if (currentToken == null) {
420 return pe.getMessage();
421 }
422 int[][] expectedTokenSequences = pe.expectedTokenSequences;
423 String[] tokenImage = pe.tokenImage;
424 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
425 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
426 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
427 Token tok = currentToken.next;
428 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700429 StringBuilder message = new StringBuilder(128);
430 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700431 for (int i = 0; i < maxSize; i++) {
432 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700433 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700434 }
435 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700436 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700437 break;
438 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700439 final String fixedTokenImage = tokenImage[tok.kind];
440 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700441 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700442 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700443 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700444 tok = tok.next;
445 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700446 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700447 if (REPORT_EXPECTED_TOKENS) {
448 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700449 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700450 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700451 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700452 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700453 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700454 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700455 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700456 }
457
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700458 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700459 return
460 token == null ? null :
461 token.sourceLocation != null ? token.sourceLocation :
462 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700463 }
464
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700465 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
466 expr.setSourceLocation(getSourceLocation(token));
467 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700468 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800469
470 private boolean isToken(String image) {
471 return token.image.equalsIgnoreCase(image);
472 }
473
474 private void expectToken(String image) throws SqlppParseException {
475 if (!isToken(image)) {
476 throw createUnexpectedTokenError();
477 }
478 }
479
480 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700481 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700482 }
483
484 private SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700485 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800486 }
487
488 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700489 return createUnexpectedTokenError(token, expected);
490 }
491
492 private SqlppParseException createUnexpectedTokenError(Token t, String expected) {
493 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800494 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700495 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800496 }
497
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700498 private boolean laToken(int idx, int kind) {
499 Token t = getToken(idx);
500 return t.kind == kind;
501 }
502
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800503 private boolean laToken(int idx, int kind, String image) {
504 Token t = getToken(idx);
505 return t.kind == kind && t.image.equalsIgnoreCase(image);
506 }
507
508 private boolean laIdentifier(int idx, String image) {
509 return laToken(idx, IDENTIFIER, image);
510 }
511
512 private boolean laIdentifier(String image) {
513 return laIdentifier(1, image);
514 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700515
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700516 private Token fetchHint(Token token, SqlppHint... expectedHints) {
517 Token hintToken = token.specialToken;
518 if (hintToken == null) {
519 return null;
520 }
521 SourceLocation sourceLoc = getSourceLocation(hintToken);
522 hintCollector.remove(sourceLoc);
523 if (hintToken.hint == null) {
524 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
525 return null;
526 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
527 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
528 return null;
529 } else {
530 return hintToken;
531 }
532 }
533
534 private void reportUnclaimedHints() {
535 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
536 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
537 }
538 }
539
540 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
541 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
542 }
543
544 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700545 if (warningCollector.shouldWarn()) {
546 warningCollector.warn(WarningUtil.forAsterix(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
547 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700548 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700549}
550
551PARSER_END(SQLPPParser)
552
Yingyi Bu391f09e2015-10-29 13:49:39 -0700553List<Statement> Statement() throws ParseException:
554{
555 scopeStack.push(RootScopeFactory.createRootScope(this));
556 List<Statement> decls = new ArrayList<Statement>();
557 Statement stmt = null;
558}
559{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300560 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700561 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300562 {
563 decls.add(stmt);
564 }
565 )?
566 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700567 )*
568 <EOF>
569 {
570 return decls;
571 }
572}
573
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700574Statement ExplainStatement() throws ParseException:
575{
576 Statement stmt = null;
577 Token explainToken = null;
578}
579{
580 ( <EXPLAIN> { explainToken = token; } )?
581 stmt = SingleStatement()
582 {
583 if (explainToken != null) {
584 if (stmt.getKind() == Statement.Kind.QUERY) {
585 ((Query)stmt).setExplain(true);
586 } else {
587 throw new SqlppParseException(getSourceLocation(explainToken),
588 "EXPLAIN is not supported for this kind of statement");
589 }
590 }
591 return stmt;
592 }
593}
594
Yingyi Bu391f09e2015-10-29 13:49:39 -0700595Statement SingleStatement() throws ParseException:
596{
597 Statement stmt = null;
598}
599{
600 (
601 stmt = DataverseDeclaration()
602 | stmt = FunctionDeclaration()
603 | stmt = CreateStatement()
604 | stmt = LoadStatement()
605 | stmt = DropStatement()
606 | stmt = WriteStatement()
607 | stmt = SetStatement()
608 | stmt = InsertStatement()
609 | stmt = DeleteStatement()
610 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800611 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700612 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700613 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700614 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700615 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700616 )
617 {
618 return stmt;
619 }
620}
621
622DataverseDecl DataverseDeclaration() throws ParseException:
623{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700624 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800625 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700626}
627{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800628 <USE> { startToken = token; } dvName = MultipartIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700629 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800630 defaultDataverse = DataverseName.create(dvName);
631 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700632 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700633 }
634}
635
636Statement CreateStatement() throws ParseException:
637{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700638 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700639 Statement stmt = null;
640}
641{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700642 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700643 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800644 stmt = CreateTypeStatement(startToken)
645 | stmt = CreateNodegroupStatement(startToken)
646 | stmt = CreateDatasetStatement(startToken)
647 | stmt = CreateIndexStatement(startToken)
648 | stmt = CreateDataverseStatement(startToken)
649 | stmt = CreateFunctionStatement(startToken)
Ian Maxon38fe9402020-01-29 19:27:40 -0800650 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800651 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800652 | stmt = CreateFeedStatement(startToken)
653 | stmt = CreateFeedPolicyStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700654 )
655 {
656 return stmt;
657 }
658}
659
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800660TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
661{
662 TypeDecl stmt = null;
663}
664{
665 <TYPE> stmt = TypeSpecification(startStmtToken)
666 {
667 return stmt;
668 }
669}
670
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700671TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700672{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800673 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700674 boolean ifNotExists = false;
675 TypeExpression typeExpr = null;
676}
677{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800678 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700679 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800680 {
681 boolean dgen = false;
682 long numValues = -1;
683 String filename = null;
684 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
685 if (hintToken != null) {
686 String hintParams = hintToken.hintParams;
687 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
688 if (splits == null || splits.length != 2) {
689 throw new SqlppParseException(getSourceLocation(hintToken),
690 "Expecting /*+ dgen <filename> <numberOfItems> */");
691 }
692 dgen = true;
693 filename = splits[0];
694 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700695 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800696 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
697 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
698 return addSourceLocation(stmt, startStmtToken);
699 }
700}
701
702NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
703{
704 NodegroupDecl stmt = null;
705}
706{
707 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
708 {
709 return stmt;
710 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700711}
712
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700713NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700714{
715 String name = null;
716 String tmp = null;
717 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800718 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700719}
720{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800721 name = Identifier() ifNotExists = IfNotExists()
722 <ON> tmp = Identifier()
723 {
724 ncNames.add(new Identifier(tmp));
725 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700726 ( <COMMA> tmp = Identifier()
727 {
728 ncNames.add(new Identifier(tmp));
729 }
730 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800731 {
732 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
733 return addSourceLocation(stmt, startStmtToken);
734 }
735}
736
737void Dataset() throws ParseException:
738{
739}
740{
741 (<DATASET>|<COLLECTION>)
742}
743
744DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
745{
746 DatasetDecl stmt = null;
747}
748{
749 (
750 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
751 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
752 )
753 {
754 return stmt;
755 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700756}
757
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700758DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700759{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800760 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700761 boolean ifNotExists = false;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700762 TypeExpression typeExpr = null;
763 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800764 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700765 String nodeGroupName = null;
766 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700767 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700768 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800769 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -0800770 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700771}
772{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800773 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700774 typeExpr = DatasetTypeSpecification()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700775 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800776 { String name; }
777 <WITH>
778 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700779 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800780 if (!name.equalsIgnoreCase("meta")){
781 throw new SqlppParseException(getSourceLocation(startStmtToken),
782 "We can only support one additional associated field called \"meta\".");
783 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700784 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700785 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800786 )?
787 ifNotExists = IfNotExists()
788 primaryKeyFields = PrimaryKey()
789 (<AUTOGENERATED> { autogenerated = true; } )?
790 (<ON> nodeGroupName = Identifier() )?
791 ( <HINTS> hints = Properties() )?
792 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
793 ( <WITH> withRecord = RecordConstructor() )?
794 {
795 if(filterField!=null && filterField.first!=0){
796 throw new SqlppParseException(getSourceLocation(startStmtToken),
797 "A filter field can only be a field in the main record of the dataset.");
798 }
799 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
800 filterField == null? null : filterField.second);
801 try {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700802 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr,
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800803 nodeGroupName != null ? new Identifier(nodeGroupName) : null, hints, DatasetType.INTERNAL, idd, withRecord,
804 ifNotExists);
805 return addSourceLocation(stmt, startStmtToken);
806 } catch (CompilationException e) {
807 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
808 }
809 }
810}
811
812DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
813{
814 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700815 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800816 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800817 String adapterName = null;
818 Map<String,String> properties = null;
819 String nodeGroupName = null;
820 Map<String,String> hints = new HashMap<String,String>();
821 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800822 RecordConstructor withRecord = null;
823}
824{
825 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700826 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800827 ifNotExists = IfNotExists()
828 <USING> adapterName = AdapterName() properties = Configuration()
829 ( <ON> nodeGroupName = Identifier() )?
830 ( <HINTS> hints = Properties() )?
831 ( <WITH> withRecord = RecordConstructor() )?
832 {
833 ExternalDetailsDecl edd = new ExternalDetailsDecl();
834 edd.setAdapter(adapterName);
835 edd.setProperties(properties);
836 try {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700837 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null,
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800838 nodeGroupName != null? new Identifier(nodeGroupName): null, hints, DatasetType.EXTERNAL, edd, withRecord,
839 ifNotExists);
840 return addSourceLocation(stmt, startStmtToken);
841 } catch (CompilationException e) {
842 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
843 }
844 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700845}
846
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700847TypeExpression DatasetTypeSpecification() throws ParseException:
848{
849 TypeExpression typeExpr = null;
850}
851{
852 (
853 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
854 | typeExpr = DatasetReferenceTypeSpecification()
855 )
856 {
857 return typeExpr;
858 }
859}
860
861TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
862{
863 TypeExpression typeExpr = null;
864}
865{
866 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
867 {
868 return typeExpr;
869 }
870}
871
872TypeExpression DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
873{
874 RecordTypeDefinition recordTypeDef = null;
875 RecordTypeDefinition.RecordKind recordKind = null;
876 Token recordKindToken = null;
877}
878{
879 <LEFTPAREN> recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
880 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
881 {
882 if (recordKind == null) {
883 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
884 } else if (!allowRecordKindModifier) {
885 throw createUnexpectedTokenError(recordKindToken);
886 }
887 recordTypeDef.setRecordKind(recordKind);
888 return recordTypeDef;
889 }
890}
891
892RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
893{
894 RecordTypeDefinition recType = new RecordTypeDefinition();
895}
896{
897 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
898 {
899 return recType;
900 }
901}
902
903void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
904{
905 String fieldName;
906 TypeExpression type = null;
907 boolean isUnknownable = true;
908}
909{
910 fieldName = Identifier()
911 type = TypeReference() ( <NOT> <NULL> { isUnknownable = false; } )?
912 {
913 recType.addField(fieldName, type, isUnknownable);
914 }
915}
916
Yingyi Bu391f09e2015-10-29 13:49:39 -0700917RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
918{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700919 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800920 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700921 String datasetName = null;
922}
923{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800924 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
925 {
926 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
927 stmt.setDataverseName(nameComponents.first);
928 stmt.setDatasetName(nameComponents.second);
929 return addSourceLocation(stmt, startToken);
930 }
931}
932
933CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
934{
935 CreateIndexStatement stmt = null;
936}
937{
938 (
939 <INDEX> stmt = IndexSpecification(startStmtToken)
940 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
941 )
942 {
943 return stmt;
944 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700945}
946
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700947CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700948{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700949 CreateIndexStatement stmt = new CreateIndexStatement();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700950 String indexName = null;
951 boolean ifNotExists = false;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800952 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700953 Pair<Integer, Pair<List<String>, IndexedTypeExpression>> fieldPair = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700954 IndexParams indexType = null;
955 boolean enforced = false;
956}
957{
Ali Alsuliman8351d252017-09-24 00:43:15 -0700958 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800959 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -0700960 <ON> nameComponents = QualifiedName()
961 <LEFTPAREN> ( fieldPair = OpenField()
962 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700963 stmt.addFieldExprPair(fieldPair.second);
964 stmt.addFieldIndexIndicator(fieldPair.first);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700965 }
Ali Alsuliman8351d252017-09-24 00:43:15 -0700966 ) (<COMMA> fieldPair = OpenField()
967 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700968 stmt.addFieldExprPair(fieldPair.second);
969 stmt.addFieldIndexIndicator(fieldPair.first);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700970 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800971 )* <RIGHTPAREN> ( <TYPE> indexType = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman8351d252017-09-24 00:43:15 -0700972 )
973 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700974 stmt.setIndexName(new Identifier(indexName));
975 stmt.setIfNotExists(ifNotExists);
976 stmt.setDataverseName(nameComponents.first);
977 stmt.setDatasetName(nameComponents.second);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700978 if (indexType != null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700979 stmt.setIndexType(indexType.type);
980 stmt.setGramLength(indexType.gramLength);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700981 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700982 stmt.setEnforced(enforced);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700983 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700984 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700985}
986
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800987CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
988{
989 CreateIndexStatement stmt = new CreateIndexStatement();
990 String indexName = null;
991 boolean ifNotExists = false;
992 Pair<DataverseName,Identifier> nameComponents = null;
993}
994{
995 (indexName = Identifier())? ifNotExists = IfNotExists()
996 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
997 {
998 if (indexName == null) {
999 indexName = "primary_idx_" + nameComponents.second;
1000 }
1001 stmt.setIndexName(new Identifier(indexName));
1002 stmt.setIfNotExists(ifNotExists);
1003 stmt.setDataverseName(nameComponents.first);
1004 stmt.setDatasetName(nameComponents.second);
1005 return addSourceLocation(stmt, startStmtToken);
1006 }
1007}
1008
Yingyi Bu391f09e2015-10-29 13:49:39 -07001009String CompactionPolicy() throws ParseException :
1010{
1011 String compactionPolicy = null;
1012}
1013{
1014 compactionPolicy = Identifier()
1015 {
1016 return compactionPolicy;
1017 }
1018}
1019
1020String FilterField() throws ParseException :
1021{
1022 String filterField = null;
1023}
1024{
1025 filterField = Identifier()
1026 {
1027 return filterField;
1028 }
1029}
1030
1031IndexParams IndexType() throws ParseException:
1032{
1033 IndexType type = null;
1034 int gramLength = 0;
1035}
1036{
1037 (<BTREE>
1038 {
1039 type = IndexType.BTREE;
1040 }
1041 | <RTREE>
1042 {
1043 type = IndexType.RTREE;
1044 }
1045 | <KEYWORD>
1046 {
1047 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1048 }
Taewoo Kimc49405a2017-01-04 00:30:43 -08001049 |<FULLTEXT>
1050 {
1051 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1052 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001053 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1054 {
1055 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1056 gramLength = Integer.valueOf(token.image);
1057 }
1058 <RIGHTPAREN>)
1059 {
1060 return new IndexParams(type, gramLength);
1061 }
1062}
1063
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001064CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1065{
1066 CreateDataverseStatement stmt = null;
1067}
1068{
1069 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1070 {
1071 return stmt;
1072 }
1073}
1074
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001075CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001076{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001077 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001078 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001079}
1080{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001081 dvName = MultipartIdentifier() ifNotExists = IfNotExists()
1082 {
1083 CreateDataverseStatement stmt = new CreateDataverseStatement(DataverseName.create(dvName), null, ifNotExists);
1084 return addSourceLocation(stmt, startStmtToken);
1085 }
1086}
1087
1088CreateFunctionStatement CreateFunctionStatement(Token startStmtToken) throws ParseException:
1089{
1090 CreateFunctionStatement stmt = null;
1091}
1092{
1093 <FUNCTION> stmt = FunctionSpecification(startStmtToken)
1094 {
1095 return stmt;
1096 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001097}
1098
Ian Maxon38fe9402020-01-29 19:27:40 -08001099
1100CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1101{
1102 CreateAdapterStatement stmt = null;
1103}
1104{
1105 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1106 {
1107 return stmt;
1108 }
1109}
1110
1111CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1112{
1113 AdapterIdentifier signature = null;
1114 Token beginPos;
1115 Token endPos;
1116 Pair<DataverseName,Identifier> adaptName = null;
1117 DataverseName currentDataverse = defaultDataverse;
1118 String lang = null;
1119 String libName ="";
1120 String externalIdent = "";
1121 ListConstructor resources = null;
1122 boolean ifNotExists = false;
1123}
1124{
1125 adaptName = QualifiedName()
1126 <IDENTIFIER> {expectToken(LANGUAGE);} lang = Identifier() <AS> libName = ConstantString() <COMMA> externalIdent = ConstantString() ifNotExists = IfNotExists()
1127 {
1128 signature = new AdapterIdentifier(adaptName.getFirst(),adaptName.getSecond().getValue());
1129 CreateAdapterStatement stmt =
1130 new CreateAdapterStatement(signature, lang, libName, externalIdent, ifNotExists);
1131 return addSourceLocation(stmt, startStmtToken);
1132 }
1133
1134}
1135
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001136CreateFunctionStatement FunctionSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001137{
Ian Maxon38fe9402020-01-29 19:27:40 -08001138 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001139 boolean ifNotExists = false;
Ian Maxon38fe9402020-01-29 19:27:40 -08001140 String functionBody = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001141 VarIdentifier var = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001142 Expression functionBodyExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001143 Token beginPos;
1144 Token endPos;
1145 FunctionName fctName = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001146 DataverseName currentDataverse = defaultDataverse;
Ian Maxon38fe9402020-01-29 19:27:40 -08001147 IndexedTypeExpression returnType = null;
1148 boolean deterministic = false;
1149 boolean nullCall = false;
1150 String lang = null;
1151 String libName ="";
1152 String externalIdent = "";
Dmitry Lychagin74fac082020-03-05 11:28:18 -08001153 List<String> externalIdentList = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001154 List<Pair<VarIdentifier,IndexedTypeExpression>> args = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001155 RecordConstructor resources = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001156}
1157{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001158 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001159 {
1160 defaultDataverse = fctName.dataverse;
1161 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001162 ifNotExists = IfNotExists()
Ian Maxon38fe9402020-01-29 19:27:40 -08001163 args = FunctionParameters()
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001164 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001165 (
1166 (
1167 <LEFTBRACE>
1168 {
1169 createNewScope();
1170 beginPos = token;
1171 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001172 functionBodyExpr = FunctionBody()
Ian Maxon38fe9402020-01-29 19:27:40 -08001173 <RIGHTBRACE>{
1174 endPos = token;
1175 functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine, endPos.beginColumn);
1176 signature = new FunctionSignature(fctName.dataverse, fctName.function, args.size());
1177 getCurrentScope().addFunctionDescriptor(signature, false);
1178 removeCurrentScope();
1179 defaultDataverse = currentDataverse;
1180 CreateFunctionStatement stmt = new CreateFunctionStatement(signature, args, returnType, functionBody, functionBodyExpr, ifNotExists);
1181 return addSourceLocation(stmt, startStmtToken);
1182 }
1183 )
1184 |
1185 <IDENTIFIER> {expectToken(LANGUAGE);}
1186 (
1187 LOOKAHEAD({laIdentifier(INLINE)})
1188 (
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001189 <IDENTIFIER> <AS>
1190 {
Ian Maxon38fe9402020-01-29 19:27:40 -08001191 createNewScope();
1192 beginPos = token;
1193 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001194 functionBodyExpr = FunctionBody()
Ian Maxon38fe9402020-01-29 19:27:40 -08001195 {
1196 endPos = token;
1197 functionBody = extractFragment(beginPos.endLine, beginPos.beginColumn+1, endPos.endLine, endPos.endColumn+1);
1198 signature = new FunctionSignature(fctName.dataverse, fctName.function, args.size());
1199 getCurrentScope().addFunctionDescriptor(signature, false);
1200 removeCurrentScope();
1201 defaultDataverse = currentDataverse;
1202 CreateFunctionStatement stmt = new CreateFunctionStatement(signature, args, returnType, functionBody, functionBodyExpr, ifNotExists);
1203 return addSourceLocation(stmt, startStmtToken);
1204 }
1205 )
1206 |
1207 (
1208 lang = Identifier()
1209 ( (<NOT> <IDENTIFIER> {expectToken(DETERMINISTIC); deterministic = false;}) | (<IDENTIFIER> {expectToken(DETERMINISTIC); deterministic = true;}) )?
1210 (<NULL> <IDENTIFIER> { expectToken(CALL); nullCall = true;})?
Dmitry Lychagin74fac082020-03-05 11:28:18 -08001211 <AS> libName = ConstantString()
1212 { externalIdentList = new ArrayList<String>(2); }
1213 ( <COMMA> externalIdent = ConstantString() { externalIdentList.add(externalIdent); } )+
Ian Maxon38fe9402020-01-29 19:27:40 -08001214 (<WITH> resources = RecordConstructor())?
1215 {
1216 signature = new FunctionSignature(fctName.dataverse, fctName.function, args.size());
1217 defaultDataverse = currentDataverse;
1218 CreateFunctionStatement stmt = null;
1219 try{
1220 stmt =
1221 new CreateFunctionStatement(signature, args, returnType, deterministic, nullCall,
Dmitry Lychagin74fac082020-03-05 11:28:18 -08001222 lang, libName, externalIdentList, resources, ifNotExists);
Ian Maxon38fe9402020-01-29 19:27:40 -08001223 } catch (AlgebricksException e) {
1224 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1225 }
1226 return addSourceLocation(stmt, startStmtToken);
1227 }
1228 )
1229 )
1230 )
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001231}
1232
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001233List<Pair<VarIdentifier,IndexedTypeExpression>> FunctionParameters() :
1234{
1235 List<Pair<VarIdentifier,IndexedTypeExpression>> paramList = Collections.<Pair<VarIdentifier,IndexedTypeExpression>>emptyList();
1236}
1237{
1238 <LEFTPAREN> (paramList = FunctionParameterList())? <RIGHTPAREN>
1239 {
1240 return paramList;
1241 }
1242}
1243
1244List<Pair<VarIdentifier,IndexedTypeExpression>> FunctionParameterList() :
1245{
1246 List<Pair<VarIdentifier,IndexedTypeExpression>> paramList = new ArrayList<Pair<VarIdentifier,IndexedTypeExpression>>();
1247 Pair<VarIdentifier,IndexedTypeExpression> varPair = new Pair<VarIdentifier,IndexedTypeExpression>(null,null);
1248 String var = null;
1249 IndexedTypeExpression type = null;
1250}
1251{
1252 var = VariableIdentifier() {
1253 varPair.setFirst(new VarIdentifier(var));
1254 paramList.add(varPair);
1255 } (LOOKAHEAD(3)(<COLON>)? type = IndexedTypeExpr() {varPair.setSecond(type);} )? (<COMMA> var = VariableIdentifier() {
1256 varPair = new Pair<VarIdentifier,IndexedTypeExpression>(null,null);
1257 varPair.setFirst(new VarIdentifier(var));
1258 paramList.add(varPair);
1259 } (LOOKAHEAD(3)(<COLON>)? type = IndexedTypeExpr() {varPair.setSecond(type);})?)*
1260 {
1261 return paramList;
1262 }
1263}
1264
1265IndexedTypeExpression FunctionReturnType() throws ParseException:
1266{
1267 IndexedTypeExpression returnType = null;
1268}
1269{
1270 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = IndexedTypeExpr() )?
1271 {
1272 return returnType;
1273 }
1274}
1275
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001276Expression FunctionBody() throws ParseException:
1277{
1278 Expression functionBodyExpr = null;
1279}
1280{
1281 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1282 {
1283 return functionBodyExpr;
1284 }
1285}
1286
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001287CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1288{
1289 CreateFeedStatement stmt = null;
1290}
1291{
1292 <FEED> stmt = FeedSpecification(startStmtToken)
1293 {
1294 return stmt;
1295 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001296}
1297
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001298CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001299{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001300 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001301 boolean ifNotExists = false;
1302 String adapterName = null;
1303 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001304 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001305 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001306 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001307}
1308{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001309 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001310 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001311 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001312 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001313 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001314 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001315 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001316 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001317 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001318 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001319}
1320
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001321CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1322{
1323 CreateFeedPolicyStatement stmt = null;
1324}
1325{
1326 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1327 {
1328 return stmt;
1329 }
1330}
1331
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001332CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001333{
Michael Blowd6cf6412016-06-30 02:44:35 -04001334 String policyName = null;
1335 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001336 String sourcePolicyFile = null;
1337 String definition = null;
1338 boolean ifNotExists = false;
1339 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001340 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001341}
1342{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001343 policyName = Identifier() ifNotExists = IfNotExists()
1344 <FROM>
1345 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001346 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001347 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001348 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001349 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1350 {
1351 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1352 }
1353 )
1354 {
1355 return addSourceLocation(stmt, startStmtToken);
1356 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001357}
1358
Yingyi Bu391f09e2015-10-29 13:49:39 -07001359
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001360CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
1361{
1362 CreateSynonymStatement stmt = null;
1363}
1364{
1365 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
1366 {
1367 return stmt;
1368 }
1369}
1370
1371CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
1372{
1373 Pair<DataverseName,Identifier> nameComponents = null;
1374 Pair<DataverseName,Identifier> objectNameComponents = null;
1375 boolean ifNotExists = false;
1376}
1377{
1378 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1379 <FOR> objectNameComponents = QualifiedName()
1380 {
1381 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
1382 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
1383 return addSourceLocation(stmt, startStmtToken);
1384 }
1385}
1386
Yingyi Bu391f09e2015-10-29 13:49:39 -07001387boolean IfNotExists() throws ParseException:
1388{
1389}
1390{
Yingyi Budaa549c2016-06-28 22:30:52 -07001391 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001392 {
1393 return true;
1394 }
1395 )?
1396 {
1397 return false;
1398 }
1399}
1400
Xikui Wang9d63f622017-05-18 17:50:44 -07001401void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001402{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001403 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07001404 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001405}
1406{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001407 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001408 {
1409 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1410 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
1411 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07001412 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001413 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07001414 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001415 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1416 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07001417 }
1418 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07001419}
1420
1421String GetPolicy() throws ParseException:
1422{
1423 String policy = null;
1424}
1425{
1426 <USING> <POLICY> policy = Identifier()
1427 {
1428 return policy;
1429 }
1430
1431}
1432
1433FunctionSignature FunctionSignature() throws ParseException:
1434{
1435 FunctionName fctName = null;
1436 int arity = 0;
1437}
1438{
1439 fctName = FunctionName() <ATT> <INTEGER_LITERAL>
1440 {
1441 arity = new Integer(token.image);
1442 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001443 throw new SqlppParseException(getSourceLocation(token), "Invalid arity:" + arity);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001444 }
1445
1446 // TODO use fctName.library
1447 String fqFunctionName = fctName.library == null ? fctName.function : fctName.library + "#" + fctName.function;
1448 return new FunctionSignature(fctName.dataverse, fqFunctionName, arity);
1449 }
1450}
1451
Yingyi Buc9bfe252016-03-01 00:02:40 -08001452Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001453{
Yingyi Buc9bfe252016-03-01 00:02:40 -08001454 Pair<Integer, List<String>> tmp = null;
1455 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001456 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
1457}
1458{
1459 <PRIMARY> <KEY> tmp = NestedField()
1460 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001461 keyFieldSourceIndicators.add(tmp.first);
1462 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001463 }
1464 ( <COMMA> tmp = NestedField()
1465 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001466 keyFieldSourceIndicators.add(tmp.first);
1467 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001468 }
1469 )*
1470 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001471 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001472 }
1473}
1474
1475Statement DropStatement() throws ParseException:
1476{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001477 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001478 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001479}
1480{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001481 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001482 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001483 stmt = DropDatasetStatement(startToken)
1484 | stmt = DropIndexStatement(startToken)
1485 | stmt = DropNodeGroupStatement(startToken)
1486 | stmt = DropTypeStatement(startToken)
1487 | stmt = DropDataverseStatement(startToken)
1488 | stmt = DropFunctionStatement(startToken)
1489 | stmt = DropFeedStatement(startToken)
1490 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001491 | stmt = DropSynonymStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07001492 )
1493 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001494 return stmt;
1495 }
1496}
1497
1498DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
1499{
1500 DropDatasetStatement stmt = null;
1501}
1502{
1503 Dataset() stmt = DropDatasetSpecification(startStmtToken)
1504 {
1505 return stmt;
1506 }
1507}
1508
1509DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
1510{
1511 Pair<DataverseName,Identifier> pairId = null;
1512 boolean ifExists = false;
1513}
1514{
1515 pairId = QualifiedName() ifExists = IfExists()
1516 {
1517 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
1518 return addSourceLocation(stmt, startStmtToken);
1519 }
1520}
1521
1522IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
1523{
1524 IndexDropStatement stmt = null;
1525}
1526{
1527 <INDEX> stmt = DropIndexSpecification(startStmtToken)
1528 {
1529 return stmt;
1530 }
1531}
1532
1533IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
1534{
1535 Triple<DataverseName,Identifier,Identifier> tripleId = null;
1536 boolean ifExists = false;
1537}
1538{
1539 tripleId = DoubleQualifiedName() ifExists = IfExists()
1540 {
1541 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
1542 return addSourceLocation(stmt, startStmtToken);
1543 }
1544}
1545
1546NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
1547{
1548 NodeGroupDropStatement stmt = null;
1549}
1550{
1551 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
1552 {
1553 return stmt;
1554 }
1555}
1556
1557NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
1558{
1559 String id = null;
1560 boolean ifExists = false;
1561}
1562{
1563 id = Identifier() ifExists = IfExists()
1564 {
1565 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
1566 return addSourceLocation(stmt, startStmtToken);
1567 }
1568}
1569
1570TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
1571{
1572 TypeDropStatement stmt = null;
1573}
1574{
1575 <TYPE> stmt = DropTypeSpecification(startStmtToken)
1576 {
1577 return stmt;
1578 }
1579}
1580
1581TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
1582{
1583 Pair<DataverseName,Identifier> pairId = null;
1584 boolean ifExists = false;
1585}
1586{
1587 pairId = TypeName() ifExists = IfExists()
1588 {
1589 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
1590 return addSourceLocation(stmt, startStmtToken);
1591 }
1592}
1593
1594DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
1595{
1596 DataverseDropStatement stmt = null;
1597}
1598{
1599 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
1600 {
1601 return stmt;
1602 }
1603}
1604
1605DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
1606{
1607 List<String> multipartId = null;
1608 boolean ifExists = false;
1609}
1610{
1611 multipartId = MultipartIdentifier() ifExists = IfExists()
1612 {
1613 DataverseDropStatement stmt = new DataverseDropStatement(DataverseName.create(multipartId), ifExists);
1614 return addSourceLocation(stmt, startStmtToken);
1615 }
1616}
1617
1618FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
1619{
1620 FunctionDropStatement stmt = null;
1621}
1622{
1623 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
1624 {
1625 return stmt;
1626 }
1627}
1628
1629FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
1630{
1631 FunctionSignature funcSig = null;
1632 boolean ifExists = false;
1633}
1634{
1635 funcSig = FunctionSignature() ifExists = IfExists()
1636 {
1637 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
1638 return addSourceLocation(stmt, startStmtToken);
1639 }
1640}
1641
1642FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
1643{
1644 FeedDropStatement stmt = null;
1645}
1646{
1647 <FEED> stmt = DropFeedSpecification(startStmtToken)
1648 {
1649 return stmt;
1650 }
1651}
1652
1653FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
1654{
1655 Pair<DataverseName,Identifier> pairId = null;
1656 boolean ifExists = false;
1657}
1658{
1659 pairId = QualifiedName() ifExists = IfExists()
1660 {
1661 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
1662 return addSourceLocation(stmt, startStmtToken);
1663 }
1664}
1665
1666FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
1667{
1668 FeedPolicyDropStatement stmt = null;
1669}
1670{
1671 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
1672 {
1673 return stmt;
1674 }
1675}
1676
1677FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
1678{
1679 Pair<DataverseName,Identifier> pairId = null;
1680 boolean ifExists = false;
1681}
1682{
1683 pairId = QualifiedName() ifExists = IfExists()
1684 {
1685 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
1686 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001687 }
1688}
1689
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001690SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
1691{
1692 SynonymDropStatement stmt = null;
1693}
1694{
1695 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
1696 {
1697 return stmt;
1698 }
1699}
1700
1701SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
1702{
1703 Pair<DataverseName,Identifier> pairId = null;
1704 boolean ifExists = false;
1705}
1706{
1707 pairId = QualifiedName() ifExists = IfExists()
1708 {
1709 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
1710 return addSourceLocation(stmt, startStmtToken);
1711 }
1712}
1713
Yingyi Bu391f09e2015-10-29 13:49:39 -07001714boolean IfExists() throws ParseException :
1715{
1716}
1717{
1718 ( LOOKAHEAD(1) <IF> <EXISTS>
1719 {
1720 return true;
1721 }
1722 )?
1723 {
1724 return false;
1725 }
1726}
1727
1728InsertStatement InsertStatement() throws ParseException:
1729{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001730 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001731 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001732 VariableExpr var = null;
1733 Query query = null;
1734 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001735}
1736{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001737 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001738 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001739 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001740 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001741 if (var == null && returnExpression != null) {
1742 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1743 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001744 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08001745 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001746 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1747 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001748 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001749 }
1750}
1751
1752UpsertStatement UpsertStatement() throws ParseException:
1753{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001754 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001755 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001756 VariableExpr var = null;
1757 Query query = null;
1758 Expression returnExpression = null;
1759}
1760{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001761 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001762 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001763 ( <RETURNING> returnExpression = Expression())?
1764 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001765 if (var == null && returnExpression != null) {
1766 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1767 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001768 }
1769 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001770 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1771 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001772 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001773 }
1774}
1775
1776DeleteStatement DeleteStatement() throws ParseException:
1777{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001778 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001779 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001780 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001781 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001782}
1783{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001784 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001785 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001786 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07001787 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001788 if (var == null) {
1789 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1790 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001791 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001792 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07001793 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001794 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001795 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001796}
1797
1798UpdateStatement UpdateStatement() throws ParseException:
1799{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001800 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001801 VariableExpr vars;
1802 Expression target;
1803 Expression condition;
1804 UpdateClause uc;
1805 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
1806}
1807{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001808 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001809 <WHERE> condition = Expression()
1810 <LEFTPAREN> (uc = UpdateClause()
1811 {
1812 ucs.add(uc);
1813 }
1814 (<COMMA> uc = UpdateClause()
1815 {
1816 ucs.add(uc);
1817 }
1818 )*) <RIGHTPAREN>
1819 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001820 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001821 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001822 }
1823}
1824
1825UpdateClause UpdateClause() throws ParseException:
1826{
1827 Expression target = null;
1828 Expression value = null ;
1829 InsertStatement is = null;
1830 DeleteStatement ds = null;
1831 UpdateStatement us = null;
1832 Expression condition = null;
1833 UpdateClause ifbranch = null;
1834 UpdateClause elsebranch = null;
1835}
1836{
1837 (<SET> target = Expression() <EQ> value = Expression()
1838 | is = InsertStatement()
1839 | ds = DeleteStatement()
1840 | us = UpdateStatement()
1841 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
1842 <THEN> ifbranch = UpdateClause()
1843 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
1844 {
1845 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
1846 }
1847 )
1848}
1849
1850Statement SetStatement() throws ParseException:
1851{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001852 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001853 String pn = null;
1854 String pv = null;
1855}
1856{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001857 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001858 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001859 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001860 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001861 }
1862}
1863
1864Statement WriteStatement() throws ParseException:
1865{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001866 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001867 String nodeName = null;
1868 String fileName = null;
1869 Query query;
1870 String writerClass = null;
1871 Pair<Identifier,Identifier> nameComponents = null;
1872}
1873{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001874 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07001875 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001876 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001877 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001878 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001879 }
1880}
1881
1882LoadStatement LoadStatement() throws ParseException:
1883{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001884 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001885 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001886 Identifier datasetName = null;
1887 boolean alreadySorted = false;
1888 String adapterName;
1889 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001890 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001891}
1892{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001893 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001894 {
1895 dataverseName = nameComponents.first;
1896 datasetName = nameComponents.second;
1897 }
1898 <USING> adapterName = AdapterName() properties = Configuration()
1899 (<PRESORTED>
1900 {
1901 alreadySorted = true;
1902 }
1903 )?
1904 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001905 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
1906 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001907 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001908 }
1909}
1910
1911
1912String AdapterName() throws ParseException :
1913{
1914 String adapterName = null;
1915}
1916{
1917 adapterName = Identifier()
1918 {
1919 return adapterName;
1920 }
1921}
1922
1923Statement CompactStatement() throws ParseException:
1924{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001925 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001926 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001927}
1928{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001929 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001930 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001931 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001932 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001933 }
1934}
1935
Yingyi Buab817482016-08-19 21:29:31 -07001936Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001937{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001938 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001939 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001940}
1941{
1942 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001943 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
1944 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
1945 | <START> { startToken = token; } stmt = StartStatement(startToken)
1946 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07001947 )
1948 {
1949 return stmt;
1950 }
1951}
1952
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001953Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001954{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001955 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001956 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001957}
1958{
1959 <FEED> feedNameComponents = QualifiedName()
1960 {
1961 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001962 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001963 }
1964}
1965
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001966AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001967{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001968 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001969 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001970}
1971{
1972 <FEED> feedNameComponents = QualifiedName()
1973 {
1974 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001975 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001976 }
1977}
1978
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001979AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001980{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001981 Pair<DataverseName,Identifier> feedNameComponents = null;
1982 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07001983
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001984 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07001985}
1986{
1987 (
1988 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001989 {
1990 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
1991 }
1992 )
Yingyi Buab817482016-08-19 21:29:31 -07001993 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001994 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07001995 }
1996}
1997
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001998AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001999{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002000 Pair<DataverseName,Identifier> feedNameComponents = null;
2001 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002002
2003 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002004 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002005 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002006 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002007 String whereClauseBody = null;
2008 WhereClause whereClause = null;
2009 Token beginPos = null;
2010 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002011}
2012{
2013 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002014 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002015 (ApplyFunction(appliedFunctions))?
2016 (policy = GetPolicy())?
2017 (
2018 <WHERE>
2019 {
2020 beginPos = token;
2021 whereClause = new WhereClause();
2022 Expression whereExpr;
2023 }
2024 whereExpr = Expression()
2025 {
2026 whereClause.setWhereExpr(whereExpr);
2027 }
2028 )?
2029 {
2030 if (whereClause != null) {
2031 endPos = token;
2032 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2033 }
2034 }
Yingyi Buab817482016-08-19 21:29:31 -07002035 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002036 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002037 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002038 }
2039 )
2040 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002041 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002042 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002043}
2044
2045Map<String,String> Configuration() throws ParseException :
2046{
2047 Map<String,String> configuration = new LinkedHashMap<String,String>();
2048 Pair<String, String> keyValuePair = null;
2049}
2050{
2051 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2052 {
2053 configuration.put(keyValuePair.first, keyValuePair.second);
2054 }
2055 ( <COMMA> keyValuePair = KeyValuePair()
2056 {
2057 configuration.put(keyValuePair.first, keyValuePair.second);
2058 }
2059 )* )? <RIGHTPAREN>
2060 {
2061 return configuration;
2062 }
2063}
2064
2065Pair<String, String> KeyValuePair() throws ParseException:
2066{
2067 String key;
2068 String value;
2069}
2070{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002071 <LEFTPAREN> key = ConstantString() <EQ> value = ConstantString() <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002072 {
2073 return new Pair<String, String>(key, value);
2074 }
2075}
2076
2077Map<String,String> Properties() throws ParseException:
2078{
2079 Map<String,String> properties = new HashMap<String,String>();
2080 Pair<String, String> property;
2081}
2082{
2083 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
2084 {
2085 properties.put(property.first, property.second);
2086 }
2087 ( <COMMA> property = Property()
2088 {
2089 properties.put(property.first, property.second);
2090 }
2091 )* <RIGHTPAREN> )?
2092 {
2093 return properties;
2094 }
2095}
2096
2097Pair<String, String> Property() throws ParseException:
2098{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002099 String key = null;
2100 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002101}
2102{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002103 (key = Identifier() | key = StringLiteral())
2104 <EQ>
2105 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002106 {
2107 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002108 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002109 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002110 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002111 }
2112 }
2113 )
2114 {
2115 return new Pair<String, String>(key.toUpperCase(), value);
2116 }
2117}
2118
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002119IndexedTypeExpression IndexedTypeExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002120{
2121 TypeExpression typeExpr = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002122 boolean isUnknownable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002123}
2124{
2125 (
2126 typeExpr = TypeReference()
2127 | typeExpr = OrderedListTypeDef()
2128 | typeExpr = UnorderedListTypeDef()
2129 )
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002130 ( <QUES> { isUnknownable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002131 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002132 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002133 }
2134}
2135
2136TypeExpression TypeExpr() throws ParseException:
2137{
2138 TypeExpression typeExpr = null;
2139}
2140{
2141 (
2142 typeExpr = RecordTypeDef()
2143 | typeExpr = TypeReference()
2144 | typeExpr = OrderedListTypeDef()
2145 | typeExpr = UnorderedListTypeDef()
2146 )
2147 {
2148 return typeExpr;
2149 }
2150}
2151
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002152RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
2153{
2154 RecordTypeDefinition.RecordKind recordKind = null;
2155}
2156{
2157 (
2158 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
2159 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
2160 )
2161 {
2162 return recordKind;
2163 }
2164}
2165
Yingyi Bu391f09e2015-10-29 13:49:39 -07002166RecordTypeDefinition RecordTypeDef() throws ParseException:
2167{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002168 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002169 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002170 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002171}
2172{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002173 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002174 <LEFTBRACE>
2175 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002176 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002177 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
2178 if (hintToken != null) {
2179 String hintParams = hintToken.hintParams;
2180 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
2181 if (splits == null || splits.length != 4) {
2182 throw new SqlppParseException(getSourceLocation(hintToken),
2183 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002184 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002185 if (!splits[0].equals("int")) {
2186 throw new SqlppParseException(getSourceLocation(hintToken),
2187 "The only supported type for gen-fields is int.");
2188 }
2189 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
2190 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
2191 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002192 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002193 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002194 (
2195 RecordField(recType)
2196 ( <COMMA> RecordField(recType) )*
2197 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002198 <RIGHTBRACE>
2199 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002200 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002201 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002202 }
2203}
2204
2205void RecordField(RecordTypeDefinition recType) throws ParseException:
2206{
2207 String fieldName;
2208 TypeExpression type = null;
2209 boolean nullable = false;
2210}
2211{
2212 fieldName = Identifier()
2213 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002214 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
2215 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
2216 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
2217 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002218 }
2219 <COLON> type = TypeExpr() (<QUES> { nullable = true; } )?
2220 {
2221 recType.addField(fieldName, type, nullable, rfdg);
2222 }
2223}
2224
2225TypeReferenceExpression TypeReference() throws ParseException:
2226{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002227 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002228}
2229{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002230 id = QualifiedName()
2231 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07002232 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
2233 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002234 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002235
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002236 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002237 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002238 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002239}
2240
2241OrderedListTypeDefinition OrderedListTypeDef() throws ParseException:
2242{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002243 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002244 TypeExpression type = null;
2245}
2246{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002247 <LEFTBRACKET> { startToken = token; }
2248 ( type = TypeExpr() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002249 <RIGHTBRACKET>
2250 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002251 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002252 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002253 }
2254}
2255
Yingyi Bu391f09e2015-10-29 13:49:39 -07002256UnorderedListTypeDefinition UnorderedListTypeDef() throws ParseException:
2257{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002258 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002259 TypeExpression type = null;
2260}
2261{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002262 <LEFTDBLBRACE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002263 ( type = TypeExpr() )
2264 <RIGHTDBLBRACE>
2265 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002266 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002267 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002268 }
2269}
2270
2271FunctionName FunctionName() throws ParseException:
2272{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002273 Triple<List<String>, SourceLocation, SqlppHint> prefix = null;
2274 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002275}
2276{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002277 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
2278 // that copy must be kept in sync with this code
2279 prefix = MultipartIdentifierWithHints(SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
2280 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT)
2281 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002282 {
2283 FunctionName result = new FunctionName();
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002284 result.sourceLoc = prefix.second;
2285 result.hint = prefix.third;
2286 List<String> list = prefix.first;
2287 int ln = list.size();
2288 String last = list.get(ln - 1);
2289 if (suffix == null) {
2290 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
2291 // no library name
2292 result.function = last;
2293 } else {
2294 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
2295 // suffix = func_name
2296 result.library = last;
2297 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002298 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002299 if (ln > 1) {
2300 result.dataverse = DataverseName.create(list, 0, ln - 1);
2301 } else {
2302 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002303 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002304
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002305 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
2306 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002307 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002308 return result;
2309 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002310}
2311
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002312Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002313{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002314 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002315}
2316{
2317 name = QualifiedName()
2318 {
2319 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002320 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002321 }
2322 return name;
2323 }
2324}
2325
2326String Identifier() throws ParseException:
2327{
2328 String lit = null;
2329}
2330{
2331 (<IDENTIFIER>
2332 {
2333 return token.image;
2334 }
2335 | lit = QuotedString()
2336 {
2337 return lit;
2338 }
2339 )
2340}
2341
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002342Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002343{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002344 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002345 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002346}
2347{
2348 fieldList = NestedField()
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002349 ( <COLON> fieldType = IndexedTypeExpr() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002350 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002351 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
2352 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002353 }
2354}
2355
Yingyi Buc9bfe252016-03-01 00:02:40 -08002356Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002357{
2358 List<String> exprList = new ArrayList<String>();
2359 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002360 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002361 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002362}
2363{
2364 lit = Identifier()
2365 {
Yingyi Bub9169b62016-02-26 21:21:49 -08002366 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002367 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08002368 }
2369 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08002370 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08002371 <LEFTPAREN><RIGHTPAREN>
2372 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002373 if(!lit.equalsIgnoreCase("meta")){
2374 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08002375 }
2376 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002377 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08002378 }
2379 )?
2380 {
2381 if(!meetParens){
2382 exprList.add(lit);
2383 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002384 }
2385 (<DOT>
2386 lit = Identifier()
2387 {
2388 exprList.add(lit);
2389 }
2390 )*
2391 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002392 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002393 }
2394}
2395
Yingyi Bu6d57e492016-06-06 21:24:42 -07002396String ConstantString() throws ParseException:
2397{
2398 String value = null;
2399}
2400{
2401 (value = QuotedString() | value = StringLiteral())
2402 {
2403 return value;
2404 }
2405}
2406
Yingyi Bu391f09e2015-10-29 13:49:39 -07002407String QuotedString() throws ParseException:
2408{
2409}
2410{
2411 <QUOTED_STRING>
2412 {
2413 return removeQuotesAndEscapes(token.image);
2414 }
2415}
2416
Yingyi Bu391f09e2015-10-29 13:49:39 -07002417String StringLiteral() throws ParseException:
2418{
2419}
2420{
2421 <STRING_LITERAL>
2422 {
2423 return removeQuotesAndEscapes(token.image);
2424 }
2425}
2426
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002427List<String> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002428{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002429 Triple<List<String>, SourceLocation, SqlppHint> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002430}
2431{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002432 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002433 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002434 return result.first;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002435 }
2436}
2437
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002438Triple<List<String>, SourceLocation, SqlppHint> MultipartIdentifierWithHints(SqlppHint... expectedHints)
2439 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002440{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002441 List<String> list = new ArrayList<String>();
2442 SourceLocation sourceLoc = null;
2443 SqlppHint hint = null;
2444 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002445}
2446{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002447 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002448 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002449 list.add(item);
2450 sourceLoc = getSourceLocation(token);
2451 if (expectedHints != null && expectedHints.length > 0) {
2452 Token hintToken = fetchHint(token, expectedHints);
2453 if (hintToken != null) {
2454 hint = hintToken.hint;
2455 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002456 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002457 }
2458 (<DOT> item = Identifier() { list.add(item); } )*
2459 {
2460 return new Triple<List<String>, SourceLocation, SqlppHint>(list, sourceLoc, hint);
2461 }
2462}
2463
2464Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
2465{
2466 List<String> list = null;
2467}
2468{
2469 list = MultipartIdentifier()
2470 {
2471 int len = list.size();
2472 DataverseName id1 = len > 1 ? DataverseName.create(list, 0, len - 1) : null;
2473 Identifier id2 = new Identifier(list.get(len - 1));
2474 return new Pair<DataverseName,Identifier>(id1, id2);
2475 }
2476}
2477
2478Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
2479{
2480 List<String> list = new ArrayList<String>();
2481 String item = null;
2482}
2483{
2484 item = Identifier() { list.add(item); }
2485 (<DOT> item = Identifier() { list.add(item); } )+
2486 {
2487 int len = list.size();
2488 DataverseName id1 = len > 2 ? DataverseName.create(list, 0, len - 2) : null;
2489 Identifier id2 = new Identifier(list.get(len - 2));
2490 Identifier id3 = new Identifier(list.get(len - 1));
2491 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002492 }
2493}
2494
2495FunctionDecl FunctionDeclaration() throws ParseException:
2496{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002497 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002498 String functionName;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08002499 List<Pair<VarIdentifier,IndexedTypeExpression>> paramList = Collections.<Pair<VarIdentifier,IndexedTypeExpression>>emptyList();;
Ian Maxon38fe9402020-01-29 19:27:40 -08002500 List<VarIdentifier> paramVarOnly = new ArrayList<VarIdentifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002501 Expression funcBody;
2502 createNewScope();
2503}
2504{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002505 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07002506 functionName = Identifier()
Ian Maxon38fe9402020-01-29 19:27:40 -08002507 paramList = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07002508 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08002509 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07002510 <RIGHTBRACE>
2511 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002512 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, paramList.size());
Xikui Wang3de700a2018-03-15 16:32:55 -07002513 getCurrentScope().addFunctionDescriptor(signature, false);
Ian Maxon38fe9402020-01-29 19:27:40 -08002514 for(Pair<VarIdentifier,IndexedTypeExpression> p: paramList){
2515 paramVarOnly.add(p.getFirst());
2516 }
2517 FunctionDecl stmt = new FunctionDecl(signature, paramVarOnly, funcBody);
Xikui Wang3de700a2018-03-15 16:32:55 -07002518 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002519 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07002520 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002521}
2522
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002523Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002524{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002525 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002526 Expression expr;
2527}
2528{
2529 (
2530 expr = Expression()
2531 |
2532 expr = SelectExpression(false)
2533 )
2534 {
2535 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002536 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07002537 return query;
2538 }
2539}
2540
2541
Yingyi Bu391f09e2015-10-29 13:49:39 -07002542Expression Expression():
2543{
2544 Expression expr = null;
2545 Expression exprP = null;
2546}
2547{
2548(
2549 LOOKAHEAD(2)
2550 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002551 | expr = QuantifiedExpression()
2552)
2553 {
2554 return (exprP==null) ? expr : exprP;
2555 }
2556}
2557
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002558Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002559{
2560 OperatorExpr op = null;
2561 Expression operand = null;
2562}
2563{
2564 operand = AndExpr()
2565 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002566 <OR>
2567 {
2568 if (op == null) {
2569 op = new OperatorExpr();
2570 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07002571 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002572 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002573 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002574 try{
2575 op.addOperator(token.image.toLowerCase());
2576 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002577 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002578 }
Xikui Wang3de700a2018-03-15 16:32:55 -07002579 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002580
Xikui Wang3de700a2018-03-15 16:32:55 -07002581 operand = AndExpr()
2582 {
2583 op.addOperand(operand);
2584 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002585
2586 )*
2587
2588 {
2589 return op==null? operand: op;
2590 }
2591}
2592
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002593Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002594{
2595 OperatorExpr op = null;
2596 Expression operand = null;
2597}
2598{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002599 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002600 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002601 <AND>
2602 {
2603 if (op == null) {
2604 op = new OperatorExpr();
2605 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002606 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002607 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002608 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002609 try{
2610 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002611 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002612 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002613 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002614 }
2615
Yingyi Bu196db5d2016-07-15 19:07:20 -07002616 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002617 {
2618 op.addOperand(operand);
2619 }
2620
2621 )*
2622
2623 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002624 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002625 }
2626}
2627
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002628Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07002629{
2630 Expression inputExpr;
2631 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002632 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07002633}
2634{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002635 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07002636 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002637 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002638 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002639 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002640 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002641 } else {
2642 return inputExpr;
2643 }
2644 }
2645}
Yingyi Bu391f09e2015-10-29 13:49:39 -07002646
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002647Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002648{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002649 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002650 OperatorExpr op = null;
2651 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002652 IExpressionAnnotation annotation = null;
2653}
2654{
Yingyi Bu6c638342016-09-02 17:54:34 -07002655 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002656
2657 (
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002658 LOOKAHEAD(2)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN>)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002659 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002660 Token hintToken = fetchHint(token, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
Shiva2ea73232019-10-09 18:04:05 -07002661 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.HASH_BROADCAST_JOIN_HINT);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002662 if (hintToken != null) {
2663 switch (hintToken.hint) {
2664 case INDEXED_NESTED_LOOP_JOIN_HINT:
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002665 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002666 break;
2667 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002668 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002669 break;
Shiva2ea73232019-10-09 18:04:05 -07002670 case HASH_BROADCAST_JOIN_HINT:
2671 annotation = new BroadcastExpressionAnnotation();
2672 annotation.setObject(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002673 break;
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002674 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002675 }
Yingyi Buea4ec722016-11-04 01:26:16 -07002676
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002677 String operator = token.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07002678 if (operator.equals("<>")){
2679 operator = "!=";
2680 }
2681 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002682 operator = "not_" + operator;
2683 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002684 if (op == null) {
2685 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07002686 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002687 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002688 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002689 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002690 try{
2691 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002692 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002693 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002694 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002695 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002696
Yingyi Bu6c638342016-09-02 17:54:34 -07002697 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07002698 {
Shiva2ea73232019-10-09 18:04:05 -07002699 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07002700 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002701 )?
2702
2703 {
2704 if (annotation != null) {
2705 op.addHint(annotation);
2706 }
2707 return op==null? operand: op;
2708 }
2709}
2710
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002711Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07002712{
2713 boolean not = false;
2714 OperatorExpr op = null;
2715 Expression operand = null;
2716 IExpressionAnnotation annotation = null;
2717}
2718{
2719 operand = IsExpr()
2720 (
2721 LOOKAHEAD(2)
2722 (<NOT> { not = true; })? <BETWEEN>
2723 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002724 Token hintToken = fetchHint(token, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
2725 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT);
2726 if (hintToken != null) {
2727 switch (hintToken.hint) {
2728 case INDEXED_NESTED_LOOP_JOIN_HINT:
Yingyi Bu6c638342016-09-02 17:54:34 -07002729 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002730 break;
2731 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
Yingyi Bu6c638342016-09-02 17:54:34 -07002732 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002733 break;
Yingyi Bu6c638342016-09-02 17:54:34 -07002734 }
2735 }
2736 String operator = token.image.toLowerCase();
2737 if(not){
2738 operator = "not_" + operator;
2739 }
2740 if (op == null) {
2741 op = new OperatorExpr();
2742 op.addOperand(operand);
2743 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002744 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07002745 }
2746 try{
2747 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002748 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002749 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07002750 }
2751 }
2752
2753 operand = IsExpr()
2754 {
2755 op.addOperand(operand);
2756 }
2757
2758 <AND>
2759 operand = IsExpr()
2760 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08002761 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07002762 op.addOperand(operand);
2763 }
2764 )?
2765
2766 {
2767 if (annotation != null) {
2768 op.addHint(annotation);
2769 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002770 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07002771 }
2772}
2773
Yingyi Budaa549c2016-06-28 22:30:52 -07002774Expression IsExpr() throws ParseException:
2775{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002776 Token notToken = null;
2777 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002778 Expression operand = null;
2779 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002780 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002781}
2782{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002783 operand = LikeExpr()
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002784 ( <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002785 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002786 (
2787 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
2788 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08002789 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07002790 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002791 )
Yingyi Budaa549c2016-06-28 22:30:52 -07002792 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002793 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07002794 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002795 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002796 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002797 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07002798 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002799 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07002800 }
2801 }
2802 )?
2803 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002804 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07002805 }
2806}
2807
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002808Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002809{
2810 boolean not = false;
2811 OperatorExpr op = null;
2812 Expression operand = null;
2813}
2814{
2815 operand = ConcatExpr()
2816 (
2817 LOOKAHEAD(2)
2818 (<NOT> { not = true; })? <LIKE>
2819 {
2820 op = new OperatorExpr();
2821 op.addOperand(operand);
2822 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002823 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002824
2825 String operator = token.image.toLowerCase();
2826 if (not) {
2827 operator = "not_" + operator;
2828 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002829 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002830 op.addOperator(operator);
2831 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002832 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002833 }
2834 }
2835
2836 operand = ConcatExpr()
2837 {
2838 op.addOperand(operand);
2839 }
2840 )?
2841
2842 {
2843 return op == null ? operand : op;
2844 }
2845}
2846
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002847Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002848{
2849 OperatorExpr op = null;
2850 Expression operand = null;
2851}
2852{
2853 operand = AddExpr()
2854 (
2855 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002856 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002857 {
2858 if (op == null) {
2859 op = new OperatorExpr();
2860 op.addOperand(operand);
2861 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002862 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002863 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002864 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002865 }
2866 operand = AddExpr()
2867 {
2868 op.addOperand(operand);
2869 }
2870 )*
2871
2872 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002873 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002874 }
2875}
Yingyi Budaa549c2016-06-28 22:30:52 -07002876
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002877Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002878{
2879 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002880 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002881 Expression operand = null;
2882}
2883{
2884 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07002885 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002886 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002887 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002888 {
2889 if (op == null) {
2890 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002891 op.addOperand(operand);
2892 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002893 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002894 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002895 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002896 }
2897
2898 operand = MultExpr()
2899 {
2900 op.addOperand(operand);
2901 }
2902 )*
2903
2904 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002905 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002906 }
2907}
2908
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002909Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002910{
2911 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002912 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002913 Expression operand = null;
2914}
2915{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002916 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002917
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002918 ( (
2919 <MUL> { opType = OperatorType.MUL; } |
2920 <DIVIDE> { opType = OperatorType.DIVIDE; } |
2921 <DIV> { opType = OperatorType.DIV; } |
2922 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
2923 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002924 {
2925 if (op == null) {
2926 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002927 op.addOperand(operand);
2928 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002929 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002930 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002931 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002932 }
2933 operand = ExponentExpr()
2934 {
2935 op.addOperand(operand);
2936 }
2937 )*
2938
2939 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002940 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002941 }
2942}
2943
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002944Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002945{
2946 OperatorExpr op = null;
2947 Expression operand = null;
2948}
2949{
2950 operand = UnaryExpr()
2951 (<CARET>
2952 {
2953 if (op == null) {
2954 op = new OperatorExpr();
2955 op.addOperand(operand);
2956 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002957 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002958 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002959 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002960 }
2961 operand = UnaryExpr()
2962 {
2963 op.addOperand(operand);
2964 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002965 )?
2966 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002967 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002968 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002969}
2970
2971Expression UnaryExpr() throws ParseException:
2972{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002973 boolean not = false;
2974 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002975 Expression expr = null;
2976}
Yingyi Budaa549c2016-06-28 22:30:52 -07002977{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002978 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002979 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002980 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002981 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002982 exprType = "not_" + exprType;
2983 }
2984 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002985 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002986 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002987 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002988 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002989 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07002990 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002991 }
2992 )?
2993
2994 expr = ValueExpr()
2995 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002996 if (uexpr == null) {
2997 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002998 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002999 uexpr.setExpr(expr);
3000 return uexpr;
3001 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003002 }
3003}
3004
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003005Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003006{
3007 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003008 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003009}
3010{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003011 expr = PrimaryExpr()
3012 (
3013 accessor = FieldAccessor(accessor != null ? accessor : expr)
3014 |
3015 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003016 )*
3017 {
3018 return accessor == null ? expr : accessor;
3019 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003020}
3021
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003022FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003023{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003024 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003025 String ident = null;
3026}
3027{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003028 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003029 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003030 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003031 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003032 }
3033}
3034
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003035AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003036{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003037 Token startToken = null;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003038 boolean isListSliceExpression = false;
3039 AbstractAccessor resultExpression = null;
3040 Expression argumentExpression1 = null;
3041 Expression argumentExpression2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003042}
3043{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003044 <LEFTBRACKET> { startToken = token; }
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003045 ( argumentExpression1 = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003046 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003047 if (argumentExpression1.getKind() == Expression.Kind.LITERAL_EXPRESSION)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003048 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003049 Literal lit = ((LiteralExpr)argumentExpression1).getValue();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003050 if (lit.getLiteralType() != Literal.Type.INTEGER &&
3051 lit.getLiteralType() != Literal.Type.LONG) {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003052 throw new SqlppParseException(argumentExpression1.getSourceLocation(), "Index should be an INTEGER");
Yingyi Bu391f09e2015-10-29 13:49:39 -07003053 }
3054 }
3055 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003056 )
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003057 (<COLON>
3058 {
3059 isListSliceExpression = true;
3060 }
3061 ( argumentExpression2 = Expression()
3062 {
3063 if (argumentExpression2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3064 Literal lit = ((LiteralExpr)argumentExpression2).getValue();
3065 if (lit.getLiteralType() != Literal.Type.INTEGER &&
3066 lit.getLiteralType() != Literal.Type.LONG) {
3067 throw new SqlppParseException(argumentExpression2.getSourceLocation(), "Index should be an INTEGER");
3068 }
3069 }
3070 })?
3071 )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003072 <RIGHTBRACKET>
3073 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003074 if (!isListSliceExpression) {
3075 resultExpression = new IndexAccessor(inputExpr, argumentExpression1);
3076 } else {
3077 resultExpression = new ListSliceExpression(inputExpr, argumentExpression1, argumentExpression2);
3078 }
3079
3080 return addSourceLocation(resultExpression, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003081 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003082}
3083
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003084Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003085{
3086 Expression expr = null;
3087}
3088{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003089 (
3090 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07003091 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003092 | expr = Literal()
3093 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003094 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003095 | expr = ListConstructor()
3096 | expr = RecordConstructor()
3097 | expr = ParenthesizedExpression()
3098 )
3099 {
3100 return expr;
3101 }
3102}
3103
3104Expression Literal() throws ParseException:
3105{
3106 LiteralExpr lit = new LiteralExpr();
3107 String str = null;
3108}
3109{
3110 ( str = StringLiteral()
3111 {
3112 lit.setValue(new StringLiteral(str));
3113 }
3114 | <INTEGER_LITERAL>
3115 {
Till Westmann68c6a992016-09-30 00:19:12 -07003116 try {
3117 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
3118 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003119 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003120 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003121 }
3122 | <FLOAT_LITERAL>
3123 {
Till Westmann68c6a992016-09-30 00:19:12 -07003124 try {
3125 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
3126 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003127 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003128 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003129 }
3130 | <DOUBLE_LITERAL>
3131 {
Till Westmann68c6a992016-09-30 00:19:12 -07003132 try {
3133 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
3134 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003135 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003136 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003137 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07003138 | <MISSING>
3139 {
3140 lit.setValue(MissingLiteral.INSTANCE);
3141 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003142 | <NULL>
3143 {
3144 lit.setValue(NullLiteral.INSTANCE);
3145 }
3146 | <TRUE>
3147 {
3148 lit.setValue(TrueLiteral.INSTANCE);
3149 }
3150 | <FALSE>
3151 {
3152 lit.setValue(FalseLiteral.INSTANCE);
3153 }
3154 )
3155 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003156 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003157 }
3158}
3159
Yingyi Bu391f09e2015-10-29 13:49:39 -07003160VariableExpr VariableRef() throws ParseException:
3161{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003162 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003163}
3164{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003165 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003166 {
3167 Identifier ident = lookupSymbol(id);
3168 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003169 throw new SqlppParseException(getSourceLocation(token),
3170 "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 -07003171 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003172 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003173 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003174 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003175 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003176 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07003177 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003178 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003179 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003180 }
3181}
3182
Yingyi Bu391f09e2015-10-29 13:49:39 -07003183VariableExpr Variable() throws ParseException:
3184{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003185 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003186}
3187{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003188 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003189 {
3190 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003191 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
3192 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07003193 varExp.setIsNewVar(false);
3194 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003195 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003196 }
3197}
3198
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003199String VariableIdentifier() throws ParseException:
3200{
3201 String id = null;
3202}
3203{
3204 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3205 {
3206 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
3207 }
3208}
3209
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003210Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
3211{
3212 VariableExpr var = null;
3213 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
3214}
3215{
3216 var = Variable()
3217 ( LOOKAHEAD(1)
3218 {
3219 VariableExpr fieldVarExpr = null;
3220 String fieldIdentifierStr = null;
3221 }
3222 <LEFTPAREN>
3223 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3224 {
3225 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3226 }
3227 (<COMMA>
3228 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3229 {
3230 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3231 }
3232 )*
3233 <RIGHTPAREN>
3234 )?
3235 {
3236 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
3237 }
3238}
3239
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003240VariableExpr ExternalVariableRef() throws ParseException:
3241{
3242 String name = null;
3243}
3244{
3245 (
3246 (
3247 <DOLLAR>
3248 (
3249 <INTEGER_LITERAL> { name = token.image; } |
3250 <IDENTIFIER> { name = token.image; } |
3251 name = QuotedString()
3252 )
3253 )
3254 |
3255 (
3256 <QUES> { name = String.valueOf(++externalVarCounter); }
3257 )
3258 )
3259 {
3260 String idName = SqlppVariableUtil.toExternalVariableName(name);
3261 VarIdentifier id = new VarIdentifier(idName);
3262 VariableExpr varExp = new VariableExpr(id);
3263 return addSourceLocation(varExp, token);
3264 }
3265}
3266
Yingyi Bu391f09e2015-10-29 13:49:39 -07003267Expression ListConstructor() throws ParseException:
3268{
3269 Expression expr = null;
3270}
3271{
3272 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003273 expr = OrderedListConstructor() |
3274 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003275 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003276 {
3277 return expr;
3278 }
3279}
3280
Yingyi Bu391f09e2015-10-29 13:49:39 -07003281ListConstructor OrderedListConstructor() throws ParseException:
3282{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003283 Token startToken = null;
3284 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003285}
3286{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003287 <LEFTBRACKET> { startToken = token; }
3288 exprList = ExpressionList()
3289 <RIGHTBRACKET>
3290 {
3291 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003292 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003293 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003294}
3295
3296ListConstructor UnorderedListConstructor() throws ParseException:
3297{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003298 Token startToken = null;
3299 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003300}
3301{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003302 <LEFTDBLBRACE> { startToken = token; }
3303 exprList = ExpressionList()
3304 <RIGHTDBLBRACE>
3305 {
3306 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003307 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003308 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003309}
3310
3311List<Expression> ExpressionList() throws ParseException:
3312{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003313 Expression expr = null;
3314 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003315}
3316{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003317 (
3318 expr = Expression()
3319 {
3320 exprList.add(expr);
3321 }
3322 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07003323 {
3324 exprList.add(expr);
3325 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003326 )*
3327 )?
3328 {
3329 return exprList;
3330 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003331}
3332
Yingyi Bu391f09e2015-10-29 13:49:39 -07003333RecordConstructor RecordConstructor() throws ParseException:
3334{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003335 Token startToken = null;
3336 FieldBinding fb = null;
3337 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003338}
3339{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003340 <LEFTBRACE> { startToken = token; }
3341 (
3342 fb = FieldBinding() { fbList.add(fb); }
3343 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
3344 )?
3345 <RIGHTBRACE>
3346 {
3347 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003348 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003349 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003350}
3351
3352FieldBinding FieldBinding() throws ParseException:
3353{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003354 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003355}
3356{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003357 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003358 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003359 if (right == null) {
3360 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
3361 if (generatedIdentifier == null) {
3362 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
3363 }
3364 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07003365 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
3366 generatedNameExpr.setSourceLocation(left.getSourceLocation());
3367 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003368 } else {
3369 return new FieldBinding(left, right);
3370 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003371 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003372}
3373
Yingyi Bu391f09e2015-10-29 13:49:39 -07003374Expression FunctionCallExpr() throws ParseException:
3375{
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003376 Expression resultExpr;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003377 CallExpr callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003378 List<Expression> argList = new ArrayList<Expression>();
Yingyi Buf4d09842016-08-26 00:03:52 -07003379 Expression tmp = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003380 int arity = 0;
3381 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07003382 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003383 boolean distinct = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003384}
3385{
3386 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003387 <LEFTPAREN> (
3388 ( <DISTINCT> { distinct = true; } )?
3389 ( tmp = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003390 {
Yingyi Buf4d09842016-08-26 00:03:52 -07003391 if(star){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003392 if(!funcName.function.equalsIgnoreCase("count")){
3393 throw new SqlppParseException(getSourceLocation(token), "The parameter * can only be used in COUNT().");
Yingyi Buf4d09842016-08-26 00:03:52 -07003394 }
3395 argList.add(new LiteralExpr(new LongIntegerLiteral(1L)));
3396 } else {
3397 argList.add(tmp);
3398 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003399 arity ++;
3400 }
3401 (<COMMA> tmp = Expression()
3402 {
3403 argList.add(tmp);
3404 arity++;
3405 }
3406 )*)? <RIGHTPAREN>
3407 {
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003408 String name = funcName.function;
3409 if (distinct) {
Dmitry Lychagine5ea42d2019-03-15 11:38:09 -07003410 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003411 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003412 // TODO use funcName.library
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003413 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003414 FunctionSignature signature
3415 = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
3416 if (signature == null) {
3417 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
3418 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003419 callExpr = FunctionMapUtil.normalizedListInputFunctions(new CallExpr(signature,argList));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003420 if (funcName.hint != null) {
3421 switch (funcName.hint) {
3422 case INDEXED_NESTED_LOOP_JOIN_HINT:
3423 callExpr.addHint(IndexedNLJoinExpressionAnnotation.INSTANCE);
3424 break;
3425 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
3426 callExpr.addHint(SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE);
3427 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003428 }
3429 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003430 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003431 resultExpr = callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003432 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003433
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003434 ( LOOKAHEAD(5) resultExpr = WindowExpr(callExpr.getFunctionSignature(), callExpr.getExprList()) )?
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003435
3436 {
3437 return resultExpr;
3438 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003439}
3440
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003441WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList) throws ParseException:
3442{
3443 Boolean fromLast = null, ignoreNulls = null;
3444}
3445{
3446 (
3447 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
3448 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
3449 <FROM> <IDENTIFIER>
3450 {
3451 if (isToken(FIRST)) {
3452 fromLast = false;
3453 } else if (isToken(LAST)) {
3454 fromLast = true;
3455 } else {
3456 throw createUnexpectedTokenError();
3457 }
3458 }
3459 )?
3460 (
3461 // ( RESPECT | IGNORE ) NULLS OVER
3462 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
3463 <IDENTIFIER>
3464 {
3465 if (isToken(RESPECT)) {
3466 ignoreNulls = false;
3467 } else if (isToken(IGNORE)) {
3468 ignoreNulls = true;
3469 } else {
3470 throw createUnexpectedTokenError();
3471 }
3472 }
3473 <IDENTIFIER>
3474 {
3475 if (!isToken(NULLS)) {
3476 throw createUnexpectedTokenError();
3477 }
3478 }
3479 )?
3480 <OVER>
3481 {
3482 return OverClause(signature, argList, token, fromLast, ignoreNulls);
3483 }
3484}
3485
3486WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Token startToken, Boolean fromLast,
3487 Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003488{
3489 Expression partitionExpr = null;
3490 List<Expression> partitionExprs = new ArrayList<Expression>();
3491 OrderbyClause orderByClause = null;
3492 List<Expression> orderbyList = null;
3493 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
3494 WindowExpression.FrameMode frameMode = null;
3495 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
3496 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
3497 Expression frameStartExpr = null, frameEndExpr = null;
3498 WindowExpression.FrameExclusionKind frameExclusionKind = null;
3499 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
3500 VariableExpr windowVar = null;
3501 List<Pair<Expression, Identifier>> windowFieldList = null;
3502}
3503{
3504 (
3505 windowVarWithFieldList = VariableWithFieldMap() <AS>
3506 {
3507 windowVar = windowVarWithFieldList.first;
3508 windowFieldList = windowVarWithFieldList.second;
3509 }
3510 )?
3511 <LEFTPAREN>
3512 (
3513 <IDENTIFIER> { expectToken(PARTITION); } <BY>
3514 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
3515 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
3516 )?
3517 (
3518 orderByClause = OrderbyClause()
3519 {
3520 orderbyList = orderByClause.getOrderbyList();
3521 orderbyModifierList = orderByClause.getModifierList();
3522 }
3523 (
3524 frameMode = WindowFrameMode()
3525 (
3526 frameStart = WindowFrameBoundary() |
3527 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
3528 )
3529 ( frameExclusionKind = WindowFrameExclusion() )?
3530 {
3531 frameStartKind = frameStart.first;
3532 frameStartExpr = frameStart.second;
3533 if (frameEnd == null) {
3534 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3535 } else {
3536 frameEndKind = frameEnd.first;
3537 frameEndExpr = frameEnd.second;
3538 }
3539 if (frameExclusionKind == null) {
3540 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
3541 }
3542 }
3543 )?
3544 )?
3545 <RIGHTPAREN>
3546 {
3547 WindowExpression winExp = new WindowExpression(signature, argList, partitionExprs, orderbyList, orderbyModifierList,
3548 frameMode, frameStartKind, frameStartExpr, frameEndKind, frameEndExpr, frameExclusionKind, windowVar,
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003549 windowFieldList, ignoreNulls, fromLast);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003550 return addSourceLocation(winExp, startToken);
3551 }
3552}
3553
3554WindowExpression.FrameMode WindowFrameMode() throws ParseException:
3555{
3556}
3557{
3558 <IDENTIFIER>
3559 {
3560 if (isToken(RANGE)) {
3561 return WindowExpression.FrameMode.RANGE;
3562 } else if (isToken(ROWS)) {
3563 return WindowExpression.FrameMode.ROWS;
3564 } else if (isToken(GROUPS)) {
3565 return WindowExpression.FrameMode.GROUPS;
3566 } else {
3567 throw createUnexpectedTokenError();
3568 }
3569 }
3570}
3571
3572Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
3573{
3574 boolean current = false;
3575 Expression expr = null;
3576}
3577{
3578 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003579 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003580 | expr = Expression()
3581 )
3582 <IDENTIFIER>
3583 {
3584 WindowExpression.FrameBoundaryKind kind;
3585 if (current && isToken(ROW)) {
3586 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3587 } else if (!current && isToken(PRECEDING)) {
3588 kind = expr == null
3589 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
3590 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
3591 } else if (!current && isToken(FOLLOWING)) {
3592 kind = expr == null
3593 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
3594 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
3595 } else {
3596 throw createUnexpectedTokenError();
3597 }
3598 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
3599 }
3600}
3601
3602WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
3603{
3604 boolean current = false, no = false;
3605}
3606{
3607 <IDENTIFIER>
3608 {
3609 expectToken(EXCLUDE);
3610 }
3611 (
3612 <GROUP>
3613 {
3614 return WindowExpression.FrameExclusionKind.GROUP;
3615 }
3616 |
3617 (
3618 <IDENTIFIER>
3619 {
3620 if (isToken(TIES)) {
3621 return WindowExpression.FrameExclusionKind.TIES;
3622 } else if (isToken(CURRENT)) {
3623 current = true;
3624 } else if (isToken(NO)) {
3625 no = true;
3626 } else {
3627 throw createUnexpectedTokenError();
3628 }
3629 }
3630 <IDENTIFIER>
3631 {
3632 if (current && isToken(ROW)) {
3633 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
3634 } else if (no && isToken(OTHERS)) {
3635 return WindowExpression.FrameExclusionKind.NO_OTHERS;
3636 } else {
3637 throw createUnexpectedTokenError();
3638 }
3639 }
3640 )
3641 )
3642}
3643
Yingyi Bu391f09e2015-10-29 13:49:39 -07003644Expression ParenthesizedExpression() throws ParseException:
3645{
3646 Expression expr;
3647}
3648{
3649 (
3650 LOOKAHEAD(2)
3651 <LEFTPAREN> expr = Expression() <RIGHTPAREN>
3652 |
3653 expr = Subquery()
3654 )
3655 {
3656 return expr;
3657 }
3658}
3659
Yingyi Buc8c067c2016-07-25 23:37:19 -07003660Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003661{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003662 Token startToken = null;
3663 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003664 List<Expression> whenExprs = new ArrayList<Expression>();
3665 List<Expression> thenExprs = new ArrayList<Expression>();
3666 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003667 Expression whenExpr = null;
3668 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003669}
3670{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003671 <CASE> { startToken = token; }
3672 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07003673 (
3674 <WHEN> whenExpr = Expression()
3675 {
3676 whenExprs.add(whenExpr);
3677 }
3678 <THEN> thenExpr = Expression()
3679 {
3680 thenExprs.add(thenExpr);
3681 }
3682 )*
3683 (<ELSE> elseExpr = Expression() )?
3684 <END>
3685 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003686 if (conditionExpr == null) {
3687 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003688 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003689 }
3690 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003691 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07003692 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003693}
3694
Yingyi Buab817482016-08-19 21:29:31 -07003695SelectExpression SelectExpression(boolean subquery) throws ParseException:
3696{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003697 List<LetClause> letClauses = new ArrayList<LetClause>();
3698 SelectSetOperation selectSetOperation;
3699 OrderbyClause orderbyClause = null;
3700 LimitClause limitClause = null;
3701 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07003702}
3703{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003704 ( letClauses = LetClause() )?
3705 selectSetOperation = SelectSetOperation()
3706 (orderbyClause = OrderbyClause() {})?
3707 (limitClause = LimitClause() {})?
3708 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003709 SelectExpression selectExpr =
3710 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
3711 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
3712 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003713 }
3714}
3715
Yingyi Buab817482016-08-19 21:29:31 -07003716SelectSetOperation SelectSetOperation() throws ParseException:
3717{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003718 SetOperationInput setOperationInputLeft;
3719 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
3720}
3721{
3722 {
3723 SelectBlock selectBlockLeft = null;
3724 SelectExpression subqueryLeft = null;
3725 Expression expr = null;
3726 }
3727 selectBlockLeft = SelectBlock()
3728 {
3729 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
3730 }
3731 (
3732 {
3733 SetOpType opType = SetOpType.UNION;
3734 boolean setSemantics = true;
3735 SelectBlock selectBlockRight = null;
3736 SelectExpression subqueryRight = null;
3737 }
3738 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
3739 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
3740 {
3741 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
3742 }
3743 )*
3744 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003745 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
3746 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
3747 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003748 }
3749}
3750
Yingyi Buab817482016-08-19 21:29:31 -07003751SelectExpression Subquery() throws ParseException:
3752{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003753 SelectExpression selectExpr = null;
3754}
3755{
3756 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
3757 {
3758 return selectExpr;
3759 }
3760}
3761
Yingyi Buab817482016-08-19 21:29:31 -07003762SelectBlock SelectBlock() throws ParseException:
3763{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003764 SelectClause selectClause = null;
3765 FromClause fromClause = null;
3766 List<LetClause> fromLetClauses = null;
3767 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003768 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003769 GroupbyClause groupbyClause = null;
3770 List<LetClause> gbyLetClauses = null;
3771 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003772 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003773 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003774}
3775{
3776 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003777 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003778 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003779 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07003780 fromClause = FromClause()
3781 (
3782 fromLetClauses = LetClause()
3783 )?
3784 )
3785 |
3786 (
3787 fromLetClauses = LetClause()
3788 {
3789 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
3790 SourceLocation sourceLoc = getSourceLocation(token);
3791 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
3792 missingExpr.setSourceLocation(sourceLoc);
3793 List<Expression> list = new ArrayList<Expression>(1);
3794 list.add(missingExpr);
3795 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
3796 listExpr.setSourceLocation(sourceLoc);
3797 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
3798 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
3799 fromVar.setSourceLocation(sourceLoc);
3800 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
3801 fromClause = new FromClause(fromTerms);
3802 }
3803 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003804 )?
3805 (whereClause = WhereClause())?
3806 (
3807 groupbyClause = GroupbyClause()
3808 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003809 gbyLetClauses = LetClause()
3810 )?
3811 (havingClause = HavingClause())?
3812 )?
3813 |
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003814 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003815 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003816 fromLetClauses = LetClause()
3817 )?
3818 (whereClause = WhereClause())?
3819 (
3820 groupbyClause = GroupbyClause()
3821 (
3822 gbyLetClauses = LetClause()
3823 )?
3824 (havingClause = HavingClause())?
3825 )?
3826 selectClause = SelectClause()
3827 )
3828 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003829 if (fromLetClauses != null) {
3830 fromLetWhereClauses.addAll(fromLetClauses);
3831 }
3832 if (whereClause != null) {
3833 fromLetWhereClauses.add(whereClause);
3834 }
3835 if (gbyLetClauses != null) {
3836 gbyLetHavingClauses.addAll(gbyLetClauses);
3837 }
3838 if (havingClause != null) {
3839 gbyLetHavingClauses.add(havingClause);
3840 }
3841 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
3842 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003843 selectBlock.setSourceLocation(startSrcLoc);
3844 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003845 }
3846}
3847
Yingyi Buab817482016-08-19 21:29:31 -07003848SelectClause SelectClause() throws ParseException:
3849{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003850 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003851 SelectRegular selectRegular = null;
3852 SelectElement selectElement = null;
3853 boolean distinct = false;
3854}
3855{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003856 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003857 (
3858 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04003859 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07003860 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07003861 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003862 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003863 SourceLocation sourceLoc = getSourceLocation(startToken);
3864 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07003865 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003866 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003867 List<Projection> projections = new ArrayList<Projection>();
3868 projections.add(projection);
3869 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003870 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003871 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003872 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
3873 selectClause.setSourceLocation(sourceLoc);
3874 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003875 }
3876}
3877
Yingyi Buab817482016-08-19 21:29:31 -07003878SelectRegular SelectRegular() throws ParseException:
3879{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003880 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003881 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003882 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003883}
3884{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003885 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003886 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003887 projections.add(projection);
3888 startSrcLoc = projection.getSourceLocation();
3889 }
3890 ( LOOKAHEAD(2) <COMMA> projection = Projection()
3891 {
3892 projections.add(projection);
3893 }
3894 )*
3895 {
3896 SelectRegular selectRegular = new SelectRegular(projections);
3897 selectRegular.setSourceLocation(startSrcLoc);
3898 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003899 }
3900}
3901
Yingyi Buab817482016-08-19 21:29:31 -07003902SelectElement SelectElement() throws ParseException:
3903{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003904 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003905 Expression expr = null;
3906 String name = null;
3907}
3908{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003909 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003910 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003911 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003912 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003913 }
3914}
3915
Yingyi Buab817482016-08-19 21:29:31 -07003916Projection Projection() throws ParseException :
3917{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003918 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003919 Expression expr = null;
3920 Identifier identifier = null;
3921 String name = null;
3922 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003923 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003924}
3925{
3926 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003927 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
3928 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003929 | expr = Expression() ((<AS>)? name = Identifier())?
3930 {
3931 if (name == null) {
3932 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
3933 if (generatedColumnIdentifier != null) {
3934 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
3935 }
3936 }
3937 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003938 )
3939 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003940 Projection projection = new Projection(expr, name, star, varStar);
3941 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
3942 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003943 }
3944}
3945
3946FromClause FromClause() throws ParseException :
3947{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003948 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003949 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
3950 extendCurrentScope();
3951}
3952{
3953 {
3954 FromTerm fromTerm = null;
3955 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003956 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003957 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
3958 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003959 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003960 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003961 }
3962}
3963
3964FromTerm FromTerm() throws ParseException :
3965{
3966 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003967 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003968 VariableExpr posVar = null;
3969 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
3970}
3971{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003972 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003973 (
3974 {JoinType joinType = JoinType.INNER; }
3975 (joinType = JoinType())?
3976 {
3977 AbstractBinaryCorrelateClause correlateClause = null;
3978 }
3979 (correlateClause = JoinClause(joinType)
Yingyi Bu6d999ed2016-07-13 11:59:06 -07003980 | correlateClause = UnnestClause(joinType)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003981 )
3982 {
3983 correlateClauses.add(correlateClause);
3984 }
3985 )*
3986 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003987 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003988 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003989 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003990 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
3991 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
3992 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003993 }
3994}
3995
3996JoinClause JoinClause(JoinType joinType) throws ParseException :
3997{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003998 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003999 Expression rightExpr = null;
4000 VariableExpr rightVar = null;
4001 VariableExpr posVar = null;
4002 Expression conditionExpr = null;
4003}
4004{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004005 <JOIN> { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())? <ON> conditionExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004006 {
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004007 if(rightVar==null){
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07004008 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004009 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004010 JoinClause joinClause = new JoinClause(joinType, rightExpr, rightVar, posVar, conditionExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004011 return addSourceLocation(joinClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004012 }
4013}
4014
Yingyi Bu391f09e2015-10-29 13:49:39 -07004015UnnestClause UnnestClause(JoinType joinType) throws ParseException :
4016{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004017 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004018 Expression rightExpr;
4019 VariableExpr rightVar;
4020 VariableExpr posVar = null;
4021}
4022{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004023 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable()) (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004024 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004025 if (rightVar == null) {
4026 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004027 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004028 UnnestClause unnestClause = new UnnestClause(joinType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004029 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004030 }
4031}
4032
Yingyi Bu391f09e2015-10-29 13:49:39 -07004033JoinType JoinType() throws ParseException :
4034{
4035 JoinType joinType = JoinType.INNER;
4036}
4037{
4038 (<INNER>|<LEFT> (<OUTER>)? {joinType = JoinType.LEFTOUTER; })
4039 {
4040 return joinType;
4041 }
4042}
4043
4044List<LetClause> LetClause() throws ParseException:
4045{
4046 List<LetClause> letList = new ArrayList<LetClause>();
4047 LetClause letClause;
4048}
4049{
4050 (
4051 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
4052 |
4053 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
4054 )
4055 {
4056 return letList;
4057 }
4058}
4059
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004060WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004061{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004062 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004063 Expression whereExpr;
4064}
4065{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004066 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004067 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004068 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004069 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004070 }
4071}
4072
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004073OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004074{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004075 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004076 OrderbyClause oc = new OrderbyClause();
4077 Expression orderbyExpr;
4078 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004079 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004080 int numOfOrderby = 0;
4081}
4082{
4083 <ORDER>
4084 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004085 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004086 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
4087 if (hintToken != null) {
4088 switch (hintToken.hint) {
4089 case INMEMORY_HINT:
4090 String[] splits = hintToken.hintParams.split("\\s+");
4091 int numFrames = Integer.parseInt(splits[0]);
4092 int numTuples = Integer.parseInt(splits[1]);
4093 oc.setNumFrames(numFrames);
4094 oc.setNumTuples(numTuples);
4095 break;
4096 case RANGE_HINT:
4097 try {
4098 oc.setRangeMap(RangeMapBuilder.parseHint(parseExpression(hintToken.hintParams)));
4099 } catch (CompilationException e) {
4100 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
4101 }
4102 break;
Ali Alsuliman80225e22018-10-15 14:17:07 -07004103 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004104 }
4105 }
4106 <BY> orderbyExpr = Expression()
4107 {
4108 orderbyList.add(orderbyExpr);
4109 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
4110 }
4111 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
4112 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
4113 {
4114 modifierList.add(modif);
4115 }
4116
4117 (LOOKAHEAD(2) <COMMA> orderbyExpr = Expression()
4118 {
4119 orderbyList.add(orderbyExpr);
4120 modif = OrderbyClause.OrderModifier.ASC;
4121 }
4122 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
4123 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
4124 {
4125 modifierList.add(modif);
4126 }
4127 )*
4128
4129 {
4130 oc.setModifierList(modifierList);
4131 oc.setOrderbyList(orderbyList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004132 return addSourceLocation(oc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004133 }
4134}
4135
4136GroupbyClause GroupbyClause()throws ParseException :
4137{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004138 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004139 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004140 List<List<GbyVariableExpressionPair>> gbyList = null;
4141 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004142 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07004143 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004144 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004145}
4146{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004147 {
4148 Scope newScope = extendCurrentScopeNoPush(true);
4149 // extendCurrentScope(true);
4150 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004151 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004152 {
4153 startToken = token;
4154 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
4155 if (hintToken != null) {
4156 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004157 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004158 }
4159 <BY> groupingElementList = GroupingElementList()
4160 (
4161 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004162 {
4163 groupVar = groupVarWithFieldList.first;
4164 groupFieldList = groupVarWithFieldList.second;
4165 }
Yingyi Buacc12a92016-03-26 17:25:05 -07004166 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004167 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004168 if (groupingSetsParser == null) {
4169 groupingSetsParser = new SqlppGroupingSetsParser();
4170 }
4171 SourceLocation sourceLoc = getSourceLocation(startToken);
4172 try {
4173 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
4174 } catch (CompilationException e) {
4175 throw new SqlppParseException(sourceLoc, e.getMessage());
4176 }
4177 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004178 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07004179 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07004180 gbc.setGroupVar(groupVar);
4181 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004182 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004183 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004184 }
4185}
4186
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004187List<GroupingElement> GroupingElementList() throws ParseException:
4188{
4189 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
4190 GroupingElement groupingElement = null;
4191}
4192{
4193 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
4194 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
4195 {
4196 return groupingElementList;
4197 }
4198}
4199
4200GroupingElement GroupingElement() throws ParseException:
4201{
4202 GroupingElement groupingElement = null;
4203 List<GroupingSet> groupingSets = null;
4204 List<GroupingElement> groupingElements = null;
4205}
4206{
4207 (
4208 LOOKAHEAD(2)
4209 <LEFTPAREN> <RIGHTPAREN>
4210 {
4211 groupingElement = GroupingSet.EMPTY;
4212 }
4213 |
4214 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
4215 <IDENTIFIER> { expectToken(ROLLUP); }
4216 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4217 {
4218 groupingElement = new RollupCube(groupingSets, false);
4219 }
4220 |
4221 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
4222 <IDENTIFIER> { expectToken(CUBE); }
4223 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4224 {
4225 groupingElement = new RollupCube(groupingSets, true);
4226 }
4227 |
4228 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
4229 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
4230 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
4231 {
4232 groupingElement = new GroupingSets(groupingElements);
4233 }
4234 |
4235 groupingElement = OrdinaryGroupingSet()
4236 )
4237 {
4238 return groupingElement;
4239 }
4240}
4241
4242GroupingSet OrdinaryGroupingSet() throws ParseException:
4243{
4244 GbyVariableExpressionPair gbyExprPair = null;
4245 List<GbyVariableExpressionPair> items = null;
4246}
4247{
4248 (
4249 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
4250 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
4251 )
4252 {
4253 return new GroupingSet(items);
4254 }
4255}
4256
4257List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
4258{
4259 GroupingSet groupingSet = null;
4260 List<GroupingSet> items = new ArrayList<GroupingSet>();
4261}
4262{
4263 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
4264 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
4265 {
4266 return items;
4267 }
4268}
4269
4270List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
4271{
4272 GbyVariableExpressionPair gbyExprPair = null;
4273 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
4274}
4275{
4276 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
4277 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
4278 {
4279 return items;
4280 }
4281}
4282
4283GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
4284{
4285 Expression expr = null;
4286 VariableExpr var = null;
4287}
4288{
4289 expr = Expression() ( (<AS>)? var = Variable() )?
4290 {
4291 return new GbyVariableExpressionPair(var, expr);
4292 }
4293}
4294
Yingyi Bu391f09e2015-10-29 13:49:39 -07004295HavingClause HavingClause() throws ParseException:
4296{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004297 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004298 Expression filterExpr = null;
4299}
4300{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004301 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004302 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004303 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004304 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004305 }
4306}
4307
4308LimitClause LimitClause() throws ParseException:
4309{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004310 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004311 LimitClause lc = new LimitClause();
4312 Expression expr;
4313 pushForbiddenScope(getCurrentScope());
4314}
4315{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004316 <LIMIT> { startToken = token; } expr = Expression() { lc.setLimitExpr(expr); }
4317 (<OFFSET> expr = Expression() { lc.setOffset(expr); })?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004318
4319 {
4320 popForbiddenScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004321 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004322 }
4323}
4324
4325QuantifiedExpression QuantifiedExpression()throws ParseException:
4326{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004327 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004328 QuantifiedExpression qc = new QuantifiedExpression();
4329 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
4330 Expression satisfiesExpr;
4331 VariableExpr var;
4332 Expression inExpr;
4333 QuantifiedPair pair;
4334}
4335{
4336 {
4337 createNewScope();
4338 }
4339
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004340 ( ((<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); })
4341 | (<EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); }))
Yingyi Bu391f09e2015-10-29 13:49:39 -07004342 var = Variable() <IN> inExpr = Expression()
4343 {
4344 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004345 quantifiedList.add(pair);
4346 }
4347 (
4348 <COMMA> var = Variable() <IN> inExpr = Expression()
4349 {
4350 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004351 quantifiedList.add(pair);
4352 }
4353 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07004354 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004355 {
4356 qc.setSatisfiesExpr(satisfiesExpr);
4357 qc.setQuantifiedList(quantifiedList);
4358 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004359 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004360 }
4361}
4362
4363LetClause LetElement() throws ParseException:
4364{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004365 VariableExpr varExp;
4366 Expression beExp;
4367 extendCurrentScope();
4368}
4369{
4370 varExp = Variable() <EQ> beExp = Expression()
4371 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004372 LetClause lc = new LetClause(varExp, beExp);
4373 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004374 return lc;
4375 }
4376}
4377
4378LetClause WithElement() throws ParseException:
4379{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004380 VariableExpr varExp;
4381 Expression beExp;
4382 extendCurrentScope();
4383}
4384{
4385 varExp = Variable() <AS> beExp = Expression()
4386 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004387 LetClause lc = new LetClause(varExp, beExp);
4388 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004389 return lc;
4390 }
4391}
4392
4393TOKEN_MGR_DECLS:
4394{
4395 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07004396 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004397 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004398
4399 public void pushState() {
4400 lexerStateStack.push( curLexState );
4401 }
4402
4403 public void popState(String token) {
4404 if (lexerStateStack.size() > 0) {
4405 SwitchTo( lexerStateStack.pop() );
4406 } else {
4407 int errorLine = input_stream.getEndLine();
4408 int errorColumn = input_stream.getEndColumn();
4409 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
4410 + "\" but state stack is empty.";
4411 throw new TokenMgrError(msg, -1);
4412 }
4413 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004414
4415 void CommonTokenAction(Token token) {
4416 Token hintToken = token.specialToken;
4417 if (hintToken != null) {
4418 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
4419 String text = hintToken.image.substring(1).trim();
4420 boolean hintFound = hintToken.parseHint(text);
4421 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
4422 }
4423 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004424}
4425
4426<DEFAULT,IN_DBL_BRACE>
4427TOKEN [IGNORE_CASE]:
4428{
Ian Maxon38fe9402020-01-29 19:27:40 -08004429 <ADAPTER: "adapter">
4430 | <ALL : "all">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004431 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07004432 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004433 | <APPLY : "apply">
4434 | <AS : "as">
4435 | <ASC : "asc">
4436 | <AT : "at">
4437 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004438 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004439 | <BTREE : "btree">
4440 | <BY : "by">
4441 | <CASE : "case">
4442 | <CLOSED : "closed">
4443 | <CREATE : "create">
4444 | <COMPACTION : "compaction">
4445 | <COMPACT : "compact">
4446 | <CONNECT : "connect">
4447 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07004448 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07004449 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07004450 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004451 | <DECLARE : "declare">
4452 | <DEFINITION : "definition">
4453 | <DELETE : "delete">
4454 | <DESC : "desc">
4455 | <DISCONNECT : "disconnect">
4456 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004457 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004458 | <DROP : "drop">
4459 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07004460 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004461 | <ELSE : "else">
4462 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07004463 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004464 | <EVERY : "every">
4465 | <EXCEPT : "except">
4466 | <EXISTS : "exists">
4467 | <EXTERNAL : "external">
4468 | <FEED : "feed">
4469 | <FILTER : "filter">
4470 | <FLATTEN : "flatten">
4471 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004472 | <FROM : "from">
4473 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08004474 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004475 | <FUNCTION : "function">
4476 | <GROUP : "group">
4477 | <HAVING : "having">
4478 | <HINTS : "hints">
4479 | <IF : "if">
4480 | <INTO : "into">
4481 | <IN : "in">
4482 | <INDEX : "index">
4483 | <INGESTION : "ingestion">
4484 | <INNER : "inner">
4485 | <INSERT : "insert">
4486 | <INTERNAL : "internal">
4487 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07004488 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004489 | <JOIN : "join">
4490 | <KEYWORD : "keyword">
4491 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07004492 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004493 | <LEFT : "left">
4494 | <LETTING : "letting">
4495 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004496 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004497 | <LIMIT : "limit">
4498 | <LOAD : "load">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004499 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004500 | <NODEGROUP : "nodegroup">
4501 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07004502 | <NOT : "not">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004503 | <OFFSET : "offset">
4504 | <ON : "on">
4505 | <OPEN : "open">
4506 | <OR : "or">
4507 | <ORDER : "order">
4508 | <OUTER : "outer">
4509 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004510 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004511 | <PATH : "path">
4512 | <POLICY : "policy">
4513 | <PRESORTED : "pre-sorted">
4514 | <PRIMARY : "primary">
4515 | <RAW : "raw">
4516 | <REFRESH : "refresh">
4517 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004518 | <RETURNING : "returning">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004519 | <RTREE : "rtree">
4520 | <RUN : "run">
4521 | <SATISFIES : "satisfies">
4522 | <SECONDARY : "secondary">
4523 | <SELECT : "select">
4524 | <SET : "set">
4525 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08004526 | <START : "start">
4527 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08004528 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03004529 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07004530 | <THEN : "then">
4531 | <TYPE : "type">
4532 | <TO : "to">
4533 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08004534 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004535 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07004536 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004537 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07004538 | <USE : "use">
4539 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004540 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08004541 | <VALUED : "valued">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004542 | <WHEN : "when">
4543 | <WHERE : "where">
4544 | <WITH : "with">
4545 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004546}
4547
4548<DEFAULT,IN_DBL_BRACE>
4549TOKEN :
4550{
4551 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07004552 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004553 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004554 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004555 | <MUL : "*">
4556 | <PLUS : "+">
4557
4558 | <LEFTPAREN : "(">
4559 | <RIGHTPAREN : ")">
4560 | <LEFTBRACKET : "[">
4561 | <RIGHTBRACKET : "]">
4562
4563 | <ATT : "@">
4564 | <COLON : ":">
4565 | <COMMA : ",">
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004566 | <DOLLAR: "$">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004567 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004568 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004569 | <QUES : "?">
4570 | <SEMICOLON : ";">
4571 | <SHARP : "#">
4572
4573 | <LT : "<">
4574 | <GT : ">">
4575 | <LE : "<=">
4576 | <GE : ">=">
4577 | <EQ : "=">
4578 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07004579 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004580 | <SIMILAR : "~=">
4581}
4582
4583<DEFAULT,IN_DBL_BRACE>
4584TOKEN :
4585{
4586 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
4587}
4588
4589<DEFAULT>
4590TOKEN :
4591{
4592 <RIGHTBRACE : "}"> { popState("}"); }
4593}
4594
4595<DEFAULT,IN_DBL_BRACE>
4596TOKEN :
4597{
4598 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
4599}
4600
4601<IN_DBL_BRACE>
4602TOKEN :
4603{
4604 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
4605}
4606
4607<DEFAULT,IN_DBL_BRACE>
4608TOKEN :
4609{
4610 <INTEGER_LITERAL : (<DIGIT>)+ >
4611}
4612
4613<DEFAULT,IN_DBL_BRACE>
Yingyi Bue311a632016-06-07 18:23:16 -07004614TOKEN [IGNORE_CASE]:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004615{
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004616 <MISSING : "missing">
4617 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004618 | <TRUE : "true">
4619 | <FALSE : "false">
4620}
4621
4622<DEFAULT,IN_DBL_BRACE>
4623TOKEN :
4624{
4625 <#DIGIT : ["0" - "9"]>
4626}
4627
4628<DEFAULT,IN_DBL_BRACE>
4629TOKEN:
4630{
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07004631 < DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
4632 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
4633 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004634 >
Yingyi Bue4d919e2016-10-30 10:47:03 -07004635 | < FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
4636 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
4637 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004638 >
4639 | <DIGITS : (<DIGIT>)+ >
4640}
4641
4642<DEFAULT,IN_DBL_BRACE>
4643TOKEN :
4644{
4645 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08004646 | <#IDENTIFIER_SPECIALCHARS_START : ["_"]>
4647 | <#IDENTIFIER_SPECIALCHARS_REST : ["$"]>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004648}
4649
4650<DEFAULT,IN_DBL_BRACE>
4651TOKEN :
4652{
4653 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07004654 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004655 <EscapeQuot>
4656 | <EscapeBslash>
4657 | <EscapeSlash>
4658 | <EscapeBspace>
4659 | <EscapeFormf>
4660 | <EscapeNl>
4661 | <EscapeCr>
4662 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004663 | ~["`","\\"])* "`">
4664 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004665 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004666 | <EscapeBslash>
4667 | <EscapeSlash>
4668 | <EscapeBspace>
4669 | <EscapeFormf>
4670 | <EscapeNl>
4671 | <EscapeCr>
4672 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004673 | ~["\"","\\"])* "\"")
4674 | ("\'"(
4675 <EscapeApos>
4676 | <EscapeBslash>
4677 | <EscapeSlash>
4678 | <EscapeBspace>
4679 | <EscapeFormf>
4680 | <EscapeNl>
4681 | <EscapeCr>
4682 | <EscapeTab>
4683 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004684 | < #EscapeQuot: "\\\"" >
4685 | < #EscapeApos: "\\\'" >
4686 | < #EscapeBslash: "\\\\" >
4687 | < #EscapeSlash: "\\/" >
4688 | < #EscapeBspace: "\\b" >
4689 | < #EscapeFormf: "\\f" >
4690 | < #EscapeNl: "\\n" >
4691 | < #EscapeCr: "\\r" >
4692 | < #EscapeTab: "\\t" >
4693}
4694
4695<DEFAULT,IN_DBL_BRACE>
4696TOKEN :
4697{
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08004698 <IDENTIFIER : ( <LETTER> | <IDENTIFIER_SPECIALCHARS_START> )
4699 ( <LETTER> | <DIGIT> | <IDENTIFIER_SPECIALCHARS_START> | <IDENTIFIER_SPECIALCHARS_REST> )*>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004700}
4701
4702<DEFAULT,IN_DBL_BRACE>
4703SKIP:
4704{
4705 " "
4706 | "\t"
4707 | "\r"
4708 | "\n"
4709}
4710
4711<DEFAULT,IN_DBL_BRACE>
4712SKIP:
4713{
4714 <"//" (~["\n"])* "\n">
4715}
4716
4717<DEFAULT,IN_DBL_BRACE>
4718SKIP:
4719{
4720 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4721}
4722
4723<DEFAULT,IN_DBL_BRACE>
4724SKIP:
4725{
Yingyi Bu93846a72016-09-13 16:30:39 -07004726 <"--" (~["\n"])* "\n">
4727}
4728
4729
4730<DEFAULT,IN_DBL_BRACE>
4731SKIP:
4732{
4733 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4734}
4735
4736<DEFAULT,IN_DBL_BRACE>
4737SKIP:
4738{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004739 <"/*"> { pushState(); } : INSIDE_COMMENT
4740}
4741
4742<INSIDE_COMMENT>
4743SPECIAL_TOKEN:
4744{
4745 <"+"(" ")*(~["*"])*>
4746}
4747
4748<INSIDE_COMMENT>
4749SKIP:
4750{
4751 <"/*"> { pushState(); }
4752}
4753
4754<INSIDE_COMMENT>
4755SKIP:
4756{
4757 <"*/"> { popState("*/"); }
4758 | <~[]>
4759}