blob: eee3f8d2abec8366761a8ccf97b21d07cccae970 [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;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080059import org.apache.asterix.common.annotations.IndexedNLJoinExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070060import org.apache.asterix.common.annotations.InsertRandIntDataGen;
61import org.apache.asterix.common.annotations.ListDataGen;
62import org.apache.asterix.common.annotations.ListValFileDataGen;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -060063import org.apache.asterix.common.annotations.RangeAnnotation;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080064import org.apache.asterix.common.annotations.SecondaryIndexSearchPreferenceAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070065import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
66import org.apache.asterix.common.annotations.TypeDataGen;
67import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
68import org.apache.asterix.common.config.DatasetConfig.DatasetType;
69import org.apache.asterix.common.config.DatasetConfig.IndexType;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080070import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070071import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070072import org.apache.asterix.common.exceptions.WarningCollector;
73import org.apache.asterix.common.exceptions.WarningUtil;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080074import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070075import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin54c06012019-11-13 15:54:20 -080076import org.apache.asterix.common.metadata.DataverseName;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080077import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070078import org.apache.asterix.lang.common.base.AbstractLangExpression;
79import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070080import org.apache.asterix.lang.common.base.Expression;
81import org.apache.asterix.lang.common.base.Literal;
82import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070083import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070084import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070085import org.apache.asterix.lang.common.base.Statement;
86import org.apache.asterix.lang.common.clause.GroupbyClause;
87import org.apache.asterix.lang.common.clause.LetClause;
88import org.apache.asterix.lang.common.clause.LimitClause;
89import org.apache.asterix.lang.common.clause.OrderbyClause;
90import org.apache.asterix.lang.common.clause.UpdateClause;
91import org.apache.asterix.lang.common.clause.WhereClause;
92import org.apache.asterix.lang.common.context.RootScopeFactory;
93import org.apache.asterix.lang.common.context.Scope;
94import org.apache.asterix.lang.common.expression.AbstractAccessor;
95import org.apache.asterix.lang.common.expression.CallExpr;
96import org.apache.asterix.lang.common.expression.FieldAccessor;
97import org.apache.asterix.lang.common.expression.FieldBinding;
98import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
99import org.apache.asterix.lang.common.expression.IfExpr;
100import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700101import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700102import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +0300103import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700104import org.apache.asterix.lang.common.expression.LiteralExpr;
105import org.apache.asterix.lang.common.expression.OperatorExpr;
106import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
107import org.apache.asterix.lang.common.expression.QuantifiedExpression;
108import org.apache.asterix.lang.common.expression.RecordConstructor;
109import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
110import org.apache.asterix.lang.common.expression.TypeExpression;
111import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
112import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700113import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
114import org.apache.asterix.lang.common.expression.VariableExpr;
115import org.apache.asterix.lang.common.literal.DoubleLiteral;
116import org.apache.asterix.lang.common.literal.FalseLiteral;
117import org.apache.asterix.lang.common.literal.FloatLiteral;
118import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700119import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700120import org.apache.asterix.lang.common.literal.NullLiteral;
121import org.apache.asterix.lang.common.literal.StringLiteral;
122import org.apache.asterix.lang.common.literal.TrueLiteral;
123import org.apache.asterix.lang.common.parser.ScopeChecker;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -0700124import org.apache.asterix.lang.common.statement.AdapterDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700125import org.apache.asterix.lang.common.statement.CompactStatement;
126import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800127import org.apache.asterix.lang.common.statement.StartFeedStatement;
128import org.apache.asterix.lang.common.statement.StopFeedStatement;
Ian Maxon38fe9402020-01-29 19:27:40 -0800129import org.apache.asterix.lang.common.statement.CreateAdapterStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700130import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
131import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
132import org.apache.asterix.lang.common.statement.CreateFeedStatement;
133import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
134import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800135import org.apache.asterix.lang.common.statement.CreateSynonymStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700136import org.apache.asterix.lang.common.statement.DatasetDecl;
137import org.apache.asterix.lang.common.statement.DataverseDecl;
138import org.apache.asterix.lang.common.statement.DataverseDropStatement;
139import org.apache.asterix.lang.common.statement.DeleteStatement;
140import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700141import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700142import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
143import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300144import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700145import org.apache.asterix.lang.common.statement.FunctionDecl;
146import org.apache.asterix.lang.common.statement.FunctionDropStatement;
147import org.apache.asterix.lang.common.statement.IndexDropStatement;
148import org.apache.asterix.lang.common.statement.InsertStatement;
149import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
150import org.apache.asterix.lang.common.statement.LoadStatement;
151import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
152import org.apache.asterix.lang.common.statement.NodegroupDecl;
153import org.apache.asterix.lang.common.statement.Query;
154import org.apache.asterix.lang.common.statement.RefreshExternalDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700155import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800156import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700157import org.apache.asterix.lang.common.statement.TypeDecl;
158import org.apache.asterix.lang.common.statement.TypeDropStatement;
159import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800160import org.apache.asterix.lang.common.statement.UpsertStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700161import org.apache.asterix.lang.common.statement.WriteStatement;
162import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800163import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700164import org.apache.asterix.lang.common.struct.QuantifiedPair;
165import org.apache.asterix.lang.common.struct.VarIdentifier;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700166import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700167import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
168import org.apache.asterix.lang.sqlpp.clause.FromClause;
169import org.apache.asterix.lang.sqlpp.clause.FromTerm;
170import org.apache.asterix.lang.sqlpp.clause.HavingClause;
171import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700172import org.apache.asterix.lang.sqlpp.clause.Projection;
173import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
174import org.apache.asterix.lang.sqlpp.clause.SelectClause;
175import org.apache.asterix.lang.sqlpp.clause.SelectElement;
176import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
177import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
178import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800179import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700180import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700181import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700182import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700183import org.apache.asterix.lang.sqlpp.optype.JoinType;
184import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -0700185import org.apache.asterix.lang.sqlpp.optype.UnnestType;
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700186import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser;
187import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingElement;
188import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSet;
189import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSets;
190import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.RollupCube;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700191import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700192import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
193import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700194import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700195import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700196import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Ian Maxon38fe9402020-01-29 19:27:40 -0800197import org.apache.asterix.external.dataset.adapter.AdapterIdentifier;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800198import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700199import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700200import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700201import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700202import org.apache.hyracks.algebricks.common.utils.Pair;
203import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800204import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700205import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700206import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700207import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800208import org.apache.hyracks.api.exceptions.IWarningCollector;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700209import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700210import org.apache.hyracks.api.exceptions.Warning;
Dmitry Lychagin70bf47a2020-10-22 13:25:57 -0700211import org.apache.hyracks.dataflow.common.data.partition.range.RangeMap;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800212import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700213import org.apache.hyracks.util.StringUtil;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700214
Yingyi Bucaea8f02015-11-16 15:12:15 -0800215class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700216
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800217 // tokens parsed as identifiers
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700218 private static final String CUBE = "CUBE";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800219 private static final String CURRENT = "CURRENT";
220 private static final String EXCLUDE = "EXCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700221 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800222 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700223 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800224 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700225 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700226 private static final String LAST = "LAST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800227 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700228 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800229 private static final String OTHERS = "OTHERS";
230 private static final String PARTITION = "PARTITION";
231 private static final String PRECEDING = "PRECEDING";
232 private static final String RANGE = "RANGE";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700233 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700234 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800235 private static final String ROW = "ROW";
236 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700237 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800238 private static final String TIES = "TIES";
239 private static final String UNBOUNDED = "UNBOUNDED";
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700240 private static final String REPLACE = "REPLACE";
Ian Maxon38fe9402020-01-29 19:27:40 -0800241 private static final String RETURNS = "RETURNS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800242
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700243 private static final String INT_TYPE_NAME = "int";
Dmitry Lychagin541652082020-11-09 14:04:53 -0800244 private static final String UDF_VARARGS_PARAM_NAME = "args"; // Note: this value is stored in the function metadata
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700245
Till Westmann7199a562016-09-17 16:07:32 -0700246 // error configuration
247 protected static final boolean REPORT_EXPECTED_TOKENS = false;
248
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700249 private int externalVarCounter;
250
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800251 private DataverseName defaultDataverse;
252
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700253 private SqlppGroupingSetsParser groupingSetsParser;
254
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700255 private final WarningCollector warningCollector = new WarningCollector();
256
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700257 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
258
Yingyi Bu391f09e2015-10-29 13:49:39 -0700259 private static class IndexParams {
260 public IndexType type;
261 public int gramLength;
262
263 public IndexParams(IndexType type, int gramLength) {
264 this.type = type;
265 this.gramLength = gramLength;
266 }
267 };
268
269 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800270 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700271 public String library;
272 public String function;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -0600273 public Token hintToken;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700274 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700275 }
276
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700277 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
278 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
279 switch (hintToken.hint) {
280 case VAL_FILE_HINT:
281 File[] valFiles = new File[splits.length];
282 for (int k=0; k<splits.length; k++) {
283 valFiles[k] = new File(splits[k]);
284 }
285 return new FieldValFileDataGen(valFiles);
286 case VAL_FILE_SAME_INDEX_HINT:
287 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
288 case LIST_VAL_FILE_HINT:
289 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
290 case LIST_HINT:
291 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
292 case INTERVAL_HINT:
293 FieldIntervalDataGen.ValueType vt;
294 switch (splits[0]) {
295 case "int":
296 vt = FieldIntervalDataGen.ValueType.INT;
297 break;
298 case "long":
299 vt = FieldIntervalDataGen.ValueType.LONG;
300 break;
301 case "float":
302 vt = FieldIntervalDataGen.ValueType.FLOAT;
303 break;
304 case "double":
305 vt = FieldIntervalDataGen.ValueType.DOUBLE;
306 break;
307 default:
308 throw new SqlppParseException(getSourceLocation(hintToken),
309 "Unknown type for interval data gen: " + splits[0]);
310 }
311 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
312 case INSERT_RAND_INT_HINT:
313 return new InsertRandIntDataGen(splits[0], splits[1]);
314 case DATE_BETWEEN_YEARS_HINT:
315 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
316 case DATETIME_BETWEEN_YEARS_HINT:
317 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
318 case DATETIME_ADD_RAND_HOURS_HINT:
319 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
320 case AUTO_HINT:
321 return new AutoDataGen(splits[0]);
322 default:
323 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700324 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700325 }
326
Till Westmann7199a562016-09-17 16:07:32 -0700327 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700328 this(new StringReader(s));
329 super.setInput(s);
330 }
331
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800332 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700333 File file = new File(args[0]);
334 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
335 SQLPPParser parser = new SQLPPParser(fis);
336 List<Statement> st = parser.parse();
337 //st.accept(new SQLPPPrintVisitor(), 0);
338 }
339
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800340 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800341 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700342 return parseImpl(new ParseFunction<List<Statement>>() {
343 @Override
344 public List<Statement> parse() throws ParseException {
345 return SQLPPParser.this.Statement();
346 }
347 });
348 }
349
Dmitry Lychagin62f0beb2020-09-08 11:44:19 -0700350 @Override
351 public Expression parseExpression() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700352 return parseImpl(new ParseFunction<Expression>() {
353 @Override
354 public Expression parse() throws ParseException {
355 return SQLPPParser.this.Expression();
356 }
357 });
358 }
359
360 private static Expression parseExpression(String text) throws CompilationException {
361 return new SQLPPParser(text).parseExpression();
362 }
363
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800364 @Override
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700365 public List<String> parseMultipartIdentifier() throws CompilationException {
366 return parseImpl(new ParseFunction<List<String>>() {
367 @Override
368 public List<String> parse() throws ParseException {
369 return SQLPPParser.this.MultipartIdentifier();
370 }
371 });
372 }
373
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800374 private List<String> parseParenthesizedIdentifierList() throws CompilationException {
375 return parseImpl(new ParseFunction<List<String>>() {
376 @Override
377 public List<String> parse() throws ParseException {
378 return SQLPPParser.this.ParenthesizedIdentifierList();
379 }
380 });
381 }
382
383 private static List<String> parseParenthesizedIdentifierList(String text) throws CompilationException {
384 return new SQLPPParser(text).parseParenthesizedIdentifierList();
385 }
386
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700387 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800388 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800389 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800390 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800391 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800392 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800393 DataverseName dataverse = defaultDataverse;
394 defaultDataverse = signature.getDataverseName();
395 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800396 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
397 for (String paramName : paramNames) {
398 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
399 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800400 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
401 removeCurrentScope();
402 defaultDataverse = dataverse;
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800403 return new FunctionDecl(signature, paramVars, functionBodyExpr);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800404 }
405 });
406 }
407
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700408 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700409 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700410 hintCollector.clear();
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800411 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700412 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700413 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700414 } catch (SqlppParseException e) {
415 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(), LogRedactionUtil.userData(getMessage(e)));
416 } catch (ParseException e) {
417 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700418 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700419 // this is here as the JavaCharStream that's below the lexer sometimes throws Errors that are not handled
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -0700420 // by the generated lexer or parser (e.g it does this for invalid backslash u + 4 hex digits escapes)
Till Westmann60f89982017-08-11 18:14:20 -0700421 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700422 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700423 } finally {
424 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700425 }
426 }
Till Westmann7199a562016-09-17 16:07:32 -0700427
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700428 @FunctionalInterface
429 private interface ParseFunction<T> {
430 T parse() throws ParseException;
431 }
432
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700433 @Override
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800434 public void getWarnings(IWarningCollector outWarningCollector) {
435 warningCollector.getWarnings(outWarningCollector);
436 }
437
438 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700439 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
440 warningCollector.getWarnings(outWarnings, maxWarnings);
441 }
442
443 @Override
444 public long getTotalWarningsCount() {
445 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700446 }
447
Till Westmann7199a562016-09-17 16:07:32 -0700448 protected String getMessage(ParseException pe) {
449 Token currentToken = pe.currentToken;
450 if (currentToken == null) {
451 return pe.getMessage();
452 }
453 int[][] expectedTokenSequences = pe.expectedTokenSequences;
454 String[] tokenImage = pe.tokenImage;
455 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
456 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
457 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
458 Token tok = currentToken.next;
459 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700460 StringBuilder message = new StringBuilder(128);
461 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700462 for (int i = 0; i < maxSize; i++) {
463 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700464 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700465 }
466 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700467 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700468 break;
469 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700470 final String fixedTokenImage = tokenImage[tok.kind];
471 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700472 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700473 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700474 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700475 tok = tok.next;
476 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700477 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700478 if (REPORT_EXPECTED_TOKENS) {
479 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700480 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700481 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700482 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700483 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700484 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700485 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700486 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700487 }
488
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700489 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700490 return
491 token == null ? null :
492 token.sourceLocation != null ? token.sourceLocation :
493 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700494 }
495
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700496 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
497 expr.setSourceLocation(getSourceLocation(token));
498 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700499 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800500
501 private boolean isToken(String image) {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700502 return isToken(token, image);
503 }
504
505 private static boolean isToken(Token token, String image) {
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800506 return token.image.equalsIgnoreCase(image);
507 }
508
509 private void expectToken(String image) throws SqlppParseException {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700510 expectToken(token, image);
511 }
512
513 private static void expectToken(Token token, String image) throws SqlppParseException {
514 if (!isToken(token, image)) {
515 throw createUnexpectedTokenError(token);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800516 }
517 }
518
519 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700520 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700521 }
522
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700523 private static SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700524 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800525 }
526
527 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700528 return createUnexpectedTokenError(token, expected);
529 }
530
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700531 private static SqlppParseException createUnexpectedTokenError(Token t, String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700532 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800533 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700534 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800535 }
536
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700537 private boolean laToken(int idx, int kind) {
538 Token t = getToken(idx);
539 return t.kind == kind;
540 }
541
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800542 private boolean laToken(int idx, int kind, String image) {
543 Token t = getToken(idx);
544 return t.kind == kind && t.image.equalsIgnoreCase(image);
545 }
546
547 private boolean laIdentifier(int idx, String image) {
548 return laToken(idx, IDENTIFIER, image);
549 }
550
551 private boolean laIdentifier(String image) {
552 return laIdentifier(1, image);
553 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700554
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700555 private Token fetchHint(Token token, SqlppHint... expectedHints) {
556 Token hintToken = token.specialToken;
557 if (hintToken == null) {
558 return null;
559 }
560 SourceLocation sourceLoc = getSourceLocation(hintToken);
561 hintCollector.remove(sourceLoc);
562 if (hintToken.hint == null) {
563 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
564 return null;
565 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
566 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
567 return null;
568 } else {
569 return hintToken;
570 }
571 }
572
573 private void reportUnclaimedHints() {
574 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
575 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
576 }
577 }
578
579 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
580 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
581 }
582
583 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700584 if (warningCollector.shouldWarn()) {
585 warningCollector.warn(WarningUtil.forAsterix(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
586 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700587 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700588
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800589 private IExpressionAnnotation parseExpressionAnnotation(Token hintToken) {
590 // placeholder for the annotation that should be returned if this hint's parameters cannot be parsed
591 IExpressionAnnotation onParseErrorReturn = null;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800592 try {
593 switch (hintToken.hint) {
594 case HASH_BROADCAST_JOIN_HINT:
595 return new BroadcastExpressionAnnotation(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
596 case INDEXED_NESTED_LOOP_JOIN_HINT:
597 if (hintToken.hintParams == null) {
598 return IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
599 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800600 // if parameter parsing fails then return hint annotation without parameters
601 onParseErrorReturn = IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800602 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
603 return IndexedNLJoinExpressionAnnotation.newInstance(indexNames);
604 }
605 case RANGE_HINT:
606 Expression rangeExpr = parseExpression(hintToken.hintParams);
607 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
608 return new RangeAnnotation(rangeMap);
609 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
610 if (hintToken.hintParams == null) {
611 return SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE_ANY_INDEX;
612 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800613 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800614 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
615 return SkipSecondaryIndexSearchExpressionAnnotation.newInstance(indexNames);
616 }
617 case USE_SECONDARY_INDEX_SEARCH_HINT:
618 if (hintToken.hintParams == null) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800619 throw new SqlppParseException(getSourceLocation(hintToken), "Expected index name(s)");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800620 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800621 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800622 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
623 return SecondaryIndexSearchPreferenceAnnotation.newInstance(indexNames);
624 }
625 default:
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800626 throw new SqlppParseException(getSourceLocation(hintToken), "Unexpected hint");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800627 }
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800628 } catch (SqlppParseException e) {
629 if (warningCollector.shouldWarn()) {
630 warningCollector.warn(WarningUtil.forAsterix(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
631 hintToken.hint.toString(), e.getMessage()));
632 }
633 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800634 } catch (CompilationException e) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800635 if (warningCollector.shouldWarn()) {
636 warningCollector.warn(WarningUtil.forAsterix(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
637 hintToken.hint.toString(), e.getMessage()));
638 }
639 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800640 }
641 }
642
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700643 private void ensureNoTypeDeclsInFunction(String fnName, List<Pair<VarIdentifier, TypeExpression>> paramList,
644 TypeExpression returnType, Token startToken) throws SqlppParseException {
645 for (Pair<VarIdentifier, TypeExpression> p : paramList) {
646 if (p.second != null) {
647 String paramName = SqlppVariableUtil.toUserDefinedName(p.first.getValue());
648 throw new SqlppParseException(getSourceLocation(startToken),
649 "Unexpected type declaration for parameter " + paramName + " in function " + fnName);
650 }
651 }
652 if (returnType != null) {
653 throw new SqlppParseException(getSourceLocation(startToken),
654 "Unexpected return type declaration for function " + fnName);
655 }
656 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -0700657
658 private void ensureIntegerLiteral(LiteralExpr expr, String errorPrefix) throws SqlppParseException {
659 Literal lit = expr.getValue();
660 if (lit.getLiteralType() != Literal.Type.INTEGER && lit.getLiteralType() != Literal.Type.LONG) {
661 throw new SqlppParseException(expr.getSourceLocation(), errorPrefix + " should be an INTEGER");
662 }
663 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700664}
665
666PARSER_END(SQLPPParser)
667
Yingyi Bu391f09e2015-10-29 13:49:39 -0700668List<Statement> Statement() throws ParseException:
669{
670 scopeStack.push(RootScopeFactory.createRootScope(this));
671 List<Statement> decls = new ArrayList<Statement>();
672 Statement stmt = null;
673}
674{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300675 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700676 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300677 {
678 decls.add(stmt);
679 }
680 )?
681 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700682 )*
683 <EOF>
684 {
685 return decls;
686 }
687}
688
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700689Statement ExplainStatement() throws ParseException:
690{
691 Statement stmt = null;
692 Token explainToken = null;
693}
694{
695 ( <EXPLAIN> { explainToken = token; } )?
696 stmt = SingleStatement()
697 {
698 if (explainToken != null) {
699 if (stmt.getKind() == Statement.Kind.QUERY) {
700 ((Query)stmt).setExplain(true);
701 } else {
702 throw new SqlppParseException(getSourceLocation(explainToken),
703 "EXPLAIN is not supported for this kind of statement");
704 }
705 }
706 return stmt;
707 }
708}
709
Yingyi Bu391f09e2015-10-29 13:49:39 -0700710Statement SingleStatement() throws ParseException:
711{
712 Statement stmt = null;
713}
714{
715 (
716 stmt = DataverseDeclaration()
717 | stmt = FunctionDeclaration()
718 | stmt = CreateStatement()
719 | stmt = LoadStatement()
720 | stmt = DropStatement()
721 | stmt = WriteStatement()
722 | stmt = SetStatement()
723 | stmt = InsertStatement()
724 | stmt = DeleteStatement()
725 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800726 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700727 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700728 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700729 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700730 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700731 )
732 {
733 return stmt;
734 }
735}
736
737DataverseDecl DataverseDeclaration() throws ParseException:
738{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700739 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800740 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700741}
742{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800743 <USE> { startToken = token; } dvName = MultipartIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700744 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800745 defaultDataverse = DataverseName.create(dvName);
746 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700747 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700748 }
749}
750
751Statement CreateStatement() throws ParseException:
752{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700753 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700754 Statement stmt = null;
755}
756{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700757 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700758 (
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700759 stmt = CreateOrReplaceStatement(startToken)
760 | stmt = CreateTypeStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800761 | stmt = CreateNodegroupStatement(startToken)
762 | stmt = CreateDatasetStatement(startToken)
763 | stmt = CreateIndexStatement(startToken)
764 | stmt = CreateDataverseStatement(startToken)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700765 | stmt = CreateFunctionStatement(startToken, false)
Ian Maxon38fe9402020-01-29 19:27:40 -0800766 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800767 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800768 | stmt = CreateFeedStatement(startToken)
769 | stmt = CreateFeedPolicyStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700770 )
771 {
772 return stmt;
773 }
774}
775
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700776Statement CreateOrReplaceStatement(Token startStmtToken) throws ParseException:
777{
778 Statement stmt = null;
779 Token replaceToken = null;
780}
781{
782 <OR> <IDENTIFIER> { replaceToken = token; }
783 (
784 stmt = CreateFunctionStatement(startStmtToken, true)
785 )
786 {
787 // check expected token here to make the grammar extension plugin happy
788 expectToken(replaceToken, REPLACE);
789 return stmt;
790 }
791}
792
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800793TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
794{
795 TypeDecl stmt = null;
796}
797{
798 <TYPE> stmt = TypeSpecification(startStmtToken)
799 {
800 return stmt;
801 }
802}
803
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700804TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700805{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800806 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700807 boolean ifNotExists = false;
808 TypeExpression typeExpr = null;
809}
810{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800811 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700812 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800813 {
814 boolean dgen = false;
815 long numValues = -1;
816 String filename = null;
817 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
818 if (hintToken != null) {
819 String hintParams = hintToken.hintParams;
820 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
821 if (splits == null || splits.length != 2) {
822 throw new SqlppParseException(getSourceLocation(hintToken),
823 "Expecting /*+ dgen <filename> <numberOfItems> */");
824 }
825 dgen = true;
826 filename = splits[0];
827 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700828 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800829 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
830 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
831 return addSourceLocation(stmt, startStmtToken);
832 }
833}
834
835NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
836{
837 NodegroupDecl stmt = null;
838}
839{
840 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
841 {
842 return stmt;
843 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700844}
845
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700846NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700847{
848 String name = null;
849 String tmp = null;
850 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800851 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700852}
853{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800854 name = Identifier() ifNotExists = IfNotExists()
855 <ON> tmp = Identifier()
856 {
857 ncNames.add(new Identifier(tmp));
858 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700859 ( <COMMA> tmp = Identifier()
860 {
861 ncNames.add(new Identifier(tmp));
862 }
863 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800864 {
865 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
866 return addSourceLocation(stmt, startStmtToken);
867 }
868}
869
870void Dataset() throws ParseException:
871{
872}
873{
874 (<DATASET>|<COLLECTION>)
875}
876
877DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
878{
879 DatasetDecl stmt = null;
880}
881{
882 (
883 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
884 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
885 )
886 {
887 return stmt;
888 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700889}
890
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700891DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700892{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800893 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700894 boolean ifNotExists = false;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700895 TypeExpression typeExpr = null;
896 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800897 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700898 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700899 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700900 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800901 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -0800902 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700903}
904{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800905 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700906 typeExpr = DatasetTypeSpecification()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700907 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800908 { String name; }
909 <WITH>
910 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700911 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800912 if (!name.equalsIgnoreCase("meta")){
913 throw new SqlppParseException(getSourceLocation(startStmtToken),
914 "We can only support one additional associated field called \"meta\".");
915 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700916 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700917 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800918 )?
919 ifNotExists = IfNotExists()
920 primaryKeyFields = PrimaryKey()
921 (<AUTOGENERATED> { autogenerated = true; } )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800922 ( <HINTS> hints = Properties() )?
923 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
924 ( <WITH> withRecord = RecordConstructor() )?
925 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800926 try {
Xikui Wangca7927f2020-09-03 11:27:57 -0700927 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
928 filterField == null? null : filterField.first, filterField == null? null : filterField.second);
Murtadha Hubail353e95f2020-08-24 22:18:04 +0300929 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr, hints,
930 DatasetType.INTERNAL, idd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800931 return addSourceLocation(stmt, startStmtToken);
932 } catch (CompilationException e) {
933 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
934 }
935 }
936}
937
938DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
939{
940 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700941 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800942 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800943 String adapterName = null;
944 Map<String,String> properties = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800945 Map<String,String> hints = new HashMap<String,String>();
946 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800947 RecordConstructor withRecord = null;
948}
949{
950 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700951 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800952 ifNotExists = IfNotExists()
953 <USING> adapterName = AdapterName() properties = Configuration()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800954 ( <HINTS> hints = Properties() )?
955 ( <WITH> withRecord = RecordConstructor() )?
956 {
957 ExternalDetailsDecl edd = new ExternalDetailsDecl();
958 edd.setAdapter(adapterName);
959 edd.setProperties(properties);
960 try {
Murtadha Hubail353e95f2020-08-24 22:18:04 +0300961 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null, hints, DatasetType.EXTERNAL,
962 edd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800963 return addSourceLocation(stmt, startStmtToken);
964 } catch (CompilationException e) {
965 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
966 }
967 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700968}
969
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700970TypeExpression DatasetTypeSpecification() throws ParseException:
971{
972 TypeExpression typeExpr = null;
973}
974{
975 (
976 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
977 | typeExpr = DatasetReferenceTypeSpecification()
978 )
979 {
980 return typeExpr;
981 }
982}
983
984TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
985{
986 TypeExpression typeExpr = null;
987}
988{
989 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
990 {
991 return typeExpr;
992 }
993}
994
995TypeExpression DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
996{
997 RecordTypeDefinition recordTypeDef = null;
998 RecordTypeDefinition.RecordKind recordKind = null;
Dmitry Lychagin6c239e72020-06-19 19:46:27 -0700999 Token startToken = null, recordKindToken = null;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001000}
1001{
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001002 <LEFTPAREN> { startToken = token; } recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001003 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
1004 {
1005 if (recordKind == null) {
1006 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
1007 } else if (!allowRecordKindModifier) {
1008 throw createUnexpectedTokenError(recordKindToken);
1009 }
1010 recordTypeDef.setRecordKind(recordKind);
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001011 return addSourceLocation(recordTypeDef, startToken);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001012 }
1013}
1014
1015RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
1016{
1017 RecordTypeDefinition recType = new RecordTypeDefinition();
1018}
1019{
1020 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
1021 {
1022 return recType;
1023 }
1024}
1025
1026void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
1027{
1028 String fieldName;
1029 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001030 boolean nullable = true, missable = true;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001031}
1032{
1033 fieldName = Identifier()
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001034 type = TypeReference() ( <NOT> <UNKNOWN> { nullable = false; missable = false; } )?
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001035 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001036 recType.addField(fieldName, type, nullable, missable);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001037 }
1038}
1039
Yingyi Bu391f09e2015-10-29 13:49:39 -07001040RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
1041{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001042 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001043 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001044 String datasetName = null;
1045}
1046{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001047 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
1048 {
1049 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
1050 stmt.setDataverseName(nameComponents.first);
1051 stmt.setDatasetName(nameComponents.second);
1052 return addSourceLocation(stmt, startToken);
1053 }
1054}
1055
1056CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
1057{
1058 CreateIndexStatement stmt = null;
1059}
1060{
1061 (
1062 <INDEX> stmt = IndexSpecification(startStmtToken)
1063 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
1064 )
1065 {
1066 return stmt;
1067 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001068}
1069
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001070CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001071{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001072 CreateIndexStatement stmt = new CreateIndexStatement();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001073 String indexName = null;
1074 boolean ifNotExists = false;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001075 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001076 Pair<Integer, Pair<List<String>, IndexedTypeExpression>> fieldPair = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001077 IndexParams indexType = null;
1078 boolean enforced = false;
1079}
1080{
Ali Alsuliman8351d252017-09-24 00:43:15 -07001081 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001082 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -07001083 <ON> nameComponents = QualifiedName()
1084 <LEFTPAREN> ( fieldPair = OpenField()
1085 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001086 stmt.addFieldExprPair(fieldPair.second);
1087 stmt.addFieldIndexIndicator(fieldPair.first);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001088 }
Ali Alsuliman8351d252017-09-24 00:43:15 -07001089 ) (<COMMA> fieldPair = OpenField()
1090 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001091 stmt.addFieldExprPair(fieldPair.second);
1092 stmt.addFieldIndexIndicator(fieldPair.first);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001093 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001094 )* <RIGHTPAREN> ( <TYPE> indexType = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman8351d252017-09-24 00:43:15 -07001095 )
1096 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001097 stmt.setIndexName(new Identifier(indexName));
1098 stmt.setIfNotExists(ifNotExists);
1099 stmt.setDataverseName(nameComponents.first);
1100 stmt.setDatasetName(nameComponents.second);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001101 if (indexType != null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001102 stmt.setIndexType(indexType.type);
1103 stmt.setGramLength(indexType.gramLength);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001104 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001105 stmt.setEnforced(enforced);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001106 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001107 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001108}
1109
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001110CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
1111{
1112 CreateIndexStatement stmt = new CreateIndexStatement();
1113 String indexName = null;
1114 boolean ifNotExists = false;
1115 Pair<DataverseName,Identifier> nameComponents = null;
1116}
1117{
1118 (indexName = Identifier())? ifNotExists = IfNotExists()
1119 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
1120 {
1121 if (indexName == null) {
1122 indexName = "primary_idx_" + nameComponents.second;
1123 }
1124 stmt.setIndexName(new Identifier(indexName));
1125 stmt.setIfNotExists(ifNotExists);
1126 stmt.setDataverseName(nameComponents.first);
1127 stmt.setDatasetName(nameComponents.second);
1128 return addSourceLocation(stmt, startStmtToken);
1129 }
1130}
1131
Yingyi Bu391f09e2015-10-29 13:49:39 -07001132String FilterField() throws ParseException :
1133{
1134 String filterField = null;
1135}
1136{
1137 filterField = Identifier()
1138 {
1139 return filterField;
1140 }
1141}
1142
1143IndexParams IndexType() throws ParseException:
1144{
1145 IndexType type = null;
1146 int gramLength = 0;
1147}
1148{
1149 (<BTREE>
1150 {
1151 type = IndexType.BTREE;
1152 }
1153 | <RTREE>
1154 {
1155 type = IndexType.RTREE;
1156 }
1157 | <KEYWORD>
1158 {
1159 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1160 }
Taewoo Kimc49405a2017-01-04 00:30:43 -08001161 |<FULLTEXT>
1162 {
1163 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1164 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001165 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1166 {
1167 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1168 gramLength = Integer.valueOf(token.image);
1169 }
1170 <RIGHTPAREN>)
1171 {
1172 return new IndexParams(type, gramLength);
1173 }
1174}
1175
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001176CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1177{
1178 CreateDataverseStatement stmt = null;
1179}
1180{
1181 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1182 {
1183 return stmt;
1184 }
1185}
1186
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001187CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001188{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001189 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001190 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001191}
1192{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001193 dvName = MultipartIdentifier() ifNotExists = IfNotExists()
1194 {
1195 CreateDataverseStatement stmt = new CreateDataverseStatement(DataverseName.create(dvName), null, ifNotExists);
1196 return addSourceLocation(stmt, startStmtToken);
1197 }
1198}
1199
Ian Maxon38fe9402020-01-29 19:27:40 -08001200CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1201{
1202 CreateAdapterStatement stmt = null;
1203}
1204{
1205 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1206 {
1207 return stmt;
1208 }
1209}
1210
1211CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1212{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001213 Pair<DataverseName,Identifier> adapterName = null;
1214 Pair<DataverseName,Identifier> libraryName = null;
1215 List<String> externalIdentifier = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001216 boolean ifNotExists = false;
1217}
1218{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001219 adapterName = QualifiedName()
1220 ifNotExists = IfNotExists()
1221 <AS> externalIdentifier = FunctionExternalIdentifier()
1222 <AT> libraryName = QualifiedName()
1223 {
1224 CreateAdapterStatement stmt = new CreateAdapterStatement(adapterName.first, adapterName.second.getValue(),
1225 libraryName.first, libraryName.second.getValue(), externalIdentifier, ifNotExists);
1226 return addSourceLocation(stmt, startStmtToken);
1227 }
1228}
Ian Maxon38fe9402020-01-29 19:27:40 -08001229
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001230CreateFunctionStatement CreateFunctionStatement(Token startStmtToken, boolean orReplace) throws ParseException:
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001231{
1232 CreateFunctionStatement stmt = null;
1233}
1234{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001235 <FUNCTION> stmt = FunctionSpecification(startStmtToken, orReplace)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001236 {
1237 return stmt;
1238 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001239}
1240
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001241CreateFunctionStatement FunctionSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001242{
Ian Maxon38fe9402020-01-29 19:27:40 -08001243 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001244 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001245 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001246 List<Pair<VarIdentifier,TypeExpression>> params = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001247 int arity = 0;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001248 TypeExpression returnType = null;
1249 Token beginPos = null, endPos = null;
1250 Expression functionBodyExpr = null;
1251 Pair<DataverseName,Identifier> libraryName = null;
1252 List<String> externalIdentifier = null;
1253 RecordConstructor withOptions = null;
1254 boolean ifNotExists = false;
1255 CreateFunctionStatement stmt = null;
1256 DataverseName currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001257}
1258{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001259 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001260 {
1261 defaultDataverse = fctName.dataverse;
1262 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001263 ifNotExists = IfNotExists()
Dmitry Lychagin541652082020-11-09 14:04:53 -08001264 paramsWithArity = FunctionParameters()
1265 {
1266 arity = paramsWithArity.first;
1267 params = paramsWithArity.second;
1268 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001269 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001270 (
1271 (
1272 <LEFTBRACE>
1273 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001274 beginPos = token;
1275 createNewScope();
Ian Maxon38fe9402020-01-29 19:27:40 -08001276 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001277 functionBodyExpr = FunctionBody()
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001278 <RIGHTBRACE>
1279 {
1280 endPos = token;
1281 String functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine,
1282 endPos.beginColumn);
Dmitry Lychagin541652082020-11-09 14:04:53 -08001283 signature = new FunctionSignature(fctName.dataverse, fctName.function, arity);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001284 getCurrentScope().addFunctionDescriptor(signature, false);
1285 removeCurrentScope();
1286 defaultDataverse = currentDataverse;
1287 ensureNoTypeDeclsInFunction(fctName.function, params, returnType, startStmtToken);
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001288 stmt = new CreateFunctionStatement(signature, params, functionBody, functionBodyExpr, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001289 return addSourceLocation(stmt, startStmtToken);
1290 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001291 )
1292 |
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001293 (
1294 <AS> externalIdentifier = FunctionExternalIdentifier()
1295 <AT> libraryName = QualifiedName()
1296 (<WITH> withOptions = RecordConstructor())?
1297 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001298 signature = new FunctionSignature(fctName.dataverse, fctName.function, arity);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001299 defaultDataverse = currentDataverse;
1300 try {
1301 stmt = new CreateFunctionStatement(signature, params, returnType, libraryName.first,
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001302 libraryName.second.getValue(), externalIdentifier, withOptions, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001303 } catch (AlgebricksException e) {
1304 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1305 }
1306 return addSourceLocation(stmt, startStmtToken);
1307 }
1308 )
Ian Maxon38fe9402020-01-29 19:27:40 -08001309 )
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001310}
1311
Dmitry Lychagin541652082020-11-09 14:04:53 -08001312Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameters() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001313{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001314 Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> params = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001315}
1316{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001317 <LEFTPAREN> (params = FunctionParameterList())? <RIGHTPAREN>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001318 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001319 if (params == null) {
1320 params = new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(
1321 0, Collections.<Pair<VarIdentifier, TypeExpression>>emptyList()
1322 );
1323 }
1324 return params;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001325 }
1326}
1327
Dmitry Lychagin541652082020-11-09 14:04:53 -08001328Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameterList() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001329{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001330 List<Pair<VarIdentifier, TypeExpression>> paramList = null;
1331 Pair<VarIdentifier, TypeExpression> param = null;
1332 int arity = 0;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001333}
1334{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001335 (
1336 (
1337 <DOT> <DOT> <DOT>
1338 {
1339 param = new Pair<VarIdentifier, TypeExpression>(
1340 SqlppVariableUtil.toInternalVariableIdentifier(UDF_VARARGS_PARAM_NAME), null
1341 );
1342 paramList = Collections.<Pair<VarIdentifier, TypeExpression>>singletonList(param);
1343 arity = FunctionIdentifier.VARARGS;
1344 }
1345 )
1346 |
1347 (
1348 param = FunctionParameter()
1349 {
1350 paramList = new ArrayList<Pair<VarIdentifier, TypeExpression>>();
1351 paramList.add(param);
1352 }
1353 ( <COMMA> param = FunctionParameter() { paramList.add(param); } )*
1354 {
1355 arity = paramList.size();
1356 }
1357 )
1358 )
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001359 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001360 return new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(arity, paramList);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001361 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001362}
1363
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001364Pair<VarIdentifier,TypeExpression> FunctionParameter() throws ParseException:
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001365{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001366 String name = null;
1367 TypeExpression type = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001368}
1369{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001370 name = VariableIdentifier()
1371 ( LOOKAHEAD(3) (<COLON>)? type = TypeExpr(false) )?
1372 {
1373 return new Pair<VarIdentifier,TypeExpression>(new VarIdentifier(name), type);
1374 }
1375}
1376
1377TypeExpression FunctionReturnType() throws ParseException:
1378{
1379 TypeExpression returnType = null;
1380}
1381{
1382 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = TypeExpr(false) )?
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001383 {
1384 return returnType;
1385 }
1386}
1387
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001388Expression FunctionBody() throws ParseException:
1389{
1390 Expression functionBodyExpr = null;
1391}
1392{
1393 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1394 {
1395 return functionBodyExpr;
1396 }
1397}
1398
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001399List<String> FunctionExternalIdentifier() throws ParseException:
1400{
1401 String ident = null;
1402 List<String> identList = new ArrayList(2);
1403}
1404{
1405 ident = StringLiteral() { identList.add(ident.trim()); }
1406 ( <COMMA> ident = StringLiteral() { identList.add(ident.trim()); } )*
1407 {
1408 return identList;
1409 }
1410}
1411
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001412CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1413{
1414 CreateFeedStatement stmt = null;
1415}
1416{
1417 <FEED> stmt = FeedSpecification(startStmtToken)
1418 {
1419 return stmt;
1420 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001421}
1422
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001423CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001424{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001425 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001426 boolean ifNotExists = false;
1427 String adapterName = null;
1428 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001429 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001430 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001431 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001432}
1433{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001434 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001435 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001436 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001437 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001438 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001439 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001440 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001441 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001442 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001443 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001444}
1445
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001446CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1447{
1448 CreateFeedPolicyStatement stmt = null;
1449}
1450{
1451 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1452 {
1453 return stmt;
1454 }
1455}
1456
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001457CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001458{
Michael Blowd6cf6412016-06-30 02:44:35 -04001459 String policyName = null;
1460 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001461 String sourcePolicyFile = null;
1462 String definition = null;
1463 boolean ifNotExists = false;
1464 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001465 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001466}
1467{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001468 policyName = Identifier() ifNotExists = IfNotExists()
1469 <FROM>
1470 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001471 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001472 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001473 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001474 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1475 {
1476 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1477 }
1478 )
1479 {
1480 return addSourceLocation(stmt, startStmtToken);
1481 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001482}
1483
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001484CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
1485{
1486 CreateSynonymStatement stmt = null;
1487}
1488{
1489 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
1490 {
1491 return stmt;
1492 }
1493}
1494
1495CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
1496{
1497 Pair<DataverseName,Identifier> nameComponents = null;
1498 Pair<DataverseName,Identifier> objectNameComponents = null;
1499 boolean ifNotExists = false;
1500}
1501{
1502 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1503 <FOR> objectNameComponents = QualifiedName()
1504 {
1505 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
1506 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
1507 return addSourceLocation(stmt, startStmtToken);
1508 }
1509}
1510
Yingyi Bu391f09e2015-10-29 13:49:39 -07001511boolean IfNotExists() throws ParseException:
1512{
1513}
1514{
Yingyi Budaa549c2016-06-28 22:30:52 -07001515 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001516 {
1517 return true;
1518 }
1519 )?
1520 {
1521 return false;
1522 }
1523}
1524
Xikui Wang9d63f622017-05-18 17:50:44 -07001525void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001526{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001527 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07001528 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001529}
1530{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001531 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001532 {
1533 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1534 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
1535 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07001536 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001537 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07001538 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001539 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1540 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07001541 }
1542 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07001543}
1544
1545String GetPolicy() throws ParseException:
1546{
1547 String policy = null;
1548}
1549{
1550 <USING> <POLICY> policy = Identifier()
1551 {
1552 return policy;
1553 }
1554
1555}
1556
1557FunctionSignature FunctionSignature() throws ParseException:
1558{
1559 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001560 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> params = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001561 int arity = 0;
1562}
1563{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001564 fctName = FunctionName()
1565 (
Dmitry Lychagin541652082020-11-09 14:04:53 -08001566 LOOKAHEAD(2) params = FunctionParameters() { arity = params.first; }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001567 | ( <LEFTPAREN> arity = FunctionArity() <RIGHTPAREN> )
1568 | ( <ATT> arity = FunctionArity() ) // back-compat
1569 )
1570 {
1571 return new FunctionSignature(fctName.dataverse, fctName.function, arity);
1572 }
1573}
Yingyi Bu391f09e2015-10-29 13:49:39 -07001574
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001575int FunctionArity() throws ParseException:
1576{
1577 int arity;
1578}
1579{
1580 <INTEGER_LITERAL>
1581 {
1582 try {
1583 arity = Integer.parseInt(token.image);
1584 } catch (NumberFormatException e) {
1585 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001586 }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001587 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
1588 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + arity);
1589 }
1590 return arity;
1591 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001592}
1593
Yingyi Buc9bfe252016-03-01 00:02:40 -08001594Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001595{
Yingyi Buc9bfe252016-03-01 00:02:40 -08001596 Pair<Integer, List<String>> tmp = null;
1597 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001598 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
1599}
1600{
1601 <PRIMARY> <KEY> tmp = NestedField()
1602 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001603 keyFieldSourceIndicators.add(tmp.first);
1604 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001605 }
1606 ( <COMMA> tmp = NestedField()
1607 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001608 keyFieldSourceIndicators.add(tmp.first);
1609 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001610 }
1611 )*
1612 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001613 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001614 }
1615}
1616
1617Statement DropStatement() throws ParseException:
1618{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001619 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001620 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001621}
1622{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001623 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001624 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001625 stmt = DropDatasetStatement(startToken)
1626 | stmt = DropIndexStatement(startToken)
1627 | stmt = DropNodeGroupStatement(startToken)
1628 | stmt = DropTypeStatement(startToken)
1629 | stmt = DropDataverseStatement(startToken)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001630 | stmt = DropAdapterStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001631 | stmt = DropFunctionStatement(startToken)
1632 | stmt = DropFeedStatement(startToken)
1633 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001634 | stmt = DropSynonymStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07001635 )
1636 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001637 return stmt;
1638 }
1639}
1640
1641DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
1642{
1643 DropDatasetStatement stmt = null;
1644}
1645{
1646 Dataset() stmt = DropDatasetSpecification(startStmtToken)
1647 {
1648 return stmt;
1649 }
1650}
1651
1652DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
1653{
1654 Pair<DataverseName,Identifier> pairId = null;
1655 boolean ifExists = false;
1656}
1657{
1658 pairId = QualifiedName() ifExists = IfExists()
1659 {
1660 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
1661 return addSourceLocation(stmt, startStmtToken);
1662 }
1663}
1664
1665IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
1666{
1667 IndexDropStatement stmt = null;
1668}
1669{
1670 <INDEX> stmt = DropIndexSpecification(startStmtToken)
1671 {
1672 return stmt;
1673 }
1674}
1675
1676IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
1677{
1678 Triple<DataverseName,Identifier,Identifier> tripleId = null;
1679 boolean ifExists = false;
1680}
1681{
1682 tripleId = DoubleQualifiedName() ifExists = IfExists()
1683 {
1684 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
1685 return addSourceLocation(stmt, startStmtToken);
1686 }
1687}
1688
1689NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
1690{
1691 NodeGroupDropStatement stmt = null;
1692}
1693{
1694 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
1695 {
1696 return stmt;
1697 }
1698}
1699
1700NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
1701{
1702 String id = null;
1703 boolean ifExists = false;
1704}
1705{
1706 id = Identifier() ifExists = IfExists()
1707 {
1708 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
1709 return addSourceLocation(stmt, startStmtToken);
1710 }
1711}
1712
1713TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
1714{
1715 TypeDropStatement stmt = null;
1716}
1717{
1718 <TYPE> stmt = DropTypeSpecification(startStmtToken)
1719 {
1720 return stmt;
1721 }
1722}
1723
1724TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
1725{
1726 Pair<DataverseName,Identifier> pairId = null;
1727 boolean ifExists = false;
1728}
1729{
1730 pairId = TypeName() ifExists = IfExists()
1731 {
1732 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
1733 return addSourceLocation(stmt, startStmtToken);
1734 }
1735}
1736
1737DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
1738{
1739 DataverseDropStatement stmt = null;
1740}
1741{
1742 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
1743 {
1744 return stmt;
1745 }
1746}
1747
1748DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
1749{
1750 List<String> multipartId = null;
1751 boolean ifExists = false;
1752}
1753{
1754 multipartId = MultipartIdentifier() ifExists = IfExists()
1755 {
1756 DataverseDropStatement stmt = new DataverseDropStatement(DataverseName.create(multipartId), ifExists);
1757 return addSourceLocation(stmt, startStmtToken);
1758 }
1759}
1760
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001761AdapterDropStatement DropAdapterStatement(Token startStmtToken) throws ParseException:
1762{
1763 AdapterDropStatement stmt = null;
1764}
1765{
1766 <ADAPTER> stmt = DropAdapterSpecification(startStmtToken)
1767 {
1768 return stmt;
1769 }
1770}
1771
1772AdapterDropStatement DropAdapterSpecification(Token startStmtToken) throws ParseException:
1773{
1774 Pair<DataverseName,Identifier> adapterName = null;
1775 boolean ifExists = false;
1776}
1777{
1778 adapterName = QualifiedName() ifExists = IfExists()
1779 {
1780 AdapterDropStatement stmt = new AdapterDropStatement(adapterName.first, adapterName.second.getValue(), ifExists);
1781 return addSourceLocation(stmt, startStmtToken);
1782 }
1783}
1784
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001785FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
1786{
1787 FunctionDropStatement stmt = null;
1788}
1789{
1790 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
1791 {
1792 return stmt;
1793 }
1794}
1795
1796FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
1797{
1798 FunctionSignature funcSig = null;
1799 boolean ifExists = false;
1800}
1801{
1802 funcSig = FunctionSignature() ifExists = IfExists()
1803 {
1804 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
1805 return addSourceLocation(stmt, startStmtToken);
1806 }
1807}
1808
1809FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
1810{
1811 FeedDropStatement stmt = null;
1812}
1813{
1814 <FEED> stmt = DropFeedSpecification(startStmtToken)
1815 {
1816 return stmt;
1817 }
1818}
1819
1820FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
1821{
1822 Pair<DataverseName,Identifier> pairId = null;
1823 boolean ifExists = false;
1824}
1825{
1826 pairId = QualifiedName() ifExists = IfExists()
1827 {
1828 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
1829 return addSourceLocation(stmt, startStmtToken);
1830 }
1831}
1832
1833FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
1834{
1835 FeedPolicyDropStatement stmt = null;
1836}
1837{
1838 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
1839 {
1840 return stmt;
1841 }
1842}
1843
1844FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
1845{
1846 Pair<DataverseName,Identifier> pairId = null;
1847 boolean ifExists = false;
1848}
1849{
1850 pairId = QualifiedName() ifExists = IfExists()
1851 {
1852 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
1853 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001854 }
1855}
1856
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001857SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
1858{
1859 SynonymDropStatement stmt = null;
1860}
1861{
1862 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
1863 {
1864 return stmt;
1865 }
1866}
1867
1868SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
1869{
1870 Pair<DataverseName,Identifier> pairId = null;
1871 boolean ifExists = false;
1872}
1873{
1874 pairId = QualifiedName() ifExists = IfExists()
1875 {
1876 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
1877 return addSourceLocation(stmt, startStmtToken);
1878 }
1879}
1880
Yingyi Bu391f09e2015-10-29 13:49:39 -07001881boolean IfExists() throws ParseException :
1882{
1883}
1884{
1885 ( LOOKAHEAD(1) <IF> <EXISTS>
1886 {
1887 return true;
1888 }
1889 )?
1890 {
1891 return false;
1892 }
1893}
1894
1895InsertStatement InsertStatement() throws ParseException:
1896{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001897 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001898 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001899 VariableExpr var = null;
1900 Query query = null;
1901 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001902}
1903{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001904 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001905 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001906 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001907 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001908 if (var == null && returnExpression != null) {
1909 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1910 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001911 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08001912 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001913 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1914 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001915 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001916 }
1917}
1918
1919UpsertStatement UpsertStatement() throws ParseException:
1920{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001921 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001922 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001923 VariableExpr var = null;
1924 Query query = null;
1925 Expression returnExpression = null;
1926}
1927{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001928 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001929 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001930 ( <RETURNING> returnExpression = Expression())?
1931 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001932 if (var == null && returnExpression != null) {
1933 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1934 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001935 }
1936 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001937 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1938 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001939 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001940 }
1941}
1942
1943DeleteStatement DeleteStatement() throws ParseException:
1944{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001945 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001946 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001947 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001948 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001949}
1950{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001951 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001952 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001953 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07001954 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001955 if (var == null) {
1956 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1957 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001958 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001959 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07001960 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001961 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001962 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001963}
1964
1965UpdateStatement UpdateStatement() throws ParseException:
1966{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001967 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001968 VariableExpr vars;
1969 Expression target;
1970 Expression condition;
1971 UpdateClause uc;
1972 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
1973}
1974{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001975 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001976 <WHERE> condition = Expression()
1977 <LEFTPAREN> (uc = UpdateClause()
1978 {
1979 ucs.add(uc);
1980 }
1981 (<COMMA> uc = UpdateClause()
1982 {
1983 ucs.add(uc);
1984 }
1985 )*) <RIGHTPAREN>
1986 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001987 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001988 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001989 }
1990}
1991
1992UpdateClause UpdateClause() throws ParseException:
1993{
1994 Expression target = null;
1995 Expression value = null ;
1996 InsertStatement is = null;
1997 DeleteStatement ds = null;
1998 UpdateStatement us = null;
1999 Expression condition = null;
2000 UpdateClause ifbranch = null;
2001 UpdateClause elsebranch = null;
2002}
2003{
2004 (<SET> target = Expression() <EQ> value = Expression()
2005 | is = InsertStatement()
2006 | ds = DeleteStatement()
2007 | us = UpdateStatement()
2008 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
2009 <THEN> ifbranch = UpdateClause()
2010 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
2011 {
2012 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
2013 }
2014 )
2015}
2016
2017Statement SetStatement() throws ParseException:
2018{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002019 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002020 String pn = null;
2021 String pv = null;
2022}
2023{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002024 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002025 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002026 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002027 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002028 }
2029}
2030
2031Statement WriteStatement() throws ParseException:
2032{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002033 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002034 String nodeName = null;
2035 String fileName = null;
2036 Query query;
2037 String writerClass = null;
2038 Pair<Identifier,Identifier> nameComponents = null;
2039}
2040{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002041 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07002042 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002043 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002044 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002045 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002046 }
2047}
2048
2049LoadStatement LoadStatement() throws ParseException:
2050{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002051 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002052 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002053 Identifier datasetName = null;
2054 boolean alreadySorted = false;
2055 String adapterName;
2056 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002057 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002058}
2059{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002060 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002061 {
2062 dataverseName = nameComponents.first;
2063 datasetName = nameComponents.second;
2064 }
2065 <USING> adapterName = AdapterName() properties = Configuration()
2066 (<PRESORTED>
2067 {
2068 alreadySorted = true;
2069 }
2070 )?
2071 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002072 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
2073 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002074 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002075 }
2076}
2077
2078
2079String AdapterName() throws ParseException :
2080{
2081 String adapterName = null;
2082}
2083{
2084 adapterName = Identifier()
2085 {
2086 return adapterName;
2087 }
2088}
2089
2090Statement CompactStatement() throws ParseException:
2091{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002092 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002093 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002094}
2095{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002096 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002097 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002098 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002099 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002100 }
2101}
2102
Yingyi Buab817482016-08-19 21:29:31 -07002103Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002104{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002105 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002106 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002107}
2108{
2109 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002110 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
2111 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
2112 | <START> { startToken = token; } stmt = StartStatement(startToken)
2113 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07002114 )
2115 {
2116 return stmt;
2117 }
2118}
2119
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002120Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002121{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002122 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002123 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002124}
2125{
2126 <FEED> feedNameComponents = QualifiedName()
2127 {
2128 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002129 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002130 }
2131}
2132
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002133AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002134{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002135 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002136 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002137}
2138{
2139 <FEED> feedNameComponents = QualifiedName()
2140 {
2141 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002142 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002143 }
2144}
2145
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002146AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002147{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002148 Pair<DataverseName,Identifier> feedNameComponents = null;
2149 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002150
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002151 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002152}
2153{
2154 (
2155 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002156 {
2157 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
2158 }
2159 )
Yingyi Buab817482016-08-19 21:29:31 -07002160 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002161 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002162 }
2163}
2164
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002165AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002166{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002167 Pair<DataverseName,Identifier> feedNameComponents = null;
2168 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002169
2170 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002171 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002172 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002173 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002174 String whereClauseBody = null;
2175 WhereClause whereClause = null;
2176 Token beginPos = null;
2177 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002178}
2179{
2180 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002181 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002182 (ApplyFunction(appliedFunctions))?
2183 (policy = GetPolicy())?
2184 (
2185 <WHERE>
2186 {
2187 beginPos = token;
2188 whereClause = new WhereClause();
2189 Expression whereExpr;
2190 }
2191 whereExpr = Expression()
2192 {
2193 whereClause.setWhereExpr(whereExpr);
2194 }
2195 )?
2196 {
2197 if (whereClause != null) {
2198 endPos = token;
2199 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2200 }
2201 }
Yingyi Buab817482016-08-19 21:29:31 -07002202 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002203 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002204 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002205 }
2206 )
2207 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002208 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002209 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002210}
2211
2212Map<String,String> Configuration() throws ParseException :
2213{
2214 Map<String,String> configuration = new LinkedHashMap<String,String>();
2215 Pair<String, String> keyValuePair = null;
2216}
2217{
2218 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2219 {
2220 configuration.put(keyValuePair.first, keyValuePair.second);
2221 }
2222 ( <COMMA> keyValuePair = KeyValuePair()
2223 {
2224 configuration.put(keyValuePair.first, keyValuePair.second);
2225 }
2226 )* )? <RIGHTPAREN>
2227 {
2228 return configuration;
2229 }
2230}
2231
2232Pair<String, String> KeyValuePair() throws ParseException:
2233{
2234 String key;
2235 String value;
2236}
2237{
Ali Alsulimane2986012020-05-14 18:14:22 -07002238 <LEFTPAREN> key = ConstantString()
2239 <EQ> ( value = ConstantString() | (<TRUE> | <FALSE>) {value = token.image.toLowerCase();} )
2240 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002241 {
2242 return new Pair<String, String>(key, value);
2243 }
2244}
2245
2246Map<String,String> Properties() throws ParseException:
2247{
2248 Map<String,String> properties = new HashMap<String,String>();
2249 Pair<String, String> property;
2250}
2251{
2252 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
2253 {
2254 properties.put(property.first, property.second);
2255 }
2256 ( <COMMA> property = Property()
2257 {
2258 properties.put(property.first, property.second);
2259 }
2260 )* <RIGHTPAREN> )?
2261 {
2262 return properties;
2263 }
2264}
2265
2266Pair<String, String> Property() throws ParseException:
2267{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002268 String key = null;
2269 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002270}
2271{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002272 (key = Identifier() | key = StringLiteral())
2273 <EQ>
2274 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002275 {
2276 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002277 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002278 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002279 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002280 }
2281 }
2282 )
2283 {
2284 return new Pair<String, String>(key.toUpperCase(), value);
2285 }
2286}
2287
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002288IndexedTypeExpression IndexedTypeExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002289{
2290 TypeExpression typeExpr = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002291 boolean isUnknownable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002292}
2293{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002294 typeExpr = TypeExpr(false)
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002295 ( <QUES> { isUnknownable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002296 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002297 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002298 }
2299}
2300
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002301TypeExpression TypeExpr(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002302{
2303 TypeExpression typeExpr = null;
2304}
2305{
2306 (
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002307 typeExpr = TypeReference()
2308 | typeExpr = OrderedListTypeDef(allowRecordTypeDef)
2309 | typeExpr = UnorderedListTypeDef(allowRecordTypeDef)
2310 | typeExpr = RecordTypeDef() {
2311 if (!allowRecordTypeDef) {
2312 throw new SqlppParseException(typeExpr.getSourceLocation(), "Unexpected record type declaration");
2313 }
2314 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002315 )
2316 {
2317 return typeExpr;
2318 }
2319}
2320
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002321RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
2322{
2323 RecordTypeDefinition.RecordKind recordKind = null;
2324}
2325{
2326 (
2327 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
2328 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
2329 )
2330 {
2331 return recordKind;
2332 }
2333}
2334
Yingyi Bu391f09e2015-10-29 13:49:39 -07002335RecordTypeDefinition RecordTypeDef() throws ParseException:
2336{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002337 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002338 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002339 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002340}
2341{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002342 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002343 <LEFTBRACE>
2344 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002345 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002346 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
2347 if (hintToken != null) {
2348 String hintParams = hintToken.hintParams;
2349 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
2350 if (splits == null || splits.length != 4) {
2351 throw new SqlppParseException(getSourceLocation(hintToken),
2352 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002353 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002354 if (!splits[0].equals("int")) {
2355 throw new SqlppParseException(getSourceLocation(hintToken),
2356 "The only supported type for gen-fields is int.");
2357 }
2358 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
2359 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
2360 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002361 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002362 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002363 (
2364 RecordField(recType)
2365 ( <COMMA> RecordField(recType) )*
2366 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002367 <RIGHTBRACE>
2368 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002369 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002370 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002371 }
2372}
2373
2374void RecordField(RecordTypeDefinition recType) throws ParseException:
2375{
2376 String fieldName;
2377 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002378 boolean nullable = false, missable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002379}
2380{
2381 fieldName = Identifier()
2382 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002383 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
2384 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
2385 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
2386 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002387 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002388 <COLON> type = TypeExpr(true) ( <QUES> { nullable = true; missable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002389 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002390 recType.addField(fieldName, type, nullable, missable, rfdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002391 }
2392}
2393
2394TypeReferenceExpression TypeReference() throws ParseException:
2395{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002396 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002397}
2398{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002399 id = QualifiedName()
2400 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07002401 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
2402 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002403 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002404
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002405 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002406 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002407 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002408}
2409
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002410OrderedListTypeDefinition OrderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002411{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002412 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002413 TypeExpression type = null;
2414}
2415{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002416 <LEFTBRACKET> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002417 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002418 <RIGHTBRACKET>
2419 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002420 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002421 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002422 }
2423}
2424
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002425UnorderedListTypeDefinition UnorderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002426{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002427 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002428 TypeExpression type = null;
2429}
2430{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002431 <LEFTDBLBRACE> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002432 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002433 <RIGHTDBLBRACE>
2434 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002435 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002436 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002437 }
2438}
2439
2440FunctionName FunctionName() throws ParseException:
2441{
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002442 Triple<List<String>, SourceLocation, Token> prefix = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002443 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002444}
2445{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002446 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
2447 // that copy must be kept in sync with this code
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002448 prefix = MultipartIdentifierWithHints(
2449 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.RANGE_HINT,
2450 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
2451 )
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002452 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002453 {
2454 FunctionName result = new FunctionName();
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002455 result.sourceLoc = prefix.second;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002456 result.hintToken = prefix.third;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002457 List<String> list = prefix.first;
2458 int ln = list.size();
2459 String last = list.get(ln - 1);
2460 if (suffix == null) {
2461 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
2462 // no library name
2463 result.function = last;
2464 } else {
2465 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
2466 // suffix = func_name
2467 result.library = last;
2468 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002469 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002470 if (ln > 1) {
2471 result.dataverse = DataverseName.create(list, 0, ln - 1);
2472 } else {
2473 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002474 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002475
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002476 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
2477 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002478 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002479 return result;
2480 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002481}
2482
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002483Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002484{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002485 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002486}
2487{
2488 name = QualifiedName()
2489 {
2490 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002491 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002492 }
2493 return name;
2494 }
2495}
2496
2497String Identifier() throws ParseException:
2498{
2499 String lit = null;
2500}
2501{
2502 (<IDENTIFIER>
2503 {
2504 return token.image;
2505 }
2506 | lit = QuotedString()
2507 {
2508 return lit;
2509 }
2510 )
2511}
2512
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002513List<String> ParenthesizedIdentifierList() throws ParseException:
2514{
2515 List<String> list = new ArrayList<String>();
2516 String ident = null;
2517}
2518{
2519 <LEFTPAREN>
2520 ident = Identifier() { list.add(ident); }
2521 ( <COMMA> ident = Identifier() { list.add(ident); } )*
2522 <RIGHTPAREN>
2523 {
2524 return list;
2525 }
2526}
2527
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002528Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002529{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002530 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002531 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002532}
2533{
2534 fieldList = NestedField()
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002535 ( <COLON> fieldType = IndexedTypeExpr() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002536 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002537 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
2538 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002539 }
2540}
2541
Yingyi Buc9bfe252016-03-01 00:02:40 -08002542Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002543{
2544 List<String> exprList = new ArrayList<String>();
2545 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002546 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002547 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002548}
2549{
2550 lit = Identifier()
2551 {
Yingyi Bub9169b62016-02-26 21:21:49 -08002552 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002553 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08002554 }
2555 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08002556 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08002557 <LEFTPAREN><RIGHTPAREN>
2558 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002559 if(!lit.equalsIgnoreCase("meta")){
2560 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08002561 }
2562 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002563 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08002564 }
2565 )?
2566 {
2567 if(!meetParens){
2568 exprList.add(lit);
2569 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002570 }
2571 (<DOT>
2572 lit = Identifier()
2573 {
2574 exprList.add(lit);
2575 }
2576 )*
2577 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002578 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002579 }
2580}
2581
Yingyi Bu6d57e492016-06-06 21:24:42 -07002582String ConstantString() throws ParseException:
2583{
2584 String value = null;
2585}
2586{
2587 (value = QuotedString() | value = StringLiteral())
2588 {
2589 return value;
2590 }
2591}
2592
Yingyi Bu391f09e2015-10-29 13:49:39 -07002593String QuotedString() throws ParseException:
2594{
2595}
2596{
2597 <QUOTED_STRING>
2598 {
2599 return removeQuotesAndEscapes(token.image);
2600 }
2601}
2602
Yingyi Bu391f09e2015-10-29 13:49:39 -07002603String StringLiteral() throws ParseException:
2604{
2605}
2606{
2607 <STRING_LITERAL>
2608 {
2609 return removeQuotesAndEscapes(token.image);
2610 }
2611}
2612
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002613List<String> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002614{
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002615 Triple<List<String>, SourceLocation, Token> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002616}
2617{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002618 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002619 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002620 return result.first;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002621 }
2622}
2623
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002624Triple<List<String>, SourceLocation, Token> MultipartIdentifierWithHints(SqlppHint... expectedHints)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002625 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002626{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002627 List<String> list = new ArrayList<String>();
2628 SourceLocation sourceLoc = null;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002629 Token hint = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002630 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002631}
2632{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002633 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002634 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002635 list.add(item);
2636 sourceLoc = getSourceLocation(token);
2637 if (expectedHints != null && expectedHints.length > 0) {
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002638 hint = fetchHint(token, expectedHints);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002639 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002640 }
2641 (<DOT> item = Identifier() { list.add(item); } )*
2642 {
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002643 return new Triple<List<String>, SourceLocation, Token>(list, sourceLoc, hint);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002644 }
2645}
2646
2647Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
2648{
2649 List<String> list = null;
2650}
2651{
2652 list = MultipartIdentifier()
2653 {
2654 int len = list.size();
2655 DataverseName id1 = len > 1 ? DataverseName.create(list, 0, len - 1) : null;
2656 Identifier id2 = new Identifier(list.get(len - 1));
2657 return new Pair<DataverseName,Identifier>(id1, id2);
2658 }
2659}
2660
2661Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
2662{
2663 List<String> list = new ArrayList<String>();
2664 String item = null;
2665}
2666{
2667 item = Identifier() { list.add(item); }
2668 (<DOT> item = Identifier() { list.add(item); } )+
2669 {
2670 int len = list.size();
2671 DataverseName id1 = len > 2 ? DataverseName.create(list, 0, len - 2) : null;
2672 Identifier id2 = new Identifier(list.get(len - 2));
2673 Identifier id3 = new Identifier(list.get(len - 1));
2674 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002675 }
2676}
2677
2678FunctionDecl FunctionDeclaration() throws ParseException:
2679{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002680 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002681 String functionName;
Dmitry Lychagin541652082020-11-09 14:04:53 -08002682 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002683 Expression funcBody;
2684 createNewScope();
2685}
2686{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002687 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07002688 functionName = Identifier()
Dmitry Lychagin541652082020-11-09 14:04:53 -08002689 paramsWithArity = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07002690 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08002691 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07002692 <RIGHTBRACE>
2693 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08002694 int arity = paramsWithArity.first;
2695 List<Pair<VarIdentifier,TypeExpression>> paramList = paramsWithArity.second;
2696 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, arity);
Xikui Wang3de700a2018-03-15 16:32:55 -07002697 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002698 ensureNoTypeDeclsInFunction(functionName, paramList, null, startToken);
2699 List<VarIdentifier> params = new ArrayList<VarIdentifier>(paramList.size());
2700 for (Pair<VarIdentifier,TypeExpression> p: paramList) {
2701 params.add(p.getFirst());
Ian Maxon38fe9402020-01-29 19:27:40 -08002702 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002703 FunctionDecl stmt = new FunctionDecl(signature, params, funcBody);
Xikui Wang3de700a2018-03-15 16:32:55 -07002704 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002705 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07002706 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002707}
2708
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002709Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002710{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002711 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002712 Expression expr;
2713}
2714{
2715 (
2716 expr = Expression()
2717 |
2718 expr = SelectExpression(false)
2719 )
2720 {
2721 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002722 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07002723 return query;
2724 }
2725}
2726
2727
Yingyi Bu391f09e2015-10-29 13:49:39 -07002728Expression Expression():
2729{
2730 Expression expr = null;
2731 Expression exprP = null;
2732}
2733{
2734(
2735 LOOKAHEAD(2)
2736 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002737 | expr = QuantifiedExpression()
2738)
2739 {
2740 return (exprP==null) ? expr : exprP;
2741 }
2742}
2743
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002744Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002745{
2746 OperatorExpr op = null;
2747 Expression operand = null;
2748}
2749{
2750 operand = AndExpr()
2751 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002752 <OR>
2753 {
2754 if (op == null) {
2755 op = new OperatorExpr();
2756 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07002757 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002758 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002759 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002760 try{
2761 op.addOperator(token.image.toLowerCase());
2762 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002763 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002764 }
Xikui Wang3de700a2018-03-15 16:32:55 -07002765 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002766
Xikui Wang3de700a2018-03-15 16:32:55 -07002767 operand = AndExpr()
2768 {
2769 op.addOperand(operand);
2770 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002771
2772 )*
2773
2774 {
2775 return op==null? operand: op;
2776 }
2777}
2778
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002779Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002780{
2781 OperatorExpr op = null;
2782 Expression operand = null;
2783}
2784{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002785 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002786 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002787 <AND>
2788 {
2789 if (op == null) {
2790 op = new OperatorExpr();
2791 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002792 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002793 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002794 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002795 try{
2796 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002797 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002798 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002799 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002800 }
2801
Yingyi Bu196db5d2016-07-15 19:07:20 -07002802 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002803 {
2804 op.addOperand(operand);
2805 }
2806
2807 )*
2808
2809 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002810 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002811 }
2812}
2813
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002814Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07002815{
2816 Expression inputExpr;
2817 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002818 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07002819}
2820{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002821 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07002822 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002823 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002824 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002825 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002826 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002827 } else {
2828 return inputExpr;
2829 }
2830 }
2831}
Yingyi Bu391f09e2015-10-29 13:49:39 -07002832
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002833Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002834{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002835 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002836 OperatorExpr op = null;
2837 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002838 IExpressionAnnotation annotation = null;
2839}
2840{
Yingyi Bu6c638342016-09-02 17:54:34 -07002841 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002842
2843 (
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002844 LOOKAHEAD(2)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN>)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002845 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002846 Token hintToken = fetchHint(token,
2847 SqlppHint.HASH_BROADCAST_JOIN_HINT, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
2848 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
2849 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002850 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002851 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002852 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002853 String operator = token.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07002854 if (operator.equals("<>")){
2855 operator = "!=";
2856 }
2857 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002858 operator = "not_" + operator;
2859 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002860 if (op == null) {
2861 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07002862 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002863 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002864 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002865 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002866 try{
2867 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002868 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002869 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002870 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002871 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002872
Yingyi Bu6c638342016-09-02 17:54:34 -07002873 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07002874 {
Shiva2ea73232019-10-09 18:04:05 -07002875 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07002876 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002877 )?
2878
2879 {
2880 if (annotation != null) {
2881 op.addHint(annotation);
2882 }
2883 return op==null? operand: op;
2884 }
2885}
2886
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002887Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07002888{
2889 boolean not = false;
2890 OperatorExpr op = null;
2891 Expression operand = null;
2892 IExpressionAnnotation annotation = null;
2893}
2894{
2895 operand = IsExpr()
2896 (
2897 LOOKAHEAD(2)
2898 (<NOT> { not = true; })? <BETWEEN>
2899 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002900 Token hintToken = fetchHint(token,
2901 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT,
2902 SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
2903 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002904 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002905 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu6c638342016-09-02 17:54:34 -07002906 }
2907 String operator = token.image.toLowerCase();
2908 if(not){
2909 operator = "not_" + operator;
2910 }
2911 if (op == null) {
2912 op = new OperatorExpr();
2913 op.addOperand(operand);
2914 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002915 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07002916 }
2917 try{
2918 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002919 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002920 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07002921 }
2922 }
2923
2924 operand = IsExpr()
2925 {
2926 op.addOperand(operand);
2927 }
2928
2929 <AND>
2930 operand = IsExpr()
2931 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08002932 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07002933 op.addOperand(operand);
2934 }
2935 )?
2936
2937 {
2938 if (annotation != null) {
2939 op.addHint(annotation);
2940 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002941 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07002942 }
2943}
2944
Yingyi Budaa549c2016-06-28 22:30:52 -07002945Expression IsExpr() throws ParseException:
2946{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002947 Token notToken = null;
2948 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002949 Expression operand = null;
2950 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002951 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002952}
2953{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002954 operand = LikeExpr()
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002955 ( <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002956 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002957 (
2958 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
2959 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08002960 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07002961 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002962 )
Yingyi Budaa549c2016-06-28 22:30:52 -07002963 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002964 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07002965 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002966 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002967 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002968 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07002969 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002970 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07002971 }
2972 }
2973 )?
2974 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002975 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07002976 }
2977}
2978
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002979Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002980{
2981 boolean not = false;
2982 OperatorExpr op = null;
2983 Expression operand = null;
2984}
2985{
2986 operand = ConcatExpr()
2987 (
2988 LOOKAHEAD(2)
2989 (<NOT> { not = true; })? <LIKE>
2990 {
2991 op = new OperatorExpr();
2992 op.addOperand(operand);
2993 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002994 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002995
2996 String operator = token.image.toLowerCase();
2997 if (not) {
2998 operator = "not_" + operator;
2999 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003000 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003001 op.addOperator(operator);
3002 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003003 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003004 }
3005 }
3006
3007 operand = ConcatExpr()
3008 {
3009 op.addOperand(operand);
3010 }
3011 )?
3012
3013 {
3014 return op == null ? operand : op;
3015 }
3016}
3017
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003018Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003019{
3020 OperatorExpr op = null;
3021 Expression operand = null;
3022}
3023{
3024 operand = AddExpr()
3025 (
3026 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003027 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003028 {
3029 if (op == null) {
3030 op = new OperatorExpr();
3031 op.addOperand(operand);
3032 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003033 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003034 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003035 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003036 }
3037 operand = AddExpr()
3038 {
3039 op.addOperand(operand);
3040 }
3041 )*
3042
3043 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003044 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003045 }
3046}
Yingyi Budaa549c2016-06-28 22:30:52 -07003047
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003048Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003049{
3050 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003051 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003052 Expression operand = null;
3053}
3054{
3055 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07003056 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003057 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003058 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003059 {
3060 if (op == null) {
3061 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003062 op.addOperand(operand);
3063 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003064 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003065 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003066 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003067 }
3068
3069 operand = MultExpr()
3070 {
3071 op.addOperand(operand);
3072 }
3073 )*
3074
3075 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003076 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003077 }
3078}
3079
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003080Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003081{
3082 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003083 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003084 Expression operand = null;
3085}
3086{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003087 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003088
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003089 ( (
3090 <MUL> { opType = OperatorType.MUL; } |
3091 <DIVIDE> { opType = OperatorType.DIVIDE; } |
3092 <DIV> { opType = OperatorType.DIV; } |
3093 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
3094 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003095 {
3096 if (op == null) {
3097 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003098 op.addOperand(operand);
3099 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003100 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003101 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003102 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003103 }
3104 operand = ExponentExpr()
3105 {
3106 op.addOperand(operand);
3107 }
3108 )*
3109
3110 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003111 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003112 }
3113}
3114
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003115Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003116{
3117 OperatorExpr op = null;
3118 Expression operand = null;
3119}
3120{
3121 operand = UnaryExpr()
3122 (<CARET>
3123 {
3124 if (op == null) {
3125 op = new OperatorExpr();
3126 op.addOperand(operand);
3127 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003128 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003129 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003130 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003131 }
3132 operand = UnaryExpr()
3133 {
3134 op.addOperand(operand);
3135 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003136 )?
3137 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003138 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003139 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003140}
3141
3142Expression UnaryExpr() throws ParseException:
3143{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003144 boolean not = false;
3145 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003146 Expression expr = null;
3147}
Yingyi Budaa549c2016-06-28 22:30:52 -07003148{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003149 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003150 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003151 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003152 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003153 exprType = "not_" + exprType;
3154 }
3155 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003156 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003157 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003158 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003159 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003160 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07003161 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003162 }
3163 )?
3164
3165 expr = ValueExpr()
3166 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003167 if (uexpr == null) {
3168 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003169 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003170 uexpr.setExpr(expr);
3171 return uexpr;
3172 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003173 }
3174}
3175
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003176Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003177{
3178 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003179 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003180}
3181{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003182 expr = PrimaryExpr()
3183 (
3184 accessor = FieldAccessor(accessor != null ? accessor : expr)
3185 |
3186 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003187 )*
3188 {
3189 return accessor == null ? expr : accessor;
3190 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003191}
3192
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003193FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003194{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003195 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003196 String ident = null;
3197}
3198{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003199 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003200 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003201 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003202 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003203 }
3204}
3205
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003206AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003207{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003208 Token startToken = null;
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003209 boolean star = false, slice = false;
3210 Expression expr1 = null, expr2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003211}
3212{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003213 <LEFTBRACKET> { startToken = token; }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003214 (
3215 <MUL> { star = true; }
3216 |
3217 ( expr1 = Expression() ( <COLON> { slice = true; } ( expr2 = Expression() )? )? )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003218 )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003219 <RIGHTBRACKET>
3220 {
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003221 if (expr1 != null && expr1.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3222 ensureIntegerLiteral( (LiteralExpr) expr1, "Index");
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003223 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003224 if (expr2 != null && expr2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3225 ensureIntegerLiteral( (LiteralExpr) expr2, "Index");
3226 }
3227 AbstractAccessor resultAccessor;
3228 if (slice) {
3229 resultAccessor = new ListSliceExpression(inputExpr, expr1, expr2);
3230 } else if (star) {
3231 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.STAR, null);
3232 } else {
3233 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.ELEMENT, expr1);
3234 }
3235 return addSourceLocation(resultAccessor, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003236 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003237}
3238
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003239Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003240{
3241 Expression expr = null;
3242}
3243{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003244 (
3245 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07003246 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003247 | expr = Literal()
3248 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003249 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003250 | expr = ListConstructor()
3251 | expr = RecordConstructor()
3252 | expr = ParenthesizedExpression()
3253 )
3254 {
3255 return expr;
3256 }
3257}
3258
3259Expression Literal() throws ParseException:
3260{
3261 LiteralExpr lit = new LiteralExpr();
3262 String str = null;
3263}
3264{
3265 ( str = StringLiteral()
3266 {
3267 lit.setValue(new StringLiteral(str));
3268 }
3269 | <INTEGER_LITERAL>
3270 {
Till Westmann68c6a992016-09-30 00:19:12 -07003271 try {
3272 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
3273 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003274 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003275 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003276 }
3277 | <FLOAT_LITERAL>
3278 {
Till Westmann68c6a992016-09-30 00:19:12 -07003279 try {
3280 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
3281 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003282 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003283 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003284 }
3285 | <DOUBLE_LITERAL>
3286 {
Till Westmann68c6a992016-09-30 00:19:12 -07003287 try {
3288 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
3289 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003290 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003291 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003292 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07003293 | <MISSING>
3294 {
3295 lit.setValue(MissingLiteral.INSTANCE);
3296 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003297 | <NULL>
3298 {
3299 lit.setValue(NullLiteral.INSTANCE);
3300 }
3301 | <TRUE>
3302 {
3303 lit.setValue(TrueLiteral.INSTANCE);
3304 }
3305 | <FALSE>
3306 {
3307 lit.setValue(FalseLiteral.INSTANCE);
3308 }
3309 )
3310 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003311 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003312 }
3313}
3314
Yingyi Bu391f09e2015-10-29 13:49:39 -07003315VariableExpr VariableRef() throws ParseException:
3316{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003317 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003318}
3319{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003320 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003321 {
3322 Identifier ident = lookupSymbol(id);
3323 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003324 throw new SqlppParseException(getSourceLocation(token),
3325 "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 -07003326 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003327 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003328 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003329 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003330 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003331 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07003332 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003333 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003334 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003335 }
3336}
3337
Yingyi Bu391f09e2015-10-29 13:49:39 -07003338VariableExpr Variable() throws ParseException:
3339{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003340 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003341}
3342{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003343 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003344 {
3345 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003346 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
3347 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07003348 varExp.setIsNewVar(false);
3349 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003350 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003351 }
3352}
3353
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003354String VariableIdentifier() throws ParseException:
3355{
3356 String id = null;
3357}
3358{
3359 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3360 {
3361 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
3362 }
3363}
3364
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003365Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
3366{
3367 VariableExpr var = null;
3368 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
3369}
3370{
3371 var = Variable()
3372 ( LOOKAHEAD(1)
3373 {
3374 VariableExpr fieldVarExpr = null;
3375 String fieldIdentifierStr = null;
3376 }
3377 <LEFTPAREN>
3378 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3379 {
3380 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3381 }
3382 (<COMMA>
3383 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3384 {
3385 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3386 }
3387 )*
3388 <RIGHTPAREN>
3389 )?
3390 {
3391 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
3392 }
3393}
3394
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003395VariableExpr ExternalVariableRef() throws ParseException:
3396{
3397 String name = null;
3398}
3399{
3400 (
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08003401 <DOLLAR_IDENTIFIER> { name = token.image.substring(1); }
3402 | <DOLLAR_INTEGER_LITERAL> { name = token.image.substring(1); }
3403 | <DOLLAR_QUOTED_STRING> { name = removeQuotesAndEscapes(token.image.substring(1)); }
3404 | <QUES> { name = String.valueOf(++externalVarCounter); }
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003405 )
3406 {
3407 String idName = SqlppVariableUtil.toExternalVariableName(name);
3408 VarIdentifier id = new VarIdentifier(idName);
3409 VariableExpr varExp = new VariableExpr(id);
3410 return addSourceLocation(varExp, token);
3411 }
3412}
3413
Yingyi Bu391f09e2015-10-29 13:49:39 -07003414Expression ListConstructor() throws ParseException:
3415{
3416 Expression expr = null;
3417}
3418{
3419 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003420 expr = OrderedListConstructor() |
3421 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003422 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003423 {
3424 return expr;
3425 }
3426}
3427
Yingyi Bu391f09e2015-10-29 13:49:39 -07003428ListConstructor OrderedListConstructor() throws ParseException:
3429{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003430 Token startToken = null;
3431 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003432}
3433{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003434 <LEFTBRACKET> { startToken = token; }
3435 exprList = ExpressionList()
3436 <RIGHTBRACKET>
3437 {
3438 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003439 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003440 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003441}
3442
3443ListConstructor UnorderedListConstructor() throws ParseException:
3444{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003445 Token startToken = null;
3446 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003447}
3448{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003449 <LEFTDBLBRACE> { startToken = token; }
3450 exprList = ExpressionList()
3451 <RIGHTDBLBRACE>
3452 {
3453 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003454 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003455 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003456}
3457
3458List<Expression> ExpressionList() throws ParseException:
3459{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003460 Expression expr = null;
3461 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003462}
3463{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003464 (
3465 expr = Expression()
3466 {
3467 exprList.add(expr);
3468 }
3469 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07003470 {
3471 exprList.add(expr);
3472 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003473 )*
3474 )?
3475 {
3476 return exprList;
3477 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003478}
3479
Yingyi Bu391f09e2015-10-29 13:49:39 -07003480RecordConstructor RecordConstructor() throws ParseException:
3481{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003482 Token startToken = null;
3483 FieldBinding fb = null;
3484 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003485}
3486{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003487 <LEFTBRACE> { startToken = token; }
3488 (
3489 fb = FieldBinding() { fbList.add(fb); }
3490 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
3491 )?
3492 <RIGHTBRACE>
3493 {
3494 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003495 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003496 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003497}
3498
3499FieldBinding FieldBinding() throws ParseException:
3500{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003501 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003502}
3503{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003504 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003505 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003506 if (right == null) {
3507 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
3508 if (generatedIdentifier == null) {
3509 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
3510 }
3511 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07003512 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
3513 generatedNameExpr.setSourceLocation(left.getSourceLocation());
3514 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003515 } else {
3516 return new FieldBinding(left, right);
3517 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003518 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003519}
3520
Yingyi Bu391f09e2015-10-29 13:49:39 -07003521Expression FunctionCallExpr() throws ParseException:
3522{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003523 List<Expression> argList = new ArrayList<Expression>();
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003524 Expression argExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003525 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07003526 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003527 boolean distinct = false;
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003528 Expression filterExpr = null;
3529 WindowExpression windowExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003530}
3531{
3532 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003533 <LEFTPAREN> (
3534 ( <DISTINCT> { distinct = true; } )?
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003535 ( argExpr = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003536 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003537 if (star) {
3538 if (funcName.function.equalsIgnoreCase(BuiltinFunctions.SCALAR_COUNT.getName())) {
3539 argExpr = new LiteralExpr(new LongIntegerLiteral(1L));
3540 } else {
3541 throw new SqlppParseException(getSourceLocation(token),
3542 "The parameter * can only be used in " + BuiltinFunctions.SCALAR_COUNT.getName() + "().");
Yingyi Buf4d09842016-08-26 00:03:52 -07003543 }
Yingyi Buf4d09842016-08-26 00:03:52 -07003544 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003545 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003546 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003547 (<COMMA> argExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003548 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003549 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003550 }
3551 )*)? <RIGHTPAREN>
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003552
3553 {
3554 String name = funcName.function;
3555 if (distinct) {
3556 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
3557 }
3558 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
3559 int arity = argList.size();
3560 FunctionSignature signature = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
3561 if (signature == null) {
3562 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
3563 }
3564 }
3565
3566 ( <FILTER> <LEFTPAREN> <WHERE> filterExpr = Expression() <RIGHTPAREN> )?
3567
3568 ( LOOKAHEAD(5) windowExpr = WindowExpr(signature, argList, filterExpr) )?
3569
3570 {
3571 if (windowExpr != null) {
3572 return windowExpr;
3573 } else {
3574 CallExpr callExpr = new CallExpr(signature, argList, filterExpr);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06003575 if (funcName.hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003576 IExpressionAnnotation annotation = parseExpressionAnnotation(funcName.hintToken);
3577 if (annotation != null) {
3578 callExpr.addHint(annotation);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003579 }
3580 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003581 FunctionMapUtil.normalizedListInputFunctions(callExpr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003582 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003583 return callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003584 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003585 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003586}
3587
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003588WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr)
3589 throws ParseException:
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003590{
3591 Boolean fromLast = null, ignoreNulls = null;
3592}
3593{
3594 (
3595 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
3596 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
3597 <FROM> <IDENTIFIER>
3598 {
3599 if (isToken(FIRST)) {
3600 fromLast = false;
3601 } else if (isToken(LAST)) {
3602 fromLast = true;
3603 } else {
3604 throw createUnexpectedTokenError();
3605 }
3606 }
3607 )?
3608 (
3609 // ( RESPECT | IGNORE ) NULLS OVER
3610 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
3611 <IDENTIFIER>
3612 {
3613 if (isToken(RESPECT)) {
3614 ignoreNulls = false;
3615 } else if (isToken(IGNORE)) {
3616 ignoreNulls = true;
3617 } else {
3618 throw createUnexpectedTokenError();
3619 }
3620 }
3621 <IDENTIFIER>
3622 {
3623 if (!isToken(NULLS)) {
3624 throw createUnexpectedTokenError();
3625 }
3626 }
3627 )?
3628 <OVER>
3629 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003630 return OverClause(signature, argList, aggFilterExpr, token, fromLast, ignoreNulls);
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003631 }
3632}
3633
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003634WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr,
3635 Token startToken, Boolean fromLast, Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003636{
3637 Expression partitionExpr = null;
3638 List<Expression> partitionExprs = new ArrayList<Expression>();
3639 OrderbyClause orderByClause = null;
3640 List<Expression> orderbyList = null;
3641 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
3642 WindowExpression.FrameMode frameMode = null;
3643 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
3644 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
3645 Expression frameStartExpr = null, frameEndExpr = null;
3646 WindowExpression.FrameExclusionKind frameExclusionKind = null;
3647 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
3648 VariableExpr windowVar = null;
3649 List<Pair<Expression, Identifier>> windowFieldList = null;
3650}
3651{
3652 (
3653 windowVarWithFieldList = VariableWithFieldMap() <AS>
3654 {
3655 windowVar = windowVarWithFieldList.first;
3656 windowFieldList = windowVarWithFieldList.second;
3657 }
3658 )?
3659 <LEFTPAREN>
3660 (
3661 <IDENTIFIER> { expectToken(PARTITION); } <BY>
3662 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
3663 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
3664 )?
3665 (
3666 orderByClause = OrderbyClause()
3667 {
3668 orderbyList = orderByClause.getOrderbyList();
3669 orderbyModifierList = orderByClause.getModifierList();
3670 }
3671 (
3672 frameMode = WindowFrameMode()
3673 (
3674 frameStart = WindowFrameBoundary() |
3675 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
3676 )
3677 ( frameExclusionKind = WindowFrameExclusion() )?
3678 {
3679 frameStartKind = frameStart.first;
3680 frameStartExpr = frameStart.second;
3681 if (frameEnd == null) {
3682 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3683 } else {
3684 frameEndKind = frameEnd.first;
3685 frameEndExpr = frameEnd.second;
3686 }
3687 if (frameExclusionKind == null) {
3688 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
3689 }
3690 }
3691 )?
3692 )?
3693 <RIGHTPAREN>
3694 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003695 WindowExpression winExpr = new WindowExpression(signature, argList, aggFilterExpr, partitionExprs, orderbyList,
3696 orderbyModifierList, frameMode, frameStartKind, frameStartExpr, frameEndKind, frameEndExpr, frameExclusionKind,
3697 windowVar, windowFieldList, ignoreNulls, fromLast);
3698 return addSourceLocation(winExpr, startToken);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003699 }
3700}
3701
3702WindowExpression.FrameMode WindowFrameMode() throws ParseException:
3703{
3704}
3705{
3706 <IDENTIFIER>
3707 {
3708 if (isToken(RANGE)) {
3709 return WindowExpression.FrameMode.RANGE;
3710 } else if (isToken(ROWS)) {
3711 return WindowExpression.FrameMode.ROWS;
3712 } else if (isToken(GROUPS)) {
3713 return WindowExpression.FrameMode.GROUPS;
3714 } else {
3715 throw createUnexpectedTokenError();
3716 }
3717 }
3718}
3719
3720Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
3721{
3722 boolean current = false;
3723 Expression expr = null;
3724}
3725{
3726 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003727 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003728 | expr = Expression()
3729 )
3730 <IDENTIFIER>
3731 {
3732 WindowExpression.FrameBoundaryKind kind;
3733 if (current && isToken(ROW)) {
3734 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3735 } else if (!current && isToken(PRECEDING)) {
3736 kind = expr == null
3737 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
3738 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
3739 } else if (!current && isToken(FOLLOWING)) {
3740 kind = expr == null
3741 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
3742 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
3743 } else {
3744 throw createUnexpectedTokenError();
3745 }
3746 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
3747 }
3748}
3749
3750WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
3751{
3752 boolean current = false, no = false;
3753}
3754{
3755 <IDENTIFIER>
3756 {
3757 expectToken(EXCLUDE);
3758 }
3759 (
3760 <GROUP>
3761 {
3762 return WindowExpression.FrameExclusionKind.GROUP;
3763 }
3764 |
3765 (
3766 <IDENTIFIER>
3767 {
3768 if (isToken(TIES)) {
3769 return WindowExpression.FrameExclusionKind.TIES;
3770 } else if (isToken(CURRENT)) {
3771 current = true;
3772 } else if (isToken(NO)) {
3773 no = true;
3774 } else {
3775 throw createUnexpectedTokenError();
3776 }
3777 }
3778 <IDENTIFIER>
3779 {
3780 if (current && isToken(ROW)) {
3781 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
3782 } else if (no && isToken(OTHERS)) {
3783 return WindowExpression.FrameExclusionKind.NO_OTHERS;
3784 } else {
3785 throw createUnexpectedTokenError();
3786 }
3787 }
3788 )
3789 )
3790}
3791
Yingyi Bu391f09e2015-10-29 13:49:39 -07003792Expression ParenthesizedExpression() throws ParseException:
3793{
3794 Expression expr;
3795}
3796{
3797 (
3798 LOOKAHEAD(2)
3799 <LEFTPAREN> expr = Expression() <RIGHTPAREN>
3800 |
3801 expr = Subquery()
3802 )
3803 {
3804 return expr;
3805 }
3806}
3807
Yingyi Buc8c067c2016-07-25 23:37:19 -07003808Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003809{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003810 Token startToken = null;
3811 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003812 List<Expression> whenExprs = new ArrayList<Expression>();
3813 List<Expression> thenExprs = new ArrayList<Expression>();
3814 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003815 Expression whenExpr = null;
3816 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003817}
3818{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003819 <CASE> { startToken = token; }
3820 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07003821 (
3822 <WHEN> whenExpr = Expression()
3823 {
3824 whenExprs.add(whenExpr);
3825 }
3826 <THEN> thenExpr = Expression()
3827 {
3828 thenExprs.add(thenExpr);
3829 }
3830 )*
3831 (<ELSE> elseExpr = Expression() )?
3832 <END>
3833 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003834 if (conditionExpr == null) {
3835 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003836 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003837 }
3838 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003839 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07003840 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003841}
3842
Yingyi Buab817482016-08-19 21:29:31 -07003843SelectExpression SelectExpression(boolean subquery) throws ParseException:
3844{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003845 List<LetClause> letClauses = new ArrayList<LetClause>();
3846 SelectSetOperation selectSetOperation;
3847 OrderbyClause orderbyClause = null;
3848 LimitClause limitClause = null;
3849 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07003850}
3851{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003852 ( letClauses = LetClause() )?
3853 selectSetOperation = SelectSetOperation()
3854 (orderbyClause = OrderbyClause() {})?
3855 (limitClause = LimitClause() {})?
3856 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003857 SelectExpression selectExpr =
3858 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
3859 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
3860 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003861 }
3862}
3863
Yingyi Buab817482016-08-19 21:29:31 -07003864SelectSetOperation SelectSetOperation() throws ParseException:
3865{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003866 SetOperationInput setOperationInputLeft;
3867 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
3868}
3869{
3870 {
3871 SelectBlock selectBlockLeft = null;
3872 SelectExpression subqueryLeft = null;
3873 Expression expr = null;
3874 }
3875 selectBlockLeft = SelectBlock()
3876 {
3877 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
3878 }
3879 (
3880 {
3881 SetOpType opType = SetOpType.UNION;
3882 boolean setSemantics = true;
3883 SelectBlock selectBlockRight = null;
3884 SelectExpression subqueryRight = null;
3885 }
3886 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
3887 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
3888 {
3889 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
3890 }
3891 )*
3892 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003893 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
3894 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
3895 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003896 }
3897}
3898
Yingyi Buab817482016-08-19 21:29:31 -07003899SelectExpression Subquery() throws ParseException:
3900{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003901 SelectExpression selectExpr = null;
3902}
3903{
3904 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
3905 {
3906 return selectExpr;
3907 }
3908}
3909
Yingyi Buab817482016-08-19 21:29:31 -07003910SelectBlock SelectBlock() throws ParseException:
3911{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003912 SelectClause selectClause = null;
3913 FromClause fromClause = null;
3914 List<LetClause> fromLetClauses = null;
3915 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003916 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003917 GroupbyClause groupbyClause = null;
3918 List<LetClause> gbyLetClauses = null;
3919 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003920 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003921 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003922}
3923{
3924 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003925 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003926 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003927 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07003928 fromClause = FromClause()
3929 (
3930 fromLetClauses = LetClause()
3931 )?
3932 )
3933 |
3934 (
3935 fromLetClauses = LetClause()
3936 {
3937 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
3938 SourceLocation sourceLoc = getSourceLocation(token);
3939 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
3940 missingExpr.setSourceLocation(sourceLoc);
3941 List<Expression> list = new ArrayList<Expression>(1);
3942 list.add(missingExpr);
3943 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
3944 listExpr.setSourceLocation(sourceLoc);
3945 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
3946 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
3947 fromVar.setSourceLocation(sourceLoc);
3948 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
3949 fromClause = new FromClause(fromTerms);
3950 }
3951 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003952 )?
3953 (whereClause = WhereClause())?
3954 (
3955 groupbyClause = GroupbyClause()
3956 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003957 gbyLetClauses = LetClause()
3958 )?
3959 (havingClause = HavingClause())?
3960 )?
3961 |
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003962 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003963 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003964 fromLetClauses = LetClause()
3965 )?
3966 (whereClause = WhereClause())?
3967 (
3968 groupbyClause = GroupbyClause()
3969 (
3970 gbyLetClauses = LetClause()
3971 )?
3972 (havingClause = HavingClause())?
3973 )?
3974 selectClause = SelectClause()
3975 )
3976 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003977 if (fromLetClauses != null) {
3978 fromLetWhereClauses.addAll(fromLetClauses);
3979 }
3980 if (whereClause != null) {
3981 fromLetWhereClauses.add(whereClause);
3982 }
3983 if (gbyLetClauses != null) {
3984 gbyLetHavingClauses.addAll(gbyLetClauses);
3985 }
3986 if (havingClause != null) {
3987 gbyLetHavingClauses.add(havingClause);
3988 }
3989 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
3990 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003991 selectBlock.setSourceLocation(startSrcLoc);
3992 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003993 }
3994}
3995
Yingyi Buab817482016-08-19 21:29:31 -07003996SelectClause SelectClause() throws ParseException:
3997{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003998 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003999 SelectRegular selectRegular = null;
4000 SelectElement selectElement = null;
4001 boolean distinct = false;
4002}
4003{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004004 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004005 (
4006 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04004007 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07004008 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07004009 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004010 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004011 SourceLocation sourceLoc = getSourceLocation(startToken);
4012 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07004013 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004014 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004015 List<Projection> projections = new ArrayList<Projection>();
4016 projections.add(projection);
4017 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004018 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004019 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004020 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
4021 selectClause.setSourceLocation(sourceLoc);
4022 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004023 }
4024}
4025
Yingyi Buab817482016-08-19 21:29:31 -07004026SelectRegular SelectRegular() throws ParseException:
4027{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004028 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004029 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004030 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004031}
4032{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004033 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004034 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004035 projections.add(projection);
4036 startSrcLoc = projection.getSourceLocation();
4037 }
4038 ( LOOKAHEAD(2) <COMMA> projection = Projection()
4039 {
4040 projections.add(projection);
4041 }
4042 )*
4043 {
4044 SelectRegular selectRegular = new SelectRegular(projections);
4045 selectRegular.setSourceLocation(startSrcLoc);
4046 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004047 }
4048}
4049
Yingyi Buab817482016-08-19 21:29:31 -07004050SelectElement SelectElement() throws ParseException:
4051{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004052 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004053 Expression expr = null;
4054 String name = null;
4055}
4056{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004057 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004058 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004059 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004060 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004061 }
4062}
4063
Yingyi Buab817482016-08-19 21:29:31 -07004064Projection Projection() throws ParseException :
4065{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004066 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004067 Expression expr = null;
4068 Identifier identifier = null;
4069 String name = null;
4070 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004071 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004072}
4073{
4074 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004075 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
4076 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004077 | expr = Expression() ((<AS>)? name = Identifier())?
4078 {
4079 if (name == null) {
4080 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
4081 if (generatedColumnIdentifier != null) {
4082 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
4083 }
4084 }
4085 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004086 )
4087 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004088 Projection projection = new Projection(expr, name, star, varStar);
4089 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
4090 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004091 }
4092}
4093
4094FromClause FromClause() throws ParseException :
4095{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004096 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004097 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
4098 extendCurrentScope();
4099}
4100{
4101 {
4102 FromTerm fromTerm = null;
4103 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004104 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004105 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
4106 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004107 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004108 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004109 }
4110}
4111
4112FromTerm FromTerm() throws ParseException :
4113{
4114 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004115 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004116 VariableExpr posVar = null;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004117 AbstractBinaryCorrelateClause correlateClause = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004118 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
4119}
4120{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004121 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004122 (
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004123 (
4124 correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER)
4125 | ( <INNER> correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER) )
4126 | ( <LEFT> ( <OUTER> )? correlateClause = JoinOrUnnestClause(JoinType.LEFTOUTER, UnnestType.LEFTOUTER) )
4127 | ( <RIGHT> ( <OUTER> )? correlateClause = JoinClause(JoinType.RIGHTOUTER) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004128 )
4129 {
4130 correlateClauses.add(correlateClause);
4131 }
4132 )*
4133 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004134 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07004135 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004136 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004137 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
4138 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
4139 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004140 }
4141}
4142
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004143AbstractBinaryCorrelateClause JoinOrUnnestClause(JoinType joinType, UnnestType unnestType) throws ParseException :
4144{
4145 AbstractBinaryCorrelateClause correlateClause = null;
4146}
4147{
4148 ( correlateClause = JoinClause(joinType) | correlateClause = UnnestClause(unnestType) )
4149 {
4150 return correlateClause;
4151 }
4152}
4153
Yingyi Bu391f09e2015-10-29 13:49:39 -07004154JoinClause JoinClause(JoinType joinType) throws ParseException :
4155{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004156 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004157 Expression rightExpr = null;
4158 VariableExpr rightVar = null;
4159 VariableExpr posVar = null;
4160 Expression conditionExpr = null;
4161}
4162{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004163 <JOIN> { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())? <ON> conditionExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004164 {
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004165 if (rightVar == null) {
4166 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004167 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004168 JoinClause joinClause = new JoinClause(joinType, rightExpr, rightVar, posVar, conditionExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004169 return addSourceLocation(joinClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004170 }
4171}
4172
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004173UnnestClause UnnestClause(UnnestType unnestType) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004174{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004175 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004176 Expression rightExpr;
4177 VariableExpr rightVar;
4178 VariableExpr posVar = null;
4179}
4180{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004181 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable()) (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004182 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004183 if (rightVar == null) {
4184 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004185 }
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004186 UnnestClause unnestClause = new UnnestClause(unnestType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004187 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004188 }
4189}
4190
Yingyi Bu391f09e2015-10-29 13:49:39 -07004191List<LetClause> LetClause() throws ParseException:
4192{
4193 List<LetClause> letList = new ArrayList<LetClause>();
4194 LetClause letClause;
4195}
4196{
4197 (
4198 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
4199 |
4200 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
4201 )
4202 {
4203 return letList;
4204 }
4205}
4206
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004207WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004208{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004209 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004210 Expression whereExpr;
4211}
4212{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004213 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004214 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004215 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004216 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004217 }
4218}
4219
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004220OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004221{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004222 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004223 OrderbyClause oc = new OrderbyClause();
4224 Expression orderbyExpr;
4225 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004226 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004227 int numOfOrderby = 0;
4228}
4229{
4230 <ORDER>
4231 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004232 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004233 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
4234 if (hintToken != null) {
4235 switch (hintToken.hint) {
4236 case INMEMORY_HINT:
4237 String[] splits = hintToken.hintParams.split("\\s+");
4238 int numFrames = Integer.parseInt(splits[0]);
4239 int numTuples = Integer.parseInt(splits[1]);
4240 oc.setNumFrames(numFrames);
4241 oc.setNumTuples(numTuples);
4242 break;
4243 case RANGE_HINT:
4244 try {
Dmitry Lychagin70bf47a2020-10-22 13:25:57 -07004245 Expression rangeExpr = parseExpression(hintToken.hintParams);
4246 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
4247 oc.setRangeMap(rangeMap);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004248 } catch (CompilationException e) {
4249 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
4250 }
4251 break;
Ali Alsuliman80225e22018-10-15 14:17:07 -07004252 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004253 }
4254 }
4255 <BY> orderbyExpr = Expression()
4256 {
4257 orderbyList.add(orderbyExpr);
4258 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
4259 }
4260 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
4261 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
4262 {
4263 modifierList.add(modif);
4264 }
4265
4266 (LOOKAHEAD(2) <COMMA> orderbyExpr = Expression()
4267 {
4268 orderbyList.add(orderbyExpr);
4269 modif = OrderbyClause.OrderModifier.ASC;
4270 }
4271 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
4272 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
4273 {
4274 modifierList.add(modif);
4275 }
4276 )*
4277
4278 {
4279 oc.setModifierList(modifierList);
4280 oc.setOrderbyList(orderbyList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004281 return addSourceLocation(oc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004282 }
4283}
4284
4285GroupbyClause GroupbyClause()throws ParseException :
4286{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004287 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004288 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004289 List<List<GbyVariableExpressionPair>> gbyList = null;
4290 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004291 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07004292 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004293 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004294}
4295{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004296 {
4297 Scope newScope = extendCurrentScopeNoPush(true);
4298 // extendCurrentScope(true);
4299 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004300 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004301 {
4302 startToken = token;
4303 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
4304 if (hintToken != null) {
4305 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004306 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004307 }
4308 <BY> groupingElementList = GroupingElementList()
4309 (
4310 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004311 {
4312 groupVar = groupVarWithFieldList.first;
4313 groupFieldList = groupVarWithFieldList.second;
4314 }
Yingyi Buacc12a92016-03-26 17:25:05 -07004315 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004316 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004317 if (groupingSetsParser == null) {
4318 groupingSetsParser = new SqlppGroupingSetsParser();
4319 }
4320 SourceLocation sourceLoc = getSourceLocation(startToken);
4321 try {
4322 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
4323 } catch (CompilationException e) {
4324 throw new SqlppParseException(sourceLoc, e.getMessage());
4325 }
4326 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004327 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07004328 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07004329 gbc.setGroupVar(groupVar);
4330 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004331 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004332 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004333 }
4334}
4335
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004336List<GroupingElement> GroupingElementList() throws ParseException:
4337{
4338 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
4339 GroupingElement groupingElement = null;
4340}
4341{
4342 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
4343 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
4344 {
4345 return groupingElementList;
4346 }
4347}
4348
4349GroupingElement GroupingElement() throws ParseException:
4350{
4351 GroupingElement groupingElement = null;
4352 List<GroupingSet> groupingSets = null;
4353 List<GroupingElement> groupingElements = null;
4354}
4355{
4356 (
4357 LOOKAHEAD(2)
4358 <LEFTPAREN> <RIGHTPAREN>
4359 {
4360 groupingElement = GroupingSet.EMPTY;
4361 }
4362 |
4363 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
4364 <IDENTIFIER> { expectToken(ROLLUP); }
4365 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4366 {
4367 groupingElement = new RollupCube(groupingSets, false);
4368 }
4369 |
4370 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
4371 <IDENTIFIER> { expectToken(CUBE); }
4372 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4373 {
4374 groupingElement = new RollupCube(groupingSets, true);
4375 }
4376 |
4377 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
4378 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
4379 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
4380 {
4381 groupingElement = new GroupingSets(groupingElements);
4382 }
4383 |
4384 groupingElement = OrdinaryGroupingSet()
4385 )
4386 {
4387 return groupingElement;
4388 }
4389}
4390
4391GroupingSet OrdinaryGroupingSet() throws ParseException:
4392{
4393 GbyVariableExpressionPair gbyExprPair = null;
4394 List<GbyVariableExpressionPair> items = null;
4395}
4396{
4397 (
4398 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
4399 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
4400 )
4401 {
4402 return new GroupingSet(items);
4403 }
4404}
4405
4406List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
4407{
4408 GroupingSet groupingSet = null;
4409 List<GroupingSet> items = new ArrayList<GroupingSet>();
4410}
4411{
4412 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
4413 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
4414 {
4415 return items;
4416 }
4417}
4418
4419List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
4420{
4421 GbyVariableExpressionPair gbyExprPair = null;
4422 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
4423}
4424{
4425 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
4426 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
4427 {
4428 return items;
4429 }
4430}
4431
4432GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
4433{
4434 Expression expr = null;
4435 VariableExpr var = null;
4436}
4437{
4438 expr = Expression() ( (<AS>)? var = Variable() )?
4439 {
4440 return new GbyVariableExpressionPair(var, expr);
4441 }
4442}
4443
Yingyi Bu391f09e2015-10-29 13:49:39 -07004444HavingClause HavingClause() throws ParseException:
4445{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004446 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004447 Expression filterExpr = null;
4448}
4449{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004450 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004451 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004452 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004453 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004454 }
4455}
4456
4457LimitClause LimitClause() throws ParseException:
4458{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004459 Token startToken = null;
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08004460 Expression limitExpr = null, offsetExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004461}
4462{
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08004463 (
4464 (
4465 <LIMIT> { startToken = token; pushForbiddenScope(getCurrentScope()); } limitExpr = Expression()
4466 ( <OFFSET> offsetExpr = Expression() )?
4467 { popForbiddenScope(); }
4468 )
4469 |
4470 (
4471 <OFFSET> { startToken = token; pushForbiddenScope(getCurrentScope()); } offsetExpr = Expression()
4472 { popForbiddenScope(); }
4473 )
4474 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004475 {
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08004476 LimitClause lc = new LimitClause(limitExpr, offsetExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004477 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004478 }
4479}
4480
4481QuantifiedExpression QuantifiedExpression()throws ParseException:
4482{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004483 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004484 QuantifiedExpression qc = new QuantifiedExpression();
4485 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
4486 Expression satisfiesExpr;
4487 VariableExpr var;
4488 Expression inExpr;
4489 QuantifiedPair pair;
4490}
4491{
4492 {
4493 createNewScope();
4494 }
4495
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004496 ( ((<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); })
4497 | (<EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); }))
Yingyi Bu391f09e2015-10-29 13:49:39 -07004498 var = Variable() <IN> inExpr = Expression()
4499 {
4500 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004501 quantifiedList.add(pair);
4502 }
4503 (
4504 <COMMA> var = Variable() <IN> inExpr = Expression()
4505 {
4506 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004507 quantifiedList.add(pair);
4508 }
4509 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07004510 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004511 {
4512 qc.setSatisfiesExpr(satisfiesExpr);
4513 qc.setQuantifiedList(quantifiedList);
4514 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004515 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004516 }
4517}
4518
4519LetClause LetElement() throws ParseException:
4520{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004521 VariableExpr varExp;
4522 Expression beExp;
4523 extendCurrentScope();
4524}
4525{
4526 varExp = Variable() <EQ> beExp = Expression()
4527 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004528 LetClause lc = new LetClause(varExp, beExp);
4529 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004530 return lc;
4531 }
4532}
4533
4534LetClause WithElement() throws ParseException:
4535{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004536 VariableExpr varExp;
4537 Expression beExp;
4538 extendCurrentScope();
4539}
4540{
4541 varExp = Variable() <AS> beExp = Expression()
4542 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004543 LetClause lc = new LetClause(varExp, beExp);
4544 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004545 return lc;
4546 }
4547}
4548
4549TOKEN_MGR_DECLS:
4550{
4551 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07004552 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004553 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004554
4555 public void pushState() {
4556 lexerStateStack.push( curLexState );
4557 }
4558
4559 public void popState(String token) {
4560 if (lexerStateStack.size() > 0) {
4561 SwitchTo( lexerStateStack.pop() );
4562 } else {
4563 int errorLine = input_stream.getEndLine();
4564 int errorColumn = input_stream.getEndColumn();
4565 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
4566 + "\" but state stack is empty.";
4567 throw new TokenMgrError(msg, -1);
4568 }
4569 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004570
4571 void CommonTokenAction(Token token) {
4572 Token hintToken = token.specialToken;
4573 if (hintToken != null) {
4574 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
4575 String text = hintToken.image.substring(1).trim();
4576 boolean hintFound = hintToken.parseHint(text);
4577 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
4578 }
4579 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004580}
4581
4582<DEFAULT,IN_DBL_BRACE>
4583TOKEN [IGNORE_CASE]:
4584{
Ian Maxon38fe9402020-01-29 19:27:40 -08004585 <ADAPTER: "adapter">
4586 | <ALL : "all">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004587 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07004588 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004589 | <APPLY : "apply">
4590 | <AS : "as">
4591 | <ASC : "asc">
4592 | <AT : "at">
4593 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004594 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004595 | <BTREE : "btree">
4596 | <BY : "by">
4597 | <CASE : "case">
4598 | <CLOSED : "closed">
4599 | <CREATE : "create">
Dmitry Lychagin0ebc4202021-01-25 13:12:41 -08004600 | <COMPACTION : "compaction"> // no longer used
Yingyi Bu391f09e2015-10-29 13:49:39 -07004601 | <COMPACT : "compact">
4602 | <CONNECT : "connect">
4603 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07004604 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07004605 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07004606 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004607 | <DECLARE : "declare">
4608 | <DEFINITION : "definition">
4609 | <DELETE : "delete">
4610 | <DESC : "desc">
4611 | <DISCONNECT : "disconnect">
4612 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004613 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004614 | <DROP : "drop">
4615 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07004616 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004617 | <ELSE : "else">
4618 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07004619 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004620 | <EVERY : "every">
4621 | <EXCEPT : "except">
4622 | <EXISTS : "exists">
4623 | <EXTERNAL : "external">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004624 | <FALSE : "false">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004625 | <FEED : "feed">
4626 | <FILTER : "filter">
4627 | <FLATTEN : "flatten">
4628 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004629 | <FROM : "from">
4630 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08004631 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004632 | <FUNCTION : "function">
4633 | <GROUP : "group">
4634 | <HAVING : "having">
4635 | <HINTS : "hints">
4636 | <IF : "if">
4637 | <INTO : "into">
4638 | <IN : "in">
4639 | <INDEX : "index">
4640 | <INGESTION : "ingestion">
4641 | <INNER : "inner">
4642 | <INSERT : "insert">
4643 | <INTERNAL : "internal">
4644 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07004645 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004646 | <JOIN : "join">
4647 | <KEYWORD : "keyword">
4648 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07004649 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004650 | <LEFT : "left">
4651 | <LETTING : "letting">
4652 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004653 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004654 | <LIMIT : "limit">
4655 | <LOAD : "load">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004656 | <MISSING : "missing">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004657 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004658 | <NODEGROUP : "nodegroup">
4659 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07004660 | <NOT : "not">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004661 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004662 | <OFFSET : "offset">
4663 | <ON : "on">
4664 | <OPEN : "open">
4665 | <OR : "or">
4666 | <ORDER : "order">
4667 | <OUTER : "outer">
4668 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004669 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004670 | <PATH : "path">
4671 | <POLICY : "policy">
4672 | <PRESORTED : "pre-sorted">
4673 | <PRIMARY : "primary">
4674 | <RAW : "raw">
4675 | <REFRESH : "refresh">
4676 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004677 | <RETURNING : "returning">
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004678 | <RIGHT : "right">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004679 | <RTREE : "rtree">
4680 | <RUN : "run">
4681 | <SATISFIES : "satisfies">
4682 | <SECONDARY : "secondary">
4683 | <SELECT : "select">
4684 | <SET : "set">
4685 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08004686 | <START : "start">
4687 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08004688 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03004689 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07004690 | <THEN : "then">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004691 | <TO : "to">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004692 | <TRUE : "true">
4693 | <TYPE : "type">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004694 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08004695 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004696 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07004697 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004698 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07004699 | <USE : "use">
4700 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004701 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08004702 | <VALUED : "valued">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004703 | <WHEN : "when">
4704 | <WHERE : "where">
4705 | <WITH : "with">
4706 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004707}
4708
4709<DEFAULT,IN_DBL_BRACE>
4710TOKEN :
4711{
4712 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07004713 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004714 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004715 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004716 | <MUL : "*">
4717 | <PLUS : "+">
4718
4719 | <LEFTPAREN : "(">
4720 | <RIGHTPAREN : ")">
4721 | <LEFTBRACKET : "[">
4722 | <RIGHTBRACKET : "]">
4723
4724 | <ATT : "@">
4725 | <COLON : ":">
4726 | <COMMA : ",">
4727 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004728 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004729 | <QUES : "?">
4730 | <SEMICOLON : ";">
4731 | <SHARP : "#">
4732
4733 | <LT : "<">
4734 | <GT : ">">
4735 | <LE : "<=">
4736 | <GE : ">=">
4737 | <EQ : "=">
4738 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07004739 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004740 | <SIMILAR : "~=">
4741}
4742
4743<DEFAULT,IN_DBL_BRACE>
4744TOKEN :
4745{
4746 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
4747}
4748
4749<DEFAULT>
4750TOKEN :
4751{
4752 <RIGHTBRACE : "}"> { popState("}"); }
4753}
4754
4755<DEFAULT,IN_DBL_BRACE>
4756TOKEN :
4757{
4758 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
4759}
4760
4761<IN_DBL_BRACE>
4762TOKEN :
4763{
4764 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
4765}
4766
4767<DEFAULT,IN_DBL_BRACE>
4768TOKEN :
4769{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004770 <#DIGIT : ["0" - "9"]>
4771}
4772
4773<DEFAULT,IN_DBL_BRACE>
4774TOKEN:
4775{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004776 <INTEGER_LITERAL : <DIGITS> >
4777 | <DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07004778 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
4779 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004780 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004781 | <FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
Yingyi Bue4d919e2016-10-30 10:47:03 -07004782 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
4783 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004784 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004785 | <#DIGITS : (<DIGIT>)+ >
Yingyi Bu391f09e2015-10-29 13:49:39 -07004786}
4787
4788<DEFAULT,IN_DBL_BRACE>
4789TOKEN :
4790{
4791 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004792 | <#IDENTIFIER_START_SPECIALCHAR : ["_"]>
4793 | <#IDENTIFIER_REST_SPECIALCHAR : ["$"]>
4794 | <#IDENTIFIER_START : <LETTER> | <IDENTIFIER_START_SPECIALCHAR> >
4795 | <#IDENTIFIER_REST : <LETTER> | <DIGIT> | <IDENTIFIER_START_SPECIALCHAR> | <IDENTIFIER_REST_SPECIALCHAR> >
4796 | <IDENTIFIER : <IDENTIFIER_START> (<IDENTIFIER_REST>)* >
Yingyi Bu391f09e2015-10-29 13:49:39 -07004797}
4798
4799<DEFAULT,IN_DBL_BRACE>
4800TOKEN :
4801{
4802 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07004803 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004804 <EscapeQuot>
4805 | <EscapeBslash>
4806 | <EscapeSlash>
4807 | <EscapeBspace>
4808 | <EscapeFormf>
4809 | <EscapeNl>
4810 | <EscapeCr>
4811 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004812 | ~["`","\\"])* "`">
4813 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004814 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004815 | <EscapeBslash>
4816 | <EscapeSlash>
4817 | <EscapeBspace>
4818 | <EscapeFormf>
4819 | <EscapeNl>
4820 | <EscapeCr>
4821 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004822 | ~["\"","\\"])* "\"")
4823 | ("\'"(
4824 <EscapeApos>
4825 | <EscapeBslash>
4826 | <EscapeSlash>
4827 | <EscapeBspace>
4828 | <EscapeFormf>
4829 | <EscapeNl>
4830 | <EscapeCr>
4831 | <EscapeTab>
4832 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004833 | < #EscapeQuot: "\\\"" >
4834 | < #EscapeApos: "\\\'" >
4835 | < #EscapeBslash: "\\\\" >
4836 | < #EscapeSlash: "\\/" >
4837 | < #EscapeBspace: "\\b" >
4838 | < #EscapeFormf: "\\f" >
4839 | < #EscapeNl: "\\n" >
4840 | < #EscapeCr: "\\r" >
4841 | < #EscapeTab: "\\t" >
4842}
4843
4844<DEFAULT,IN_DBL_BRACE>
4845TOKEN :
4846{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004847 <DOLLAR_INTEGER_LITERAL : "$" <INTEGER_LITERAL> >
4848 | <DOLLAR_IDENTIFIER : "$" <IDENTIFIER> >
4849 | <DOLLAR_QUOTED_STRING: "$" <QUOTED_STRING> >
Yingyi Bu391f09e2015-10-29 13:49:39 -07004850}
4851
4852<DEFAULT,IN_DBL_BRACE>
4853SKIP:
4854{
4855 " "
4856 | "\t"
4857 | "\r"
4858 | "\n"
4859}
4860
4861<DEFAULT,IN_DBL_BRACE>
4862SKIP:
4863{
4864 <"//" (~["\n"])* "\n">
4865}
4866
4867<DEFAULT,IN_DBL_BRACE>
4868SKIP:
4869{
4870 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4871}
4872
4873<DEFAULT,IN_DBL_BRACE>
4874SKIP:
4875{
Yingyi Bu93846a72016-09-13 16:30:39 -07004876 <"--" (~["\n"])* "\n">
4877}
4878
4879
4880<DEFAULT,IN_DBL_BRACE>
4881SKIP:
4882{
4883 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4884}
4885
4886<DEFAULT,IN_DBL_BRACE>
4887SKIP:
4888{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004889 <"/*"> { pushState(); } : INSIDE_COMMENT
4890}
4891
4892<INSIDE_COMMENT>
4893SPECIAL_TOKEN:
4894{
4895 <"+"(" ")*(~["*"])*>
4896}
4897
4898<INSIDE_COMMENT>
4899SKIP:
4900{
4901 <"/*"> { pushState(); }
4902}
4903
4904<INSIDE_COMMENT>
4905SKIP:
4906{
4907 <"*/"> { popState("*/"); }
4908 | <~[]>
4909}