blob: b23492a885e12424ff2d95fb5de88c31fe26b08b [file] [log] [blame]
Michael Blow82464fb2017-03-28 18:48:13 -04001//
2// Licensed to the Apache Software Foundation (ASF) under one
3// or more contributor license agreements. See the NOTICE file
4// distributed with this work for additional information
5// regarding copyright ownership. The ASF licenses this file
6// to you under the Apache License, Version 2.0 (the
7// "License"); you may not use this file except in compliance
8// with the License. You may obtain a copy of the License at
9//
10// http://www.apache.org/licenses/LICENSE-2.0
11//
12// Unless required by applicable law or agreed to in writing,
13// software distributed under the License is distributed on an
14// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15// KIND, either express or implied. See the License for the
16// specific language governing permissions and limitations
17// under the License.
18//
Yingyi Bu391f09e2015-10-29 13:49:39 -070019options {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -070020 COMMON_TOKEN_ACTION = true;
21 STATIC = false;
22 TOKEN_EXTENDS = "org.apache.asterix.lang.sqlpp.parser.SqlppToken";
Yingyi Bu391f09e2015-10-29 13:49:39 -070023}
24
Yingyi Bu391f09e2015-10-29 13:49:39 -070025PARSER_BEGIN(SQLPPParser)
26
27package org.apache.asterix.lang.sqlpp.parser;
28
29// For SQL++ ParserTokenManager
Till Westmanne9b2adf2016-10-15 12:39:01 -070030import java.util.ArrayDeque;
Yingyi Bu391f09e2015-10-29 13:49:39 -070031
32import java.io.BufferedReader;
33import java.io.File;
34import java.io.FileInputStream;
35import java.io.FileNotFoundException;
36import java.io.IOException;
37import java.io.InputStreamReader;
38import java.io.Reader;
39import java.io.StringReader;
40import java.util.ArrayList;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070041import java.util.Arrays;
42import java.util.Collection;
Yingyi Budaa549c2016-06-28 22:30:52 -070043import java.util.Collections;
Yingyi Bu391f09e2015-10-29 13:49:39 -070044import java.util.HashMap;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070045import java.util.HashSet;
Yingyi Bu391f09e2015-10-29 13:49:39 -070046import java.util.Iterator;
47import java.util.LinkedHashMap;
48import java.util.List;
49import java.util.Map;
50
51import org.apache.asterix.common.annotations.AutoDataGen;
52import org.apache.asterix.common.annotations.DateBetweenYearsDataGen;
53import org.apache.asterix.common.annotations.DatetimeAddRandHoursDataGen;
54import org.apache.asterix.common.annotations.DatetimeBetweenYearsDataGen;
55import org.apache.asterix.common.annotations.FieldIntervalDataGen;
56import org.apache.asterix.common.annotations.FieldValFileDataGen;
57import org.apache.asterix.common.annotations.FieldValFileSameIndexDataGen;
58import org.apache.asterix.common.annotations.IRecordFieldDataGen;
59import org.apache.asterix.common.annotations.InsertRandIntDataGen;
60import org.apache.asterix.common.annotations.ListDataGen;
61import org.apache.asterix.common.annotations.ListValFileDataGen;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -060062import org.apache.asterix.common.annotations.RangeAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070063import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
64import org.apache.asterix.common.annotations.TypeDataGen;
65import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
66import org.apache.asterix.common.config.DatasetConfig.DatasetType;
67import org.apache.asterix.common.config.DatasetConfig.IndexType;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080068import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070069import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070070import org.apache.asterix.common.exceptions.WarningCollector;
71import org.apache.asterix.common.exceptions.WarningUtil;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080072import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070073import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin54c06012019-11-13 15:54:20 -080074import org.apache.asterix.common.metadata.DataverseName;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080075import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070076import org.apache.asterix.lang.common.base.AbstractLangExpression;
77import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070078import org.apache.asterix.lang.common.base.Expression;
79import org.apache.asterix.lang.common.base.Literal;
80import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070081import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070082import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070083import org.apache.asterix.lang.common.base.Statement;
84import org.apache.asterix.lang.common.clause.GroupbyClause;
85import org.apache.asterix.lang.common.clause.LetClause;
86import org.apache.asterix.lang.common.clause.LimitClause;
87import org.apache.asterix.lang.common.clause.OrderbyClause;
88import org.apache.asterix.lang.common.clause.UpdateClause;
89import org.apache.asterix.lang.common.clause.WhereClause;
90import org.apache.asterix.lang.common.context.RootScopeFactory;
91import org.apache.asterix.lang.common.context.Scope;
92import org.apache.asterix.lang.common.expression.AbstractAccessor;
93import org.apache.asterix.lang.common.expression.CallExpr;
94import org.apache.asterix.lang.common.expression.FieldAccessor;
95import org.apache.asterix.lang.common.expression.FieldBinding;
96import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
97import org.apache.asterix.lang.common.expression.IfExpr;
98import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -070099import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700100import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +0300101import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700102import org.apache.asterix.lang.common.expression.LiteralExpr;
103import org.apache.asterix.lang.common.expression.OperatorExpr;
104import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
105import org.apache.asterix.lang.common.expression.QuantifiedExpression;
106import org.apache.asterix.lang.common.expression.RecordConstructor;
107import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
108import org.apache.asterix.lang.common.expression.TypeExpression;
109import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
110import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700111import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
112import org.apache.asterix.lang.common.expression.VariableExpr;
113import org.apache.asterix.lang.common.literal.DoubleLiteral;
114import org.apache.asterix.lang.common.literal.FalseLiteral;
115import org.apache.asterix.lang.common.literal.FloatLiteral;
116import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700117import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700118import org.apache.asterix.lang.common.literal.NullLiteral;
119import org.apache.asterix.lang.common.literal.StringLiteral;
120import org.apache.asterix.lang.common.literal.TrueLiteral;
121import org.apache.asterix.lang.common.parser.ScopeChecker;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -0700122import org.apache.asterix.lang.common.statement.AdapterDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700123import org.apache.asterix.lang.common.statement.CompactStatement;
124import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800125import org.apache.asterix.lang.common.statement.StartFeedStatement;
126import org.apache.asterix.lang.common.statement.StopFeedStatement;
Ian Maxon38fe9402020-01-29 19:27:40 -0800127import org.apache.asterix.lang.common.statement.CreateAdapterStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700128import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
129import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
130import org.apache.asterix.lang.common.statement.CreateFeedStatement;
131import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
132import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800133import org.apache.asterix.lang.common.statement.CreateSynonymStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700134import org.apache.asterix.lang.common.statement.DatasetDecl;
135import org.apache.asterix.lang.common.statement.DataverseDecl;
136import org.apache.asterix.lang.common.statement.DataverseDropStatement;
137import org.apache.asterix.lang.common.statement.DeleteStatement;
138import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700139import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700140import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
141import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300142import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700143import org.apache.asterix.lang.common.statement.FunctionDecl;
144import org.apache.asterix.lang.common.statement.FunctionDropStatement;
145import org.apache.asterix.lang.common.statement.IndexDropStatement;
146import org.apache.asterix.lang.common.statement.InsertStatement;
147import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
148import org.apache.asterix.lang.common.statement.LoadStatement;
149import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
150import org.apache.asterix.lang.common.statement.NodegroupDecl;
151import org.apache.asterix.lang.common.statement.Query;
152import org.apache.asterix.lang.common.statement.RefreshExternalDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700153import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800154import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700155import org.apache.asterix.lang.common.statement.TypeDecl;
156import org.apache.asterix.lang.common.statement.TypeDropStatement;
157import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800158import org.apache.asterix.lang.common.statement.UpsertStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700159import org.apache.asterix.lang.common.statement.WriteStatement;
160import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800161import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700162import org.apache.asterix.lang.common.struct.QuantifiedPair;
163import org.apache.asterix.lang.common.struct.VarIdentifier;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700164import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700165import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
166import org.apache.asterix.lang.sqlpp.clause.FromClause;
167import org.apache.asterix.lang.sqlpp.clause.FromTerm;
168import org.apache.asterix.lang.sqlpp.clause.HavingClause;
169import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700170import org.apache.asterix.lang.sqlpp.clause.Projection;
171import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
172import org.apache.asterix.lang.sqlpp.clause.SelectClause;
173import org.apache.asterix.lang.sqlpp.clause.SelectElement;
174import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
175import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
176import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800177import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700178import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700179import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700180import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700181import org.apache.asterix.lang.sqlpp.optype.JoinType;
182import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700183import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser;
184import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingElement;
185import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSet;
186import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSets;
187import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.RollupCube;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700188import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700189import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
190import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700191import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700192import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700193import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Ian Maxon38fe9402020-01-29 19:27:40 -0800194import org.apache.asterix.external.dataset.adapter.AdapterIdentifier;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800195import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700196import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700197import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700198import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700199import org.apache.hyracks.algebricks.common.utils.Pair;
200import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800201import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700202import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700203import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
204import org.apache.hyracks.algebricks.core.algebra.expressions.IndexedNLJoinExpressionAnnotation;
205import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700206import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700207import org.apache.hyracks.api.exceptions.Warning;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800208import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700209import org.apache.hyracks.util.StringUtil;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700210
Yingyi Bucaea8f02015-11-16 15:12:15 -0800211class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700212
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800213 // tokens parsed as identifiers
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700214 private static final String CUBE = "CUBE";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800215 private static final String CURRENT = "CURRENT";
216 private static final String EXCLUDE = "EXCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700217 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800218 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700219 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800220 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700221 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700222 private static final String LAST = "LAST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800223 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700224 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800225 private static final String OTHERS = "OTHERS";
226 private static final String PARTITION = "PARTITION";
227 private static final String PRECEDING = "PRECEDING";
228 private static final String RANGE = "RANGE";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700229 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700230 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800231 private static final String ROW = "ROW";
232 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700233 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800234 private static final String TIES = "TIES";
235 private static final String UNBOUNDED = "UNBOUNDED";
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700236 private static final String REPLACE = "REPLACE";
Ian Maxon38fe9402020-01-29 19:27:40 -0800237 private static final String RETURNS = "RETURNS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800238
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700239 private static final String INT_TYPE_NAME = "int";
240
Till Westmann7199a562016-09-17 16:07:32 -0700241 // error configuration
242 protected static final boolean REPORT_EXPECTED_TOKENS = false;
243
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700244 private int externalVarCounter;
245
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800246 private DataverseName defaultDataverse;
247
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700248 private SqlppGroupingSetsParser groupingSetsParser;
249
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700250 private final WarningCollector warningCollector = new WarningCollector();
251
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700252 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
253
Yingyi Bu391f09e2015-10-29 13:49:39 -0700254 private static class IndexParams {
255 public IndexType type;
256 public int gramLength;
257
258 public IndexParams(IndexType type, int gramLength) {
259 this.type = type;
260 this.gramLength = gramLength;
261 }
262 };
263
264 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800265 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700266 public String library;
267 public String function;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -0600268 public Token hintToken;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700269 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700270 }
271
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700272 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
273 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
274 switch (hintToken.hint) {
275 case VAL_FILE_HINT:
276 File[] valFiles = new File[splits.length];
277 for (int k=0; k<splits.length; k++) {
278 valFiles[k] = new File(splits[k]);
279 }
280 return new FieldValFileDataGen(valFiles);
281 case VAL_FILE_SAME_INDEX_HINT:
282 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
283 case LIST_VAL_FILE_HINT:
284 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
285 case LIST_HINT:
286 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
287 case INTERVAL_HINT:
288 FieldIntervalDataGen.ValueType vt;
289 switch (splits[0]) {
290 case "int":
291 vt = FieldIntervalDataGen.ValueType.INT;
292 break;
293 case "long":
294 vt = FieldIntervalDataGen.ValueType.LONG;
295 break;
296 case "float":
297 vt = FieldIntervalDataGen.ValueType.FLOAT;
298 break;
299 case "double":
300 vt = FieldIntervalDataGen.ValueType.DOUBLE;
301 break;
302 default:
303 throw new SqlppParseException(getSourceLocation(hintToken),
304 "Unknown type for interval data gen: " + splits[0]);
305 }
306 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
307 case INSERT_RAND_INT_HINT:
308 return new InsertRandIntDataGen(splits[0], splits[1]);
309 case DATE_BETWEEN_YEARS_HINT:
310 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
311 case DATETIME_BETWEEN_YEARS_HINT:
312 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
313 case DATETIME_ADD_RAND_HOURS_HINT:
314 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
315 case AUTO_HINT:
316 return new AutoDataGen(splits[0]);
317 default:
318 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700319 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700320 }
321
Till Westmann7199a562016-09-17 16:07:32 -0700322 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700323 this(new StringReader(s));
324 super.setInput(s);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700325 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700326 }
327
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800328 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700329 File file = new File(args[0]);
330 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
331 SQLPPParser parser = new SQLPPParser(fis);
332 List<Statement> st = parser.parse();
333 //st.accept(new SQLPPPrintVisitor(), 0);
334 }
335
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800336 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800337 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700338 return parseImpl(new ParseFunction<List<Statement>>() {
339 @Override
340 public List<Statement> parse() throws ParseException {
341 return SQLPPParser.this.Statement();
342 }
343 });
344 }
345
Dmitry Lychagin62f0beb2020-09-08 11:44:19 -0700346 @Override
347 public Expression parseExpression() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700348 return parseImpl(new ParseFunction<Expression>() {
349 @Override
350 public Expression parse() throws ParseException {
351 return SQLPPParser.this.Expression();
352 }
353 });
354 }
355
356 private static Expression parseExpression(String text) throws CompilationException {
357 return new SQLPPParser(text).parseExpression();
358 }
359
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800360 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800361 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800362 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800363 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800364 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800365 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800366 DataverseName dataverse = defaultDataverse;
367 defaultDataverse = signature.getDataverseName();
368 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800369 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
370 for (String paramName : paramNames) {
371 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
372 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800373 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
374 removeCurrentScope();
375 defaultDataverse = dataverse;
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800376 return new FunctionDecl(signature, paramVars, functionBodyExpr);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800377 }
378 });
379 }
380
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700381 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700382 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700383 hintCollector.clear();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700384 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700385 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700386 } catch (SqlppParseException e) {
387 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(), LogRedactionUtil.userData(getMessage(e)));
388 } catch (ParseException e) {
389 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700390 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700391 // this is here as the JavaCharStream that's below the lexer sometimes throws Errors that are not handled
Yingyi Bu391f09e2015-10-29 13:49:39 -0700392 // by the ANTLR-generated lexer or parser (e.g it does this for invalid backslash u + 4 hex digits escapes)
Till Westmann60f89982017-08-11 18:14:20 -0700393 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700394 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700395 } finally {
396 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700397 }
398 }
Till Westmann7199a562016-09-17 16:07:32 -0700399
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700400 @FunctionalInterface
401 private interface ParseFunction<T> {
402 T parse() throws ParseException;
403 }
404
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700405 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700406 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
407 warningCollector.getWarnings(outWarnings, maxWarnings);
408 }
409
410 @Override
411 public long getTotalWarningsCount() {
412 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700413 }
414
Till Westmann7199a562016-09-17 16:07:32 -0700415 protected String getMessage(ParseException pe) {
416 Token currentToken = pe.currentToken;
417 if (currentToken == null) {
418 return pe.getMessage();
419 }
420 int[][] expectedTokenSequences = pe.expectedTokenSequences;
421 String[] tokenImage = pe.tokenImage;
422 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
423 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
424 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
425 Token tok = currentToken.next;
426 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700427 StringBuilder message = new StringBuilder(128);
428 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700429 for (int i = 0; i < maxSize; i++) {
430 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700431 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700432 }
433 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700434 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700435 break;
436 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700437 final String fixedTokenImage = tokenImage[tok.kind];
438 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700439 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700440 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700441 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700442 tok = tok.next;
443 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700444 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700445 if (REPORT_EXPECTED_TOKENS) {
446 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700447 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700448 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700449 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700450 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700451 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700452 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700453 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700454 }
455
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700456 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700457 return
458 token == null ? null :
459 token.sourceLocation != null ? token.sourceLocation :
460 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700461 }
462
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700463 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
464 expr.setSourceLocation(getSourceLocation(token));
465 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700466 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800467
468 private boolean isToken(String image) {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700469 return isToken(token, image);
470 }
471
472 private static boolean isToken(Token token, String image) {
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800473 return token.image.equalsIgnoreCase(image);
474 }
475
476 private void expectToken(String image) throws SqlppParseException {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700477 expectToken(token, image);
478 }
479
480 private static void expectToken(Token token, String image) throws SqlppParseException {
481 if (!isToken(token, image)) {
482 throw createUnexpectedTokenError(token);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800483 }
484 }
485
486 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700487 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700488 }
489
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700490 private static SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700491 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800492 }
493
494 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700495 return createUnexpectedTokenError(token, expected);
496 }
497
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700498 private static SqlppParseException createUnexpectedTokenError(Token t, String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700499 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800500 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700501 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800502 }
503
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700504 private boolean laToken(int idx, int kind) {
505 Token t = getToken(idx);
506 return t.kind == kind;
507 }
508
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800509 private boolean laToken(int idx, int kind, String image) {
510 Token t = getToken(idx);
511 return t.kind == kind && t.image.equalsIgnoreCase(image);
512 }
513
514 private boolean laIdentifier(int idx, String image) {
515 return laToken(idx, IDENTIFIER, image);
516 }
517
518 private boolean laIdentifier(String image) {
519 return laIdentifier(1, image);
520 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700521
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700522 private Token fetchHint(Token token, SqlppHint... expectedHints) {
523 Token hintToken = token.specialToken;
524 if (hintToken == null) {
525 return null;
526 }
527 SourceLocation sourceLoc = getSourceLocation(hintToken);
528 hintCollector.remove(sourceLoc);
529 if (hintToken.hint == null) {
530 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
531 return null;
532 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
533 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
534 return null;
535 } else {
536 return hintToken;
537 }
538 }
539
540 private void reportUnclaimedHints() {
541 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
542 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
543 }
544 }
545
546 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
547 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
548 }
549
550 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700551 if (warningCollector.shouldWarn()) {
552 warningCollector.warn(WarningUtil.forAsterix(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
553 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700554 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700555
556 private void ensureNoTypeDeclsInFunction(String fnName, List<Pair<VarIdentifier, TypeExpression>> paramList,
557 TypeExpression returnType, Token startToken) throws SqlppParseException {
558 for (Pair<VarIdentifier, TypeExpression> p : paramList) {
559 if (p.second != null) {
560 String paramName = SqlppVariableUtil.toUserDefinedName(p.first.getValue());
561 throw new SqlppParseException(getSourceLocation(startToken),
562 "Unexpected type declaration for parameter " + paramName + " in function " + fnName);
563 }
564 }
565 if (returnType != null) {
566 throw new SqlppParseException(getSourceLocation(startToken),
567 "Unexpected return type declaration for function " + fnName);
568 }
569 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -0700570
571 private void ensureIntegerLiteral(LiteralExpr expr, String errorPrefix) throws SqlppParseException {
572 Literal lit = expr.getValue();
573 if (lit.getLiteralType() != Literal.Type.INTEGER && lit.getLiteralType() != Literal.Type.LONG) {
574 throw new SqlppParseException(expr.getSourceLocation(), errorPrefix + " should be an INTEGER");
575 }
576 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700577}
578
579PARSER_END(SQLPPParser)
580
Yingyi Bu391f09e2015-10-29 13:49:39 -0700581List<Statement> Statement() throws ParseException:
582{
583 scopeStack.push(RootScopeFactory.createRootScope(this));
584 List<Statement> decls = new ArrayList<Statement>();
585 Statement stmt = null;
586}
587{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300588 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700589 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300590 {
591 decls.add(stmt);
592 }
593 )?
594 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700595 )*
596 <EOF>
597 {
598 return decls;
599 }
600}
601
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700602Statement ExplainStatement() throws ParseException:
603{
604 Statement stmt = null;
605 Token explainToken = null;
606}
607{
608 ( <EXPLAIN> { explainToken = token; } )?
609 stmt = SingleStatement()
610 {
611 if (explainToken != null) {
612 if (stmt.getKind() == Statement.Kind.QUERY) {
613 ((Query)stmt).setExplain(true);
614 } else {
615 throw new SqlppParseException(getSourceLocation(explainToken),
616 "EXPLAIN is not supported for this kind of statement");
617 }
618 }
619 return stmt;
620 }
621}
622
Yingyi Bu391f09e2015-10-29 13:49:39 -0700623Statement SingleStatement() throws ParseException:
624{
625 Statement stmt = null;
626}
627{
628 (
629 stmt = DataverseDeclaration()
630 | stmt = FunctionDeclaration()
631 | stmt = CreateStatement()
632 | stmt = LoadStatement()
633 | stmt = DropStatement()
634 | stmt = WriteStatement()
635 | stmt = SetStatement()
636 | stmt = InsertStatement()
637 | stmt = DeleteStatement()
638 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800639 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700640 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700641 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700642 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700643 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700644 )
645 {
646 return stmt;
647 }
648}
649
650DataverseDecl DataverseDeclaration() throws ParseException:
651{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700652 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800653 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700654}
655{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800656 <USE> { startToken = token; } dvName = MultipartIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700657 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800658 defaultDataverse = DataverseName.create(dvName);
659 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700660 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700661 }
662}
663
664Statement CreateStatement() throws ParseException:
665{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700666 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700667 Statement stmt = null;
668}
669{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700670 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700671 (
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700672 stmt = CreateOrReplaceStatement(startToken)
673 | stmt = CreateTypeStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800674 | stmt = CreateNodegroupStatement(startToken)
675 | stmt = CreateDatasetStatement(startToken)
676 | stmt = CreateIndexStatement(startToken)
677 | stmt = CreateDataverseStatement(startToken)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700678 | stmt = CreateFunctionStatement(startToken, false)
Ian Maxon38fe9402020-01-29 19:27:40 -0800679 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800680 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800681 | stmt = CreateFeedStatement(startToken)
682 | stmt = CreateFeedPolicyStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700683 )
684 {
685 return stmt;
686 }
687}
688
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700689Statement CreateOrReplaceStatement(Token startStmtToken) throws ParseException:
690{
691 Statement stmt = null;
692 Token replaceToken = null;
693}
694{
695 <OR> <IDENTIFIER> { replaceToken = token; }
696 (
697 stmt = CreateFunctionStatement(startStmtToken, true)
698 )
699 {
700 // check expected token here to make the grammar extension plugin happy
701 expectToken(replaceToken, REPLACE);
702 return stmt;
703 }
704}
705
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800706TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
707{
708 TypeDecl stmt = null;
709}
710{
711 <TYPE> stmt = TypeSpecification(startStmtToken)
712 {
713 return stmt;
714 }
715}
716
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700717TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700718{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800719 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700720 boolean ifNotExists = false;
721 TypeExpression typeExpr = null;
722}
723{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800724 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700725 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800726 {
727 boolean dgen = false;
728 long numValues = -1;
729 String filename = null;
730 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
731 if (hintToken != null) {
732 String hintParams = hintToken.hintParams;
733 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
734 if (splits == null || splits.length != 2) {
735 throw new SqlppParseException(getSourceLocation(hintToken),
736 "Expecting /*+ dgen <filename> <numberOfItems> */");
737 }
738 dgen = true;
739 filename = splits[0];
740 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700741 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800742 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
743 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
744 return addSourceLocation(stmt, startStmtToken);
745 }
746}
747
748NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
749{
750 NodegroupDecl stmt = null;
751}
752{
753 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
754 {
755 return stmt;
756 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700757}
758
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700759NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700760{
761 String name = null;
762 String tmp = null;
763 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800764 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700765}
766{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800767 name = Identifier() ifNotExists = IfNotExists()
768 <ON> tmp = Identifier()
769 {
770 ncNames.add(new Identifier(tmp));
771 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700772 ( <COMMA> tmp = Identifier()
773 {
774 ncNames.add(new Identifier(tmp));
775 }
776 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800777 {
778 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
779 return addSourceLocation(stmt, startStmtToken);
780 }
781}
782
783void Dataset() throws ParseException:
784{
785}
786{
787 (<DATASET>|<COLLECTION>)
788}
789
790DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
791{
792 DatasetDecl stmt = null;
793}
794{
795 (
796 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
797 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
798 )
799 {
800 return stmt;
801 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700802}
803
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700804DatasetDecl DatasetSpecification(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;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700808 TypeExpression typeExpr = null;
809 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800810 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700811 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700812 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700813 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800814 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -0800815 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700816}
817{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800818 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700819 typeExpr = DatasetTypeSpecification()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700820 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800821 { String name; }
822 <WITH>
823 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700824 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800825 if (!name.equalsIgnoreCase("meta")){
826 throw new SqlppParseException(getSourceLocation(startStmtToken),
827 "We can only support one additional associated field called \"meta\".");
828 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700829 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700830 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800831 )?
832 ifNotExists = IfNotExists()
833 primaryKeyFields = PrimaryKey()
834 (<AUTOGENERATED> { autogenerated = true; } )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800835 ( <HINTS> hints = Properties() )?
836 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
837 ( <WITH> withRecord = RecordConstructor() )?
838 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800839 try {
Xikui Wangca7927f2020-09-03 11:27:57 -0700840 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
841 filterField == null? null : filterField.first, filterField == null? null : filterField.second);
Murtadha Hubail353e95f2020-08-24 22:18:04 +0300842 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr, hints,
843 DatasetType.INTERNAL, idd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800844 return addSourceLocation(stmt, startStmtToken);
845 } catch (CompilationException e) {
846 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
847 }
848 }
849}
850
851DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
852{
853 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700854 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800855 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800856 String adapterName = null;
857 Map<String,String> properties = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800858 Map<String,String> hints = new HashMap<String,String>();
859 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800860 RecordConstructor withRecord = null;
861}
862{
863 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700864 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800865 ifNotExists = IfNotExists()
866 <USING> adapterName = AdapterName() properties = Configuration()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800867 ( <HINTS> hints = Properties() )?
868 ( <WITH> withRecord = RecordConstructor() )?
869 {
870 ExternalDetailsDecl edd = new ExternalDetailsDecl();
871 edd.setAdapter(adapterName);
872 edd.setProperties(properties);
873 try {
Murtadha Hubail353e95f2020-08-24 22:18:04 +0300874 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null, hints, DatasetType.EXTERNAL,
875 edd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800876 return addSourceLocation(stmt, startStmtToken);
877 } catch (CompilationException e) {
878 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
879 }
880 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700881}
882
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700883TypeExpression DatasetTypeSpecification() throws ParseException:
884{
885 TypeExpression typeExpr = null;
886}
887{
888 (
889 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
890 | typeExpr = DatasetReferenceTypeSpecification()
891 )
892 {
893 return typeExpr;
894 }
895}
896
897TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
898{
899 TypeExpression typeExpr = null;
900}
901{
902 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
903 {
904 return typeExpr;
905 }
906}
907
908TypeExpression DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
909{
910 RecordTypeDefinition recordTypeDef = null;
911 RecordTypeDefinition.RecordKind recordKind = null;
Dmitry Lychagin6c239e72020-06-19 19:46:27 -0700912 Token startToken = null, recordKindToken = null;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700913}
914{
Dmitry Lychagin6c239e72020-06-19 19:46:27 -0700915 <LEFTPAREN> { startToken = token; } recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700916 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
917 {
918 if (recordKind == null) {
919 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
920 } else if (!allowRecordKindModifier) {
921 throw createUnexpectedTokenError(recordKindToken);
922 }
923 recordTypeDef.setRecordKind(recordKind);
Dmitry Lychagin6c239e72020-06-19 19:46:27 -0700924 return addSourceLocation(recordTypeDef, startToken);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700925 }
926}
927
928RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
929{
930 RecordTypeDefinition recType = new RecordTypeDefinition();
931}
932{
933 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
934 {
935 return recType;
936 }
937}
938
939void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
940{
941 String fieldName;
942 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -0700943 boolean nullable = true, missable = true;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700944}
945{
946 fieldName = Identifier()
Dmitry Lychagin3a628022020-05-12 18:09:02 -0700947 type = TypeReference() ( <NOT> <UNKNOWN> { nullable = false; missable = false; } )?
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700948 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -0700949 recType.addField(fieldName, type, nullable, missable);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700950 }
951}
952
Yingyi Bu391f09e2015-10-29 13:49:39 -0700953RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
954{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700955 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800956 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700957 String datasetName = null;
958}
959{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800960 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
961 {
962 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
963 stmt.setDataverseName(nameComponents.first);
964 stmt.setDatasetName(nameComponents.second);
965 return addSourceLocation(stmt, startToken);
966 }
967}
968
969CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
970{
971 CreateIndexStatement stmt = null;
972}
973{
974 (
975 <INDEX> stmt = IndexSpecification(startStmtToken)
976 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
977 )
978 {
979 return stmt;
980 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700981}
982
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700983CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700984{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700985 CreateIndexStatement stmt = new CreateIndexStatement();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700986 String indexName = null;
987 boolean ifNotExists = false;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800988 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700989 Pair<Integer, Pair<List<String>, IndexedTypeExpression>> fieldPair = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700990 IndexParams indexType = null;
991 boolean enforced = false;
992}
993{
Ali Alsuliman8351d252017-09-24 00:43:15 -0700994 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800995 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -0700996 <ON> nameComponents = QualifiedName()
997 <LEFTPAREN> ( fieldPair = OpenField()
998 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700999 stmt.addFieldExprPair(fieldPair.second);
1000 stmt.addFieldIndexIndicator(fieldPair.first);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001001 }
Ali Alsuliman8351d252017-09-24 00:43:15 -07001002 ) (<COMMA> fieldPair = OpenField()
1003 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001004 stmt.addFieldExprPair(fieldPair.second);
1005 stmt.addFieldIndexIndicator(fieldPair.first);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001006 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001007 )* <RIGHTPAREN> ( <TYPE> indexType = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman8351d252017-09-24 00:43:15 -07001008 )
1009 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001010 stmt.setIndexName(new Identifier(indexName));
1011 stmt.setIfNotExists(ifNotExists);
1012 stmt.setDataverseName(nameComponents.first);
1013 stmt.setDatasetName(nameComponents.second);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001014 if (indexType != null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001015 stmt.setIndexType(indexType.type);
1016 stmt.setGramLength(indexType.gramLength);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001017 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001018 stmt.setEnforced(enforced);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001019 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001020 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001021}
1022
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001023CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
1024{
1025 CreateIndexStatement stmt = new CreateIndexStatement();
1026 String indexName = null;
1027 boolean ifNotExists = false;
1028 Pair<DataverseName,Identifier> nameComponents = null;
1029}
1030{
1031 (indexName = Identifier())? ifNotExists = IfNotExists()
1032 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
1033 {
1034 if (indexName == null) {
1035 indexName = "primary_idx_" + nameComponents.second;
1036 }
1037 stmt.setIndexName(new Identifier(indexName));
1038 stmt.setIfNotExists(ifNotExists);
1039 stmt.setDataverseName(nameComponents.first);
1040 stmt.setDatasetName(nameComponents.second);
1041 return addSourceLocation(stmt, startStmtToken);
1042 }
1043}
1044
Yingyi Bu391f09e2015-10-29 13:49:39 -07001045String CompactionPolicy() throws ParseException :
1046{
1047 String compactionPolicy = null;
1048}
1049{
1050 compactionPolicy = Identifier()
1051 {
1052 return compactionPolicy;
1053 }
1054}
1055
1056String FilterField() throws ParseException :
1057{
1058 String filterField = null;
1059}
1060{
1061 filterField = Identifier()
1062 {
1063 return filterField;
1064 }
1065}
1066
1067IndexParams IndexType() throws ParseException:
1068{
1069 IndexType type = null;
1070 int gramLength = 0;
1071}
1072{
1073 (<BTREE>
1074 {
1075 type = IndexType.BTREE;
1076 }
1077 | <RTREE>
1078 {
1079 type = IndexType.RTREE;
1080 }
1081 | <KEYWORD>
1082 {
1083 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1084 }
Taewoo Kimc49405a2017-01-04 00:30:43 -08001085 |<FULLTEXT>
1086 {
1087 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1088 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001089 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1090 {
1091 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1092 gramLength = Integer.valueOf(token.image);
1093 }
1094 <RIGHTPAREN>)
1095 {
1096 return new IndexParams(type, gramLength);
1097 }
1098}
1099
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001100CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1101{
1102 CreateDataverseStatement stmt = null;
1103}
1104{
1105 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1106 {
1107 return stmt;
1108 }
1109}
1110
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001111CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001112{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001113 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001114 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001115}
1116{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001117 dvName = MultipartIdentifier() ifNotExists = IfNotExists()
1118 {
1119 CreateDataverseStatement stmt = new CreateDataverseStatement(DataverseName.create(dvName), null, ifNotExists);
1120 return addSourceLocation(stmt, startStmtToken);
1121 }
1122}
1123
Ian Maxon38fe9402020-01-29 19:27:40 -08001124CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1125{
1126 CreateAdapterStatement stmt = null;
1127}
1128{
1129 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1130 {
1131 return stmt;
1132 }
1133}
1134
1135CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1136{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001137 Pair<DataverseName,Identifier> adapterName = null;
1138 Pair<DataverseName,Identifier> libraryName = null;
1139 List<String> externalIdentifier = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001140 boolean ifNotExists = false;
1141}
1142{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001143 adapterName = QualifiedName()
1144 ifNotExists = IfNotExists()
1145 <AS> externalIdentifier = FunctionExternalIdentifier()
1146 <AT> libraryName = QualifiedName()
1147 {
1148 CreateAdapterStatement stmt = new CreateAdapterStatement(adapterName.first, adapterName.second.getValue(),
1149 libraryName.first, libraryName.second.getValue(), externalIdentifier, ifNotExists);
1150 return addSourceLocation(stmt, startStmtToken);
1151 }
1152}
Ian Maxon38fe9402020-01-29 19:27:40 -08001153
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001154CreateFunctionStatement CreateFunctionStatement(Token startStmtToken, boolean orReplace) throws ParseException:
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001155{
1156 CreateFunctionStatement stmt = null;
1157}
1158{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001159 <FUNCTION> stmt = FunctionSpecification(startStmtToken, orReplace)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001160 {
1161 return stmt;
1162 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001163}
1164
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001165CreateFunctionStatement FunctionSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001166{
Ian Maxon38fe9402020-01-29 19:27:40 -08001167 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001168 FunctionName fctName = null;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001169 List<Pair<VarIdentifier,TypeExpression>> params = null;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001170 TypeExpression returnType = null;
1171 Token beginPos = null, endPos = null;
1172 Expression functionBodyExpr = null;
1173 Pair<DataverseName,Identifier> libraryName = null;
1174 List<String> externalIdentifier = null;
1175 RecordConstructor withOptions = null;
1176 boolean ifNotExists = false;
1177 CreateFunctionStatement stmt = null;
1178 DataverseName currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001179}
1180{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001181 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001182 {
1183 defaultDataverse = fctName.dataverse;
1184 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001185 ifNotExists = IfNotExists()
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001186 params = FunctionParameters()
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001187 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001188 (
1189 (
1190 <LEFTBRACE>
1191 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001192 beginPos = token;
1193 createNewScope();
Ian Maxon38fe9402020-01-29 19:27:40 -08001194 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001195 functionBodyExpr = FunctionBody()
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001196 <RIGHTBRACE>
1197 {
1198 endPos = token;
1199 String functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine,
1200 endPos.beginColumn);
1201 signature = new FunctionSignature(fctName.dataverse, fctName.function, params.size());
1202 getCurrentScope().addFunctionDescriptor(signature, false);
1203 removeCurrentScope();
1204 defaultDataverse = currentDataverse;
1205 ensureNoTypeDeclsInFunction(fctName.function, params, returnType, startStmtToken);
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001206 stmt = new CreateFunctionStatement(signature, params, functionBody, functionBodyExpr, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001207 return addSourceLocation(stmt, startStmtToken);
1208 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001209 )
1210 |
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001211 (
1212 <AS> externalIdentifier = FunctionExternalIdentifier()
1213 <AT> libraryName = QualifiedName()
1214 (<WITH> withOptions = RecordConstructor())?
1215 {
1216 signature = new FunctionSignature(fctName.dataverse, fctName.function, params.size());
1217 defaultDataverse = currentDataverse;
1218 try {
1219 stmt = new CreateFunctionStatement(signature, params, returnType, libraryName.first,
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001220 libraryName.second.getValue(), externalIdentifier, withOptions, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001221 } catch (AlgebricksException e) {
1222 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1223 }
1224 return addSourceLocation(stmt, startStmtToken);
1225 }
1226 )
Ian Maxon38fe9402020-01-29 19:27:40 -08001227 )
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001228}
1229
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001230List<Pair<VarIdentifier,TypeExpression>> FunctionParameters() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001231{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001232 List<Pair<VarIdentifier,TypeExpression>> paramList = Collections.<Pair<VarIdentifier,TypeExpression>>emptyList();
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001233}
1234{
1235 <LEFTPAREN> (paramList = FunctionParameterList())? <RIGHTPAREN>
1236 {
1237 return paramList;
1238 }
1239}
1240
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001241List<Pair<VarIdentifier,TypeExpression>> FunctionParameterList() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001242{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001243 List<Pair<VarIdentifier,TypeExpression>> paramList = new ArrayList<Pair<VarIdentifier,TypeExpression>>();
1244 Pair<VarIdentifier,TypeExpression> param = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001245}
1246{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001247 param = FunctionParameter() { paramList.add(param); }
1248 ( <COMMA> param = FunctionParameter() { paramList.add(param); } )*
1249 {
1250 return paramList;
1251 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001252}
1253
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001254Pair<VarIdentifier,TypeExpression> FunctionParameter() throws ParseException:
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001255{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001256 String name = null;
1257 TypeExpression type = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001258}
1259{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001260 name = VariableIdentifier()
1261 ( LOOKAHEAD(3) (<COLON>)? type = TypeExpr(false) )?
1262 {
1263 return new Pair<VarIdentifier,TypeExpression>(new VarIdentifier(name), type);
1264 }
1265}
1266
1267TypeExpression FunctionReturnType() throws ParseException:
1268{
1269 TypeExpression returnType = null;
1270}
1271{
1272 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = TypeExpr(false) )?
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001273 {
1274 return returnType;
1275 }
1276}
1277
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001278Expression FunctionBody() throws ParseException:
1279{
1280 Expression functionBodyExpr = null;
1281}
1282{
1283 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1284 {
1285 return functionBodyExpr;
1286 }
1287}
1288
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001289List<String> FunctionExternalIdentifier() throws ParseException:
1290{
1291 String ident = null;
1292 List<String> identList = new ArrayList(2);
1293}
1294{
1295 ident = StringLiteral() { identList.add(ident.trim()); }
1296 ( <COMMA> ident = StringLiteral() { identList.add(ident.trim()); } )*
1297 {
1298 return identList;
1299 }
1300}
1301
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001302CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1303{
1304 CreateFeedStatement stmt = null;
1305}
1306{
1307 <FEED> stmt = FeedSpecification(startStmtToken)
1308 {
1309 return stmt;
1310 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001311}
1312
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001313CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001314{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001315 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001316 boolean ifNotExists = false;
1317 String adapterName = null;
1318 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001319 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001320 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001321 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001322}
1323{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001324 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001325 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001326 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001327 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001328 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001329 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001330 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001331 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001332 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001333 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001334}
1335
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001336CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1337{
1338 CreateFeedPolicyStatement stmt = null;
1339}
1340{
1341 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1342 {
1343 return stmt;
1344 }
1345}
1346
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001347CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001348{
Michael Blowd6cf6412016-06-30 02:44:35 -04001349 String policyName = null;
1350 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001351 String sourcePolicyFile = null;
1352 String definition = null;
1353 boolean ifNotExists = false;
1354 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001355 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001356}
1357{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001358 policyName = Identifier() ifNotExists = IfNotExists()
1359 <FROM>
1360 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001361 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001362 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001363 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001364 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1365 {
1366 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1367 }
1368 )
1369 {
1370 return addSourceLocation(stmt, startStmtToken);
1371 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001372}
1373
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001374CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
1375{
1376 CreateSynonymStatement stmt = null;
1377}
1378{
1379 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
1380 {
1381 return stmt;
1382 }
1383}
1384
1385CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
1386{
1387 Pair<DataverseName,Identifier> nameComponents = null;
1388 Pair<DataverseName,Identifier> objectNameComponents = null;
1389 boolean ifNotExists = false;
1390}
1391{
1392 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1393 <FOR> objectNameComponents = QualifiedName()
1394 {
1395 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
1396 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
1397 return addSourceLocation(stmt, startStmtToken);
1398 }
1399}
1400
Yingyi Bu391f09e2015-10-29 13:49:39 -07001401boolean IfNotExists() throws ParseException:
1402{
1403}
1404{
Yingyi Budaa549c2016-06-28 22:30:52 -07001405 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001406 {
1407 return true;
1408 }
1409 )?
1410 {
1411 return false;
1412 }
1413}
1414
Xikui Wang9d63f622017-05-18 17:50:44 -07001415void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001416{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001417 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07001418 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001419}
1420{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001421 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001422 {
1423 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1424 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
1425 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07001426 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001427 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07001428 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001429 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1430 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07001431 }
1432 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07001433}
1434
1435String GetPolicy() throws ParseException:
1436{
1437 String policy = null;
1438}
1439{
1440 <USING> <POLICY> policy = Identifier()
1441 {
1442 return policy;
1443 }
1444
1445}
1446
1447FunctionSignature FunctionSignature() throws ParseException:
1448{
1449 FunctionName fctName = null;
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001450 List<Pair<VarIdentifier,TypeExpression>> params = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001451 int arity = 0;
1452}
1453{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001454 fctName = FunctionName()
1455 (
1456 LOOKAHEAD(2) params = FunctionParameters() { arity = params.size(); }
1457 | ( <LEFTPAREN> arity = FunctionArity() <RIGHTPAREN> )
1458 | ( <ATT> arity = FunctionArity() ) // back-compat
1459 )
1460 {
1461 return new FunctionSignature(fctName.dataverse, fctName.function, arity);
1462 }
1463}
Yingyi Bu391f09e2015-10-29 13:49:39 -07001464
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001465int FunctionArity() throws ParseException:
1466{
1467 int arity;
1468}
1469{
1470 <INTEGER_LITERAL>
1471 {
1472 try {
1473 arity = Integer.parseInt(token.image);
1474 } catch (NumberFormatException e) {
1475 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001476 }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001477 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
1478 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + arity);
1479 }
1480 return arity;
1481 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001482}
1483
Yingyi Buc9bfe252016-03-01 00:02:40 -08001484Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001485{
Yingyi Buc9bfe252016-03-01 00:02:40 -08001486 Pair<Integer, List<String>> tmp = null;
1487 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001488 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
1489}
1490{
1491 <PRIMARY> <KEY> tmp = NestedField()
1492 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001493 keyFieldSourceIndicators.add(tmp.first);
1494 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001495 }
1496 ( <COMMA> tmp = NestedField()
1497 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001498 keyFieldSourceIndicators.add(tmp.first);
1499 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001500 }
1501 )*
1502 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001503 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001504 }
1505}
1506
1507Statement DropStatement() throws ParseException:
1508{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001509 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001510 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001511}
1512{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001513 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001514 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001515 stmt = DropDatasetStatement(startToken)
1516 | stmt = DropIndexStatement(startToken)
1517 | stmt = DropNodeGroupStatement(startToken)
1518 | stmt = DropTypeStatement(startToken)
1519 | stmt = DropDataverseStatement(startToken)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001520 | stmt = DropAdapterStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001521 | stmt = DropFunctionStatement(startToken)
1522 | stmt = DropFeedStatement(startToken)
1523 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001524 | stmt = DropSynonymStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07001525 )
1526 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001527 return stmt;
1528 }
1529}
1530
1531DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
1532{
1533 DropDatasetStatement stmt = null;
1534}
1535{
1536 Dataset() stmt = DropDatasetSpecification(startStmtToken)
1537 {
1538 return stmt;
1539 }
1540}
1541
1542DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
1543{
1544 Pair<DataverseName,Identifier> pairId = null;
1545 boolean ifExists = false;
1546}
1547{
1548 pairId = QualifiedName() ifExists = IfExists()
1549 {
1550 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
1551 return addSourceLocation(stmt, startStmtToken);
1552 }
1553}
1554
1555IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
1556{
1557 IndexDropStatement stmt = null;
1558}
1559{
1560 <INDEX> stmt = DropIndexSpecification(startStmtToken)
1561 {
1562 return stmt;
1563 }
1564}
1565
1566IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
1567{
1568 Triple<DataverseName,Identifier,Identifier> tripleId = null;
1569 boolean ifExists = false;
1570}
1571{
1572 tripleId = DoubleQualifiedName() ifExists = IfExists()
1573 {
1574 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
1575 return addSourceLocation(stmt, startStmtToken);
1576 }
1577}
1578
1579NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
1580{
1581 NodeGroupDropStatement stmt = null;
1582}
1583{
1584 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
1585 {
1586 return stmt;
1587 }
1588}
1589
1590NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
1591{
1592 String id = null;
1593 boolean ifExists = false;
1594}
1595{
1596 id = Identifier() ifExists = IfExists()
1597 {
1598 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
1599 return addSourceLocation(stmt, startStmtToken);
1600 }
1601}
1602
1603TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
1604{
1605 TypeDropStatement stmt = null;
1606}
1607{
1608 <TYPE> stmt = DropTypeSpecification(startStmtToken)
1609 {
1610 return stmt;
1611 }
1612}
1613
1614TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
1615{
1616 Pair<DataverseName,Identifier> pairId = null;
1617 boolean ifExists = false;
1618}
1619{
1620 pairId = TypeName() ifExists = IfExists()
1621 {
1622 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
1623 return addSourceLocation(stmt, startStmtToken);
1624 }
1625}
1626
1627DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
1628{
1629 DataverseDropStatement stmt = null;
1630}
1631{
1632 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
1633 {
1634 return stmt;
1635 }
1636}
1637
1638DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
1639{
1640 List<String> multipartId = null;
1641 boolean ifExists = false;
1642}
1643{
1644 multipartId = MultipartIdentifier() ifExists = IfExists()
1645 {
1646 DataverseDropStatement stmt = new DataverseDropStatement(DataverseName.create(multipartId), ifExists);
1647 return addSourceLocation(stmt, startStmtToken);
1648 }
1649}
1650
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001651AdapterDropStatement DropAdapterStatement(Token startStmtToken) throws ParseException:
1652{
1653 AdapterDropStatement stmt = null;
1654}
1655{
1656 <ADAPTER> stmt = DropAdapterSpecification(startStmtToken)
1657 {
1658 return stmt;
1659 }
1660}
1661
1662AdapterDropStatement DropAdapterSpecification(Token startStmtToken) throws ParseException:
1663{
1664 Pair<DataverseName,Identifier> adapterName = null;
1665 boolean ifExists = false;
1666}
1667{
1668 adapterName = QualifiedName() ifExists = IfExists()
1669 {
1670 AdapterDropStatement stmt = new AdapterDropStatement(adapterName.first, adapterName.second.getValue(), ifExists);
1671 return addSourceLocation(stmt, startStmtToken);
1672 }
1673}
1674
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001675FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
1676{
1677 FunctionDropStatement stmt = null;
1678}
1679{
1680 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
1681 {
1682 return stmt;
1683 }
1684}
1685
1686FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
1687{
1688 FunctionSignature funcSig = null;
1689 boolean ifExists = false;
1690}
1691{
1692 funcSig = FunctionSignature() ifExists = IfExists()
1693 {
1694 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
1695 return addSourceLocation(stmt, startStmtToken);
1696 }
1697}
1698
1699FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
1700{
1701 FeedDropStatement stmt = null;
1702}
1703{
1704 <FEED> stmt = DropFeedSpecification(startStmtToken)
1705 {
1706 return stmt;
1707 }
1708}
1709
1710FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
1711{
1712 Pair<DataverseName,Identifier> pairId = null;
1713 boolean ifExists = false;
1714}
1715{
1716 pairId = QualifiedName() ifExists = IfExists()
1717 {
1718 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
1719 return addSourceLocation(stmt, startStmtToken);
1720 }
1721}
1722
1723FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
1724{
1725 FeedPolicyDropStatement stmt = null;
1726}
1727{
1728 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
1729 {
1730 return stmt;
1731 }
1732}
1733
1734FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
1735{
1736 Pair<DataverseName,Identifier> pairId = null;
1737 boolean ifExists = false;
1738}
1739{
1740 pairId = QualifiedName() ifExists = IfExists()
1741 {
1742 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
1743 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001744 }
1745}
1746
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001747SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
1748{
1749 SynonymDropStatement stmt = null;
1750}
1751{
1752 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
1753 {
1754 return stmt;
1755 }
1756}
1757
1758SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
1759{
1760 Pair<DataverseName,Identifier> pairId = null;
1761 boolean ifExists = false;
1762}
1763{
1764 pairId = QualifiedName() ifExists = IfExists()
1765 {
1766 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
1767 return addSourceLocation(stmt, startStmtToken);
1768 }
1769}
1770
Yingyi Bu391f09e2015-10-29 13:49:39 -07001771boolean IfExists() throws ParseException :
1772{
1773}
1774{
1775 ( LOOKAHEAD(1) <IF> <EXISTS>
1776 {
1777 return true;
1778 }
1779 )?
1780 {
1781 return false;
1782 }
1783}
1784
1785InsertStatement InsertStatement() throws ParseException:
1786{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001787 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001788 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001789 VariableExpr var = null;
1790 Query query = null;
1791 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001792}
1793{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001794 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001795 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001796 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001797 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001798 if (var == null && returnExpression != null) {
1799 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1800 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001801 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08001802 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001803 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1804 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001805 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001806 }
1807}
1808
1809UpsertStatement UpsertStatement() throws ParseException:
1810{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001811 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001812 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001813 VariableExpr var = null;
1814 Query query = null;
1815 Expression returnExpression = null;
1816}
1817{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001818 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001819 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001820 ( <RETURNING> returnExpression = Expression())?
1821 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001822 if (var == null && returnExpression != null) {
1823 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1824 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001825 }
1826 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001827 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1828 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001829 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001830 }
1831}
1832
1833DeleteStatement DeleteStatement() throws ParseException:
1834{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001835 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001836 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001837 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001838 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001839}
1840{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001841 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001842 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001843 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07001844 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001845 if (var == null) {
1846 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1847 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001848 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001849 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07001850 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001851 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001852 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001853}
1854
1855UpdateStatement UpdateStatement() throws ParseException:
1856{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001857 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001858 VariableExpr vars;
1859 Expression target;
1860 Expression condition;
1861 UpdateClause uc;
1862 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
1863}
1864{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001865 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001866 <WHERE> condition = Expression()
1867 <LEFTPAREN> (uc = UpdateClause()
1868 {
1869 ucs.add(uc);
1870 }
1871 (<COMMA> uc = UpdateClause()
1872 {
1873 ucs.add(uc);
1874 }
1875 )*) <RIGHTPAREN>
1876 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001877 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001878 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001879 }
1880}
1881
1882UpdateClause UpdateClause() throws ParseException:
1883{
1884 Expression target = null;
1885 Expression value = null ;
1886 InsertStatement is = null;
1887 DeleteStatement ds = null;
1888 UpdateStatement us = null;
1889 Expression condition = null;
1890 UpdateClause ifbranch = null;
1891 UpdateClause elsebranch = null;
1892}
1893{
1894 (<SET> target = Expression() <EQ> value = Expression()
1895 | is = InsertStatement()
1896 | ds = DeleteStatement()
1897 | us = UpdateStatement()
1898 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
1899 <THEN> ifbranch = UpdateClause()
1900 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
1901 {
1902 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
1903 }
1904 )
1905}
1906
1907Statement SetStatement() throws ParseException:
1908{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001909 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001910 String pn = null;
1911 String pv = null;
1912}
1913{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001914 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001915 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001916 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001917 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001918 }
1919}
1920
1921Statement WriteStatement() throws ParseException:
1922{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001923 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001924 String nodeName = null;
1925 String fileName = null;
1926 Query query;
1927 String writerClass = null;
1928 Pair<Identifier,Identifier> nameComponents = null;
1929}
1930{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001931 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07001932 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001933 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001934 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001935 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001936 }
1937}
1938
1939LoadStatement LoadStatement() throws ParseException:
1940{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001941 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001942 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001943 Identifier datasetName = null;
1944 boolean alreadySorted = false;
1945 String adapterName;
1946 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001947 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001948}
1949{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001950 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001951 {
1952 dataverseName = nameComponents.first;
1953 datasetName = nameComponents.second;
1954 }
1955 <USING> adapterName = AdapterName() properties = Configuration()
1956 (<PRESORTED>
1957 {
1958 alreadySorted = true;
1959 }
1960 )?
1961 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001962 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
1963 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001964 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001965 }
1966}
1967
1968
1969String AdapterName() throws ParseException :
1970{
1971 String adapterName = null;
1972}
1973{
1974 adapterName = Identifier()
1975 {
1976 return adapterName;
1977 }
1978}
1979
1980Statement CompactStatement() throws ParseException:
1981{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001982 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001983 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001984}
1985{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001986 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001987 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001988 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001989 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001990 }
1991}
1992
Yingyi Buab817482016-08-19 21:29:31 -07001993Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001994{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001995 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001996 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001997}
1998{
1999 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002000 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
2001 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
2002 | <START> { startToken = token; } stmt = StartStatement(startToken)
2003 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07002004 )
2005 {
2006 return stmt;
2007 }
2008}
2009
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002010Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002011{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002012 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002013 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002014}
2015{
2016 <FEED> feedNameComponents = QualifiedName()
2017 {
2018 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002019 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002020 }
2021}
2022
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002023AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002024{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002025 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002026 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002027}
2028{
2029 <FEED> feedNameComponents = QualifiedName()
2030 {
2031 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002032 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002033 }
2034}
2035
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002036AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002037{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002038 Pair<DataverseName,Identifier> feedNameComponents = null;
2039 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002040
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002041 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002042}
2043{
2044 (
2045 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002046 {
2047 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
2048 }
2049 )
Yingyi Buab817482016-08-19 21:29:31 -07002050 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002051 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002052 }
2053}
2054
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002055AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002056{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002057 Pair<DataverseName,Identifier> feedNameComponents = null;
2058 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002059
2060 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002061 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002062 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002063 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002064 String whereClauseBody = null;
2065 WhereClause whereClause = null;
2066 Token beginPos = null;
2067 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002068}
2069{
2070 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002071 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002072 (ApplyFunction(appliedFunctions))?
2073 (policy = GetPolicy())?
2074 (
2075 <WHERE>
2076 {
2077 beginPos = token;
2078 whereClause = new WhereClause();
2079 Expression whereExpr;
2080 }
2081 whereExpr = Expression()
2082 {
2083 whereClause.setWhereExpr(whereExpr);
2084 }
2085 )?
2086 {
2087 if (whereClause != null) {
2088 endPos = token;
2089 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2090 }
2091 }
Yingyi Buab817482016-08-19 21:29:31 -07002092 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002093 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002094 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002095 }
2096 )
2097 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002098 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002099 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002100}
2101
2102Map<String,String> Configuration() throws ParseException :
2103{
2104 Map<String,String> configuration = new LinkedHashMap<String,String>();
2105 Pair<String, String> keyValuePair = null;
2106}
2107{
2108 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2109 {
2110 configuration.put(keyValuePair.first, keyValuePair.second);
2111 }
2112 ( <COMMA> keyValuePair = KeyValuePair()
2113 {
2114 configuration.put(keyValuePair.first, keyValuePair.second);
2115 }
2116 )* )? <RIGHTPAREN>
2117 {
2118 return configuration;
2119 }
2120}
2121
2122Pair<String, String> KeyValuePair() throws ParseException:
2123{
2124 String key;
2125 String value;
2126}
2127{
Ali Alsulimane2986012020-05-14 18:14:22 -07002128 <LEFTPAREN> key = ConstantString()
2129 <EQ> ( value = ConstantString() | (<TRUE> | <FALSE>) {value = token.image.toLowerCase();} )
2130 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002131 {
2132 return new Pair<String, String>(key, value);
2133 }
2134}
2135
2136Map<String,String> Properties() throws ParseException:
2137{
2138 Map<String,String> properties = new HashMap<String,String>();
2139 Pair<String, String> property;
2140}
2141{
2142 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
2143 {
2144 properties.put(property.first, property.second);
2145 }
2146 ( <COMMA> property = Property()
2147 {
2148 properties.put(property.first, property.second);
2149 }
2150 )* <RIGHTPAREN> )?
2151 {
2152 return properties;
2153 }
2154}
2155
2156Pair<String, String> Property() throws ParseException:
2157{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002158 String key = null;
2159 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002160}
2161{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002162 (key = Identifier() | key = StringLiteral())
2163 <EQ>
2164 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002165 {
2166 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002167 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002168 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002169 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002170 }
2171 }
2172 )
2173 {
2174 return new Pair<String, String>(key.toUpperCase(), value);
2175 }
2176}
2177
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002178IndexedTypeExpression IndexedTypeExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002179{
2180 TypeExpression typeExpr = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002181 boolean isUnknownable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002182}
2183{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002184 typeExpr = TypeExpr(false)
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002185 ( <QUES> { isUnknownable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002186 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002187 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002188 }
2189}
2190
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002191TypeExpression TypeExpr(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002192{
2193 TypeExpression typeExpr = null;
2194}
2195{
2196 (
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002197 typeExpr = TypeReference()
2198 | typeExpr = OrderedListTypeDef(allowRecordTypeDef)
2199 | typeExpr = UnorderedListTypeDef(allowRecordTypeDef)
2200 | typeExpr = RecordTypeDef() {
2201 if (!allowRecordTypeDef) {
2202 throw new SqlppParseException(typeExpr.getSourceLocation(), "Unexpected record type declaration");
2203 }
2204 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002205 )
2206 {
2207 return typeExpr;
2208 }
2209}
2210
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002211RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
2212{
2213 RecordTypeDefinition.RecordKind recordKind = null;
2214}
2215{
2216 (
2217 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
2218 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
2219 )
2220 {
2221 return recordKind;
2222 }
2223}
2224
Yingyi Bu391f09e2015-10-29 13:49:39 -07002225RecordTypeDefinition RecordTypeDef() throws ParseException:
2226{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002227 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002228 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002229 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002230}
2231{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002232 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002233 <LEFTBRACE>
2234 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002235 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002236 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
2237 if (hintToken != null) {
2238 String hintParams = hintToken.hintParams;
2239 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
2240 if (splits == null || splits.length != 4) {
2241 throw new SqlppParseException(getSourceLocation(hintToken),
2242 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002243 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002244 if (!splits[0].equals("int")) {
2245 throw new SqlppParseException(getSourceLocation(hintToken),
2246 "The only supported type for gen-fields is int.");
2247 }
2248 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
2249 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
2250 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002251 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002252 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002253 (
2254 RecordField(recType)
2255 ( <COMMA> RecordField(recType) )*
2256 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002257 <RIGHTBRACE>
2258 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002259 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002260 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002261 }
2262}
2263
2264void RecordField(RecordTypeDefinition recType) throws ParseException:
2265{
2266 String fieldName;
2267 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002268 boolean nullable = false, missable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002269}
2270{
2271 fieldName = Identifier()
2272 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002273 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
2274 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
2275 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
2276 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002277 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002278 <COLON> type = TypeExpr(true) ( <QUES> { nullable = true; missable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002279 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002280 recType.addField(fieldName, type, nullable, missable, rfdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002281 }
2282}
2283
2284TypeReferenceExpression TypeReference() throws ParseException:
2285{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002286 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002287}
2288{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002289 id = QualifiedName()
2290 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07002291 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
2292 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002293 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002294
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002295 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002296 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002297 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002298}
2299
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002300OrderedListTypeDefinition OrderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002301{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002302 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002303 TypeExpression type = null;
2304}
2305{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002306 <LEFTBRACKET> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002307 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002308 <RIGHTBRACKET>
2309 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002310 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002311 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002312 }
2313}
2314
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002315UnorderedListTypeDefinition UnorderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002316{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002317 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002318 TypeExpression type = null;
2319}
2320{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002321 <LEFTDBLBRACE> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002322 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002323 <RIGHTDBLBRACE>
2324 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002325 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002326 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002327 }
2328}
2329
2330FunctionName FunctionName() throws ParseException:
2331{
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002332 Triple<List<String>, SourceLocation, Token> prefix = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002333 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002334}
2335{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002336 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
2337 // that copy must be kept in sync with this code
2338 prefix = MultipartIdentifierWithHints(SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002339 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.RANGE_HINT)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002340 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002341 {
2342 FunctionName result = new FunctionName();
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002343 result.sourceLoc = prefix.second;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002344 result.hintToken = prefix.third;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002345 List<String> list = prefix.first;
2346 int ln = list.size();
2347 String last = list.get(ln - 1);
2348 if (suffix == null) {
2349 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
2350 // no library name
2351 result.function = last;
2352 } else {
2353 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
2354 // suffix = func_name
2355 result.library = last;
2356 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002357 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002358 if (ln > 1) {
2359 result.dataverse = DataverseName.create(list, 0, ln - 1);
2360 } else {
2361 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002362 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002363
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002364 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
2365 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002366 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002367 return result;
2368 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002369}
2370
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002371Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002372{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002373 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002374}
2375{
2376 name = QualifiedName()
2377 {
2378 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002379 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002380 }
2381 return name;
2382 }
2383}
2384
2385String Identifier() throws ParseException:
2386{
2387 String lit = null;
2388}
2389{
2390 (<IDENTIFIER>
2391 {
2392 return token.image;
2393 }
2394 | lit = QuotedString()
2395 {
2396 return lit;
2397 }
2398 )
2399}
2400
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002401Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002402{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002403 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002404 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002405}
2406{
2407 fieldList = NestedField()
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002408 ( <COLON> fieldType = IndexedTypeExpr() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002409 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002410 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
2411 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002412 }
2413}
2414
Yingyi Buc9bfe252016-03-01 00:02:40 -08002415Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002416{
2417 List<String> exprList = new ArrayList<String>();
2418 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002419 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002420 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002421}
2422{
2423 lit = Identifier()
2424 {
Yingyi Bub9169b62016-02-26 21:21:49 -08002425 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002426 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08002427 }
2428 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08002429 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08002430 <LEFTPAREN><RIGHTPAREN>
2431 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002432 if(!lit.equalsIgnoreCase("meta")){
2433 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08002434 }
2435 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002436 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08002437 }
2438 )?
2439 {
2440 if(!meetParens){
2441 exprList.add(lit);
2442 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002443 }
2444 (<DOT>
2445 lit = Identifier()
2446 {
2447 exprList.add(lit);
2448 }
2449 )*
2450 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002451 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002452 }
2453}
2454
Yingyi Bu6d57e492016-06-06 21:24:42 -07002455String ConstantString() throws ParseException:
2456{
2457 String value = null;
2458}
2459{
2460 (value = QuotedString() | value = StringLiteral())
2461 {
2462 return value;
2463 }
2464}
2465
Yingyi Bu391f09e2015-10-29 13:49:39 -07002466String QuotedString() throws ParseException:
2467{
2468}
2469{
2470 <QUOTED_STRING>
2471 {
2472 return removeQuotesAndEscapes(token.image);
2473 }
2474}
2475
Yingyi Bu391f09e2015-10-29 13:49:39 -07002476String StringLiteral() throws ParseException:
2477{
2478}
2479{
2480 <STRING_LITERAL>
2481 {
2482 return removeQuotesAndEscapes(token.image);
2483 }
2484}
2485
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002486List<String> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002487{
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002488 Triple<List<String>, SourceLocation, Token> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002489}
2490{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002491 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002492 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002493 return result.first;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002494 }
2495}
2496
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002497Triple<List<String>, SourceLocation, Token> MultipartIdentifierWithHints(SqlppHint... expectedHints)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002498 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002499{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002500 List<String> list = new ArrayList<String>();
2501 SourceLocation sourceLoc = null;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002502 Token hint = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002503 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002504}
2505{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002506 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002507 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002508 list.add(item);
2509 sourceLoc = getSourceLocation(token);
2510 if (expectedHints != null && expectedHints.length > 0) {
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002511 hint = fetchHint(token, expectedHints);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002512 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002513 }
2514 (<DOT> item = Identifier() { list.add(item); } )*
2515 {
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002516 return new Triple<List<String>, SourceLocation, Token>(list, sourceLoc, hint);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002517 }
2518}
2519
2520Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
2521{
2522 List<String> list = null;
2523}
2524{
2525 list = MultipartIdentifier()
2526 {
2527 int len = list.size();
2528 DataverseName id1 = len > 1 ? DataverseName.create(list, 0, len - 1) : null;
2529 Identifier id2 = new Identifier(list.get(len - 1));
2530 return new Pair<DataverseName,Identifier>(id1, id2);
2531 }
2532}
2533
2534Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
2535{
2536 List<String> list = new ArrayList<String>();
2537 String item = null;
2538}
2539{
2540 item = Identifier() { list.add(item); }
2541 (<DOT> item = Identifier() { list.add(item); } )+
2542 {
2543 int len = list.size();
2544 DataverseName id1 = len > 2 ? DataverseName.create(list, 0, len - 2) : null;
2545 Identifier id2 = new Identifier(list.get(len - 2));
2546 Identifier id3 = new Identifier(list.get(len - 1));
2547 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002548 }
2549}
2550
2551FunctionDecl FunctionDeclaration() throws ParseException:
2552{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002553 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002554 String functionName;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002555 List<Pair<VarIdentifier,TypeExpression>> paramList;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002556 Expression funcBody;
2557 createNewScope();
2558}
2559{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002560 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07002561 functionName = Identifier()
Ian Maxon38fe9402020-01-29 19:27:40 -08002562 paramList = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07002563 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08002564 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07002565 <RIGHTBRACE>
2566 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002567 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, paramList.size());
Xikui Wang3de700a2018-03-15 16:32:55 -07002568 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002569 ensureNoTypeDeclsInFunction(functionName, paramList, null, startToken);
2570 List<VarIdentifier> params = new ArrayList<VarIdentifier>(paramList.size());
2571 for (Pair<VarIdentifier,TypeExpression> p: paramList) {
2572 params.add(p.getFirst());
Ian Maxon38fe9402020-01-29 19:27:40 -08002573 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002574 FunctionDecl stmt = new FunctionDecl(signature, params, funcBody);
Xikui Wang3de700a2018-03-15 16:32:55 -07002575 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002576 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07002577 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002578}
2579
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002580Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002581{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002582 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002583 Expression expr;
2584}
2585{
2586 (
2587 expr = Expression()
2588 |
2589 expr = SelectExpression(false)
2590 )
2591 {
2592 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002593 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07002594 return query;
2595 }
2596}
2597
2598
Yingyi Bu391f09e2015-10-29 13:49:39 -07002599Expression Expression():
2600{
2601 Expression expr = null;
2602 Expression exprP = null;
2603}
2604{
2605(
2606 LOOKAHEAD(2)
2607 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002608 | expr = QuantifiedExpression()
2609)
2610 {
2611 return (exprP==null) ? expr : exprP;
2612 }
2613}
2614
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002615Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002616{
2617 OperatorExpr op = null;
2618 Expression operand = null;
2619}
2620{
2621 operand = AndExpr()
2622 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002623 <OR>
2624 {
2625 if (op == null) {
2626 op = new OperatorExpr();
2627 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07002628 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002629 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002630 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002631 try{
2632 op.addOperator(token.image.toLowerCase());
2633 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002634 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002635 }
Xikui Wang3de700a2018-03-15 16:32:55 -07002636 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002637
Xikui Wang3de700a2018-03-15 16:32:55 -07002638 operand = AndExpr()
2639 {
2640 op.addOperand(operand);
2641 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002642
2643 )*
2644
2645 {
2646 return op==null? operand: op;
2647 }
2648}
2649
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002650Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002651{
2652 OperatorExpr op = null;
2653 Expression operand = null;
2654}
2655{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002656 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002657 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002658 <AND>
2659 {
2660 if (op == null) {
2661 op = new OperatorExpr();
2662 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002663 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002664 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002665 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002666 try{
2667 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002668 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002669 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002670 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002671 }
2672
Yingyi Bu196db5d2016-07-15 19:07:20 -07002673 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002674 {
2675 op.addOperand(operand);
2676 }
2677
2678 )*
2679
2680 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002681 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002682 }
2683}
2684
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002685Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07002686{
2687 Expression inputExpr;
2688 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002689 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07002690}
2691{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002692 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07002693 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002694 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002695 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002696 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002697 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002698 } else {
2699 return inputExpr;
2700 }
2701 }
2702}
Yingyi Bu391f09e2015-10-29 13:49:39 -07002703
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002704Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002705{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002706 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002707 OperatorExpr op = null;
2708 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002709 IExpressionAnnotation annotation = null;
2710}
2711{
Yingyi Bu6c638342016-09-02 17:54:34 -07002712 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002713
2714 (
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002715 LOOKAHEAD(2)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN>)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002716 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002717 Token hintToken = fetchHint(token, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
Shiva2ea73232019-10-09 18:04:05 -07002718 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.HASH_BROADCAST_JOIN_HINT);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002719 if (hintToken != null) {
2720 switch (hintToken.hint) {
2721 case INDEXED_NESTED_LOOP_JOIN_HINT:
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002722 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002723 break;
2724 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002725 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002726 break;
Shiva2ea73232019-10-09 18:04:05 -07002727 case HASH_BROADCAST_JOIN_HINT:
2728 annotation = new BroadcastExpressionAnnotation();
2729 annotation.setObject(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002730 break;
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002731 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002732 }
Yingyi Buea4ec722016-11-04 01:26:16 -07002733
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002734 String operator = token.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07002735 if (operator.equals("<>")){
2736 operator = "!=";
2737 }
2738 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002739 operator = "not_" + operator;
2740 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002741 if (op == null) {
2742 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07002743 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002744 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002745 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002746 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002747 try{
2748 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002749 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002750 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002751 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002752 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002753
Yingyi Bu6c638342016-09-02 17:54:34 -07002754 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07002755 {
Shiva2ea73232019-10-09 18:04:05 -07002756 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07002757 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002758 )?
2759
2760 {
2761 if (annotation != null) {
2762 op.addHint(annotation);
2763 }
2764 return op==null? operand: op;
2765 }
2766}
2767
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002768Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07002769{
2770 boolean not = false;
2771 OperatorExpr op = null;
2772 Expression operand = null;
2773 IExpressionAnnotation annotation = null;
2774}
2775{
2776 operand = IsExpr()
2777 (
2778 LOOKAHEAD(2)
2779 (<NOT> { not = true; })? <BETWEEN>
2780 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002781 Token hintToken = fetchHint(token, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
2782 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT);
2783 if (hintToken != null) {
2784 switch (hintToken.hint) {
2785 case INDEXED_NESTED_LOOP_JOIN_HINT:
Yingyi Bu6c638342016-09-02 17:54:34 -07002786 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002787 break;
2788 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
Yingyi Bu6c638342016-09-02 17:54:34 -07002789 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002790 break;
Yingyi Bu6c638342016-09-02 17:54:34 -07002791 }
2792 }
2793 String operator = token.image.toLowerCase();
2794 if(not){
2795 operator = "not_" + operator;
2796 }
2797 if (op == null) {
2798 op = new OperatorExpr();
2799 op.addOperand(operand);
2800 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002801 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07002802 }
2803 try{
2804 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002805 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002806 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07002807 }
2808 }
2809
2810 operand = IsExpr()
2811 {
2812 op.addOperand(operand);
2813 }
2814
2815 <AND>
2816 operand = IsExpr()
2817 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08002818 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07002819 op.addOperand(operand);
2820 }
2821 )?
2822
2823 {
2824 if (annotation != null) {
2825 op.addHint(annotation);
2826 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002827 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07002828 }
2829}
2830
Yingyi Budaa549c2016-06-28 22:30:52 -07002831Expression IsExpr() throws ParseException:
2832{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002833 Token notToken = null;
2834 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002835 Expression operand = null;
2836 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002837 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002838}
2839{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002840 operand = LikeExpr()
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002841 ( <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002842 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002843 (
2844 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
2845 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08002846 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07002847 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002848 )
Yingyi Budaa549c2016-06-28 22:30:52 -07002849 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002850 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07002851 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002852 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002853 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002854 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07002855 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002856 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07002857 }
2858 }
2859 )?
2860 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002861 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07002862 }
2863}
2864
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002865Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002866{
2867 boolean not = false;
2868 OperatorExpr op = null;
2869 Expression operand = null;
2870}
2871{
2872 operand = ConcatExpr()
2873 (
2874 LOOKAHEAD(2)
2875 (<NOT> { not = true; })? <LIKE>
2876 {
2877 op = new OperatorExpr();
2878 op.addOperand(operand);
2879 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002880 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002881
2882 String operator = token.image.toLowerCase();
2883 if (not) {
2884 operator = "not_" + operator;
2885 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002886 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002887 op.addOperator(operator);
2888 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002889 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002890 }
2891 }
2892
2893 operand = ConcatExpr()
2894 {
2895 op.addOperand(operand);
2896 }
2897 )?
2898
2899 {
2900 return op == null ? operand : op;
2901 }
2902}
2903
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002904Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002905{
2906 OperatorExpr op = null;
2907 Expression operand = null;
2908}
2909{
2910 operand = AddExpr()
2911 (
2912 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002913 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002914 {
2915 if (op == null) {
2916 op = new OperatorExpr();
2917 op.addOperand(operand);
2918 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002919 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002920 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002921 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002922 }
2923 operand = AddExpr()
2924 {
2925 op.addOperand(operand);
2926 }
2927 )*
2928
2929 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002930 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002931 }
2932}
Yingyi Budaa549c2016-06-28 22:30:52 -07002933
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002934Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002935{
2936 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002937 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002938 Expression operand = null;
2939}
2940{
2941 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07002942 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002943 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002944 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002945 {
2946 if (op == null) {
2947 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002948 op.addOperand(operand);
2949 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002950 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002951 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002952 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002953 }
2954
2955 operand = MultExpr()
2956 {
2957 op.addOperand(operand);
2958 }
2959 )*
2960
2961 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002962 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002963 }
2964}
2965
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002966Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002967{
2968 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002969 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002970 Expression operand = null;
2971}
2972{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002973 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002974
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002975 ( (
2976 <MUL> { opType = OperatorType.MUL; } |
2977 <DIVIDE> { opType = OperatorType.DIVIDE; } |
2978 <DIV> { opType = OperatorType.DIV; } |
2979 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
2980 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002981 {
2982 if (op == null) {
2983 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002984 op.addOperand(operand);
2985 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002986 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002987 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002988 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002989 }
2990 operand = ExponentExpr()
2991 {
2992 op.addOperand(operand);
2993 }
2994 )*
2995
2996 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002997 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002998 }
2999}
3000
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003001Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003002{
3003 OperatorExpr op = null;
3004 Expression operand = null;
3005}
3006{
3007 operand = UnaryExpr()
3008 (<CARET>
3009 {
3010 if (op == null) {
3011 op = new OperatorExpr();
3012 op.addOperand(operand);
3013 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003014 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003015 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003016 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003017 }
3018 operand = UnaryExpr()
3019 {
3020 op.addOperand(operand);
3021 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003022 )?
3023 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003024 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003025 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003026}
3027
3028Expression UnaryExpr() throws ParseException:
3029{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003030 boolean not = false;
3031 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003032 Expression expr = null;
3033}
Yingyi Budaa549c2016-06-28 22:30:52 -07003034{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003035 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003036 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003037 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003038 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003039 exprType = "not_" + exprType;
3040 }
3041 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003042 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003043 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003044 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003045 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003046 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07003047 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003048 }
3049 )?
3050
3051 expr = ValueExpr()
3052 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003053 if (uexpr == null) {
3054 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003055 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003056 uexpr.setExpr(expr);
3057 return uexpr;
3058 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003059 }
3060}
3061
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003062Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003063{
3064 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003065 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003066}
3067{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003068 expr = PrimaryExpr()
3069 (
3070 accessor = FieldAccessor(accessor != null ? accessor : expr)
3071 |
3072 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003073 )*
3074 {
3075 return accessor == null ? expr : accessor;
3076 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003077}
3078
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003079FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003080{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003081 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003082 String ident = null;
3083}
3084{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003085 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003086 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003087 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003088 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003089 }
3090}
3091
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003092AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003093{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003094 Token startToken = null;
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003095 boolean star = false, slice = false;
3096 Expression expr1 = null, expr2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003097}
3098{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003099 <LEFTBRACKET> { startToken = token; }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003100 (
3101 <MUL> { star = true; }
3102 |
3103 ( expr1 = Expression() ( <COLON> { slice = true; } ( expr2 = Expression() )? )? )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003104 )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003105 <RIGHTBRACKET>
3106 {
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003107 if (expr1 != null && expr1.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3108 ensureIntegerLiteral( (LiteralExpr) expr1, "Index");
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003109 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003110 if (expr2 != null && expr2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3111 ensureIntegerLiteral( (LiteralExpr) expr2, "Index");
3112 }
3113 AbstractAccessor resultAccessor;
3114 if (slice) {
3115 resultAccessor = new ListSliceExpression(inputExpr, expr1, expr2);
3116 } else if (star) {
3117 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.STAR, null);
3118 } else {
3119 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.ELEMENT, expr1);
3120 }
3121 return addSourceLocation(resultAccessor, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003122 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003123}
3124
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003125Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003126{
3127 Expression expr = null;
3128}
3129{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003130 (
3131 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07003132 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003133 | expr = Literal()
3134 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003135 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003136 | expr = ListConstructor()
3137 | expr = RecordConstructor()
3138 | expr = ParenthesizedExpression()
3139 )
3140 {
3141 return expr;
3142 }
3143}
3144
3145Expression Literal() throws ParseException:
3146{
3147 LiteralExpr lit = new LiteralExpr();
3148 String str = null;
3149}
3150{
3151 ( str = StringLiteral()
3152 {
3153 lit.setValue(new StringLiteral(str));
3154 }
3155 | <INTEGER_LITERAL>
3156 {
Till Westmann68c6a992016-09-30 00:19:12 -07003157 try {
3158 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
3159 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003160 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003161 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003162 }
3163 | <FLOAT_LITERAL>
3164 {
Till Westmann68c6a992016-09-30 00:19:12 -07003165 try {
3166 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
3167 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003168 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003169 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003170 }
3171 | <DOUBLE_LITERAL>
3172 {
Till Westmann68c6a992016-09-30 00:19:12 -07003173 try {
3174 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
3175 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003176 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003177 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003178 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07003179 | <MISSING>
3180 {
3181 lit.setValue(MissingLiteral.INSTANCE);
3182 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003183 | <NULL>
3184 {
3185 lit.setValue(NullLiteral.INSTANCE);
3186 }
3187 | <TRUE>
3188 {
3189 lit.setValue(TrueLiteral.INSTANCE);
3190 }
3191 | <FALSE>
3192 {
3193 lit.setValue(FalseLiteral.INSTANCE);
3194 }
3195 )
3196 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003197 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003198 }
3199}
3200
Yingyi Bu391f09e2015-10-29 13:49:39 -07003201VariableExpr VariableRef() throws ParseException:
3202{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003203 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003204}
3205{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003206 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003207 {
3208 Identifier ident = lookupSymbol(id);
3209 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003210 throw new SqlppParseException(getSourceLocation(token),
3211 "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 -07003212 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003213 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003214 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003215 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003216 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003217 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07003218 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003219 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003220 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003221 }
3222}
3223
Yingyi Bu391f09e2015-10-29 13:49:39 -07003224VariableExpr Variable() throws ParseException:
3225{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003226 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003227}
3228{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003229 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003230 {
3231 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003232 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
3233 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07003234 varExp.setIsNewVar(false);
3235 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003236 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003237 }
3238}
3239
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003240String VariableIdentifier() throws ParseException:
3241{
3242 String id = null;
3243}
3244{
3245 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3246 {
3247 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
3248 }
3249}
3250
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003251Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
3252{
3253 VariableExpr var = null;
3254 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
3255}
3256{
3257 var = Variable()
3258 ( LOOKAHEAD(1)
3259 {
3260 VariableExpr fieldVarExpr = null;
3261 String fieldIdentifierStr = null;
3262 }
3263 <LEFTPAREN>
3264 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3265 {
3266 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3267 }
3268 (<COMMA>
3269 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3270 {
3271 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3272 }
3273 )*
3274 <RIGHTPAREN>
3275 )?
3276 {
3277 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
3278 }
3279}
3280
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003281VariableExpr ExternalVariableRef() throws ParseException:
3282{
3283 String name = null;
3284}
3285{
3286 (
3287 (
3288 <DOLLAR>
3289 (
3290 <INTEGER_LITERAL> { name = token.image; } |
3291 <IDENTIFIER> { name = token.image; } |
3292 name = QuotedString()
3293 )
3294 )
3295 |
3296 (
3297 <QUES> { name = String.valueOf(++externalVarCounter); }
3298 )
3299 )
3300 {
3301 String idName = SqlppVariableUtil.toExternalVariableName(name);
3302 VarIdentifier id = new VarIdentifier(idName);
3303 VariableExpr varExp = new VariableExpr(id);
3304 return addSourceLocation(varExp, token);
3305 }
3306}
3307
Yingyi Bu391f09e2015-10-29 13:49:39 -07003308Expression ListConstructor() throws ParseException:
3309{
3310 Expression expr = null;
3311}
3312{
3313 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003314 expr = OrderedListConstructor() |
3315 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003316 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003317 {
3318 return expr;
3319 }
3320}
3321
Yingyi Bu391f09e2015-10-29 13:49:39 -07003322ListConstructor OrderedListConstructor() throws ParseException:
3323{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003324 Token startToken = null;
3325 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003326}
3327{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003328 <LEFTBRACKET> { startToken = token; }
3329 exprList = ExpressionList()
3330 <RIGHTBRACKET>
3331 {
3332 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003333 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003334 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003335}
3336
3337ListConstructor UnorderedListConstructor() throws ParseException:
3338{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003339 Token startToken = null;
3340 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003341}
3342{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003343 <LEFTDBLBRACE> { startToken = token; }
3344 exprList = ExpressionList()
3345 <RIGHTDBLBRACE>
3346 {
3347 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003348 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003349 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003350}
3351
3352List<Expression> ExpressionList() throws ParseException:
3353{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003354 Expression expr = null;
3355 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003356}
3357{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003358 (
3359 expr = Expression()
3360 {
3361 exprList.add(expr);
3362 }
3363 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07003364 {
3365 exprList.add(expr);
3366 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003367 )*
3368 )?
3369 {
3370 return exprList;
3371 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003372}
3373
Yingyi Bu391f09e2015-10-29 13:49:39 -07003374RecordConstructor RecordConstructor() throws ParseException:
3375{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003376 Token startToken = null;
3377 FieldBinding fb = null;
3378 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003379}
3380{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003381 <LEFTBRACE> { startToken = token; }
3382 (
3383 fb = FieldBinding() { fbList.add(fb); }
3384 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
3385 )?
3386 <RIGHTBRACE>
3387 {
3388 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003389 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003390 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003391}
3392
3393FieldBinding FieldBinding() throws ParseException:
3394{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003395 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003396}
3397{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003398 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003399 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003400 if (right == null) {
3401 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
3402 if (generatedIdentifier == null) {
3403 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
3404 }
3405 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07003406 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
3407 generatedNameExpr.setSourceLocation(left.getSourceLocation());
3408 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003409 } else {
3410 return new FieldBinding(left, right);
3411 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003412 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003413}
3414
Yingyi Bu391f09e2015-10-29 13:49:39 -07003415Expression FunctionCallExpr() throws ParseException:
3416{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003417 List<Expression> argList = new ArrayList<Expression>();
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003418 Expression argExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003419 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07003420 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003421 boolean distinct = false;
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003422 Expression filterExpr = null;
3423 WindowExpression windowExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003424}
3425{
3426 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003427 <LEFTPAREN> (
3428 ( <DISTINCT> { distinct = true; } )?
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003429 ( argExpr = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003430 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003431 if (star) {
3432 if (funcName.function.equalsIgnoreCase(BuiltinFunctions.SCALAR_COUNT.getName())) {
3433 argExpr = new LiteralExpr(new LongIntegerLiteral(1L));
3434 } else {
3435 throw new SqlppParseException(getSourceLocation(token),
3436 "The parameter * can only be used in " + BuiltinFunctions.SCALAR_COUNT.getName() + "().");
Yingyi Buf4d09842016-08-26 00:03:52 -07003437 }
Yingyi Buf4d09842016-08-26 00:03:52 -07003438 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003439 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003440 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003441 (<COMMA> argExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003442 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003443 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003444 }
3445 )*)? <RIGHTPAREN>
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003446
3447 {
3448 String name = funcName.function;
3449 if (distinct) {
3450 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
3451 }
3452 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
3453 int arity = argList.size();
3454 FunctionSignature signature = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
3455 if (signature == null) {
3456 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
3457 }
3458 }
3459
3460 ( <FILTER> <LEFTPAREN> <WHERE> filterExpr = Expression() <RIGHTPAREN> )?
3461
3462 ( LOOKAHEAD(5) windowExpr = WindowExpr(signature, argList, filterExpr) )?
3463
3464 {
3465 if (windowExpr != null) {
3466 return windowExpr;
3467 } else {
3468 CallExpr callExpr = new CallExpr(signature, argList, filterExpr);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06003469 if (funcName.hintToken != null) {
3470 switch (funcName.hintToken.hint) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003471 case INDEXED_NESTED_LOOP_JOIN_HINT:
3472 callExpr.addHint(IndexedNLJoinExpressionAnnotation.INSTANCE);
3473 break;
3474 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
3475 callExpr.addHint(SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE);
3476 break;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06003477 case RANGE_HINT:
3478 try {
3479 RangeAnnotation rangeAnn = new RangeAnnotation();
3480 rangeAnn.setObject((Object) RangeMapBuilder.parseHint(parseExpression(funcName.hintToken.hintParams)));
3481 callExpr.addHint(rangeAnn);
3482 } catch (CompilationException e) {
3483 {
3484 SqlppParseException e2 = new SqlppParseException(getSourceLocation(funcName.hintToken), e.getMessage());
3485 e2.initCause(e);
3486 throw e2;
3487 }
3488 }
3489 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003490 }
3491 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003492 FunctionMapUtil.normalizedListInputFunctions(callExpr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003493 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003494 return callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003495 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003496 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003497}
3498
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003499WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr)
3500 throws ParseException:
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003501{
3502 Boolean fromLast = null, ignoreNulls = null;
3503}
3504{
3505 (
3506 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
3507 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
3508 <FROM> <IDENTIFIER>
3509 {
3510 if (isToken(FIRST)) {
3511 fromLast = false;
3512 } else if (isToken(LAST)) {
3513 fromLast = true;
3514 } else {
3515 throw createUnexpectedTokenError();
3516 }
3517 }
3518 )?
3519 (
3520 // ( RESPECT | IGNORE ) NULLS OVER
3521 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
3522 <IDENTIFIER>
3523 {
3524 if (isToken(RESPECT)) {
3525 ignoreNulls = false;
3526 } else if (isToken(IGNORE)) {
3527 ignoreNulls = true;
3528 } else {
3529 throw createUnexpectedTokenError();
3530 }
3531 }
3532 <IDENTIFIER>
3533 {
3534 if (!isToken(NULLS)) {
3535 throw createUnexpectedTokenError();
3536 }
3537 }
3538 )?
3539 <OVER>
3540 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003541 return OverClause(signature, argList, aggFilterExpr, token, fromLast, ignoreNulls);
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003542 }
3543}
3544
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003545WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr,
3546 Token startToken, Boolean fromLast, Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003547{
3548 Expression partitionExpr = null;
3549 List<Expression> partitionExprs = new ArrayList<Expression>();
3550 OrderbyClause orderByClause = null;
3551 List<Expression> orderbyList = null;
3552 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
3553 WindowExpression.FrameMode frameMode = null;
3554 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
3555 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
3556 Expression frameStartExpr = null, frameEndExpr = null;
3557 WindowExpression.FrameExclusionKind frameExclusionKind = null;
3558 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
3559 VariableExpr windowVar = null;
3560 List<Pair<Expression, Identifier>> windowFieldList = null;
3561}
3562{
3563 (
3564 windowVarWithFieldList = VariableWithFieldMap() <AS>
3565 {
3566 windowVar = windowVarWithFieldList.first;
3567 windowFieldList = windowVarWithFieldList.second;
3568 }
3569 )?
3570 <LEFTPAREN>
3571 (
3572 <IDENTIFIER> { expectToken(PARTITION); } <BY>
3573 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
3574 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
3575 )?
3576 (
3577 orderByClause = OrderbyClause()
3578 {
3579 orderbyList = orderByClause.getOrderbyList();
3580 orderbyModifierList = orderByClause.getModifierList();
3581 }
3582 (
3583 frameMode = WindowFrameMode()
3584 (
3585 frameStart = WindowFrameBoundary() |
3586 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
3587 )
3588 ( frameExclusionKind = WindowFrameExclusion() )?
3589 {
3590 frameStartKind = frameStart.first;
3591 frameStartExpr = frameStart.second;
3592 if (frameEnd == null) {
3593 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3594 } else {
3595 frameEndKind = frameEnd.first;
3596 frameEndExpr = frameEnd.second;
3597 }
3598 if (frameExclusionKind == null) {
3599 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
3600 }
3601 }
3602 )?
3603 )?
3604 <RIGHTPAREN>
3605 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003606 WindowExpression winExpr = new WindowExpression(signature, argList, aggFilterExpr, partitionExprs, orderbyList,
3607 orderbyModifierList, frameMode, frameStartKind, frameStartExpr, frameEndKind, frameEndExpr, frameExclusionKind,
3608 windowVar, windowFieldList, ignoreNulls, fromLast);
3609 return addSourceLocation(winExpr, startToken);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003610 }
3611}
3612
3613WindowExpression.FrameMode WindowFrameMode() throws ParseException:
3614{
3615}
3616{
3617 <IDENTIFIER>
3618 {
3619 if (isToken(RANGE)) {
3620 return WindowExpression.FrameMode.RANGE;
3621 } else if (isToken(ROWS)) {
3622 return WindowExpression.FrameMode.ROWS;
3623 } else if (isToken(GROUPS)) {
3624 return WindowExpression.FrameMode.GROUPS;
3625 } else {
3626 throw createUnexpectedTokenError();
3627 }
3628 }
3629}
3630
3631Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
3632{
3633 boolean current = false;
3634 Expression expr = null;
3635}
3636{
3637 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003638 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003639 | expr = Expression()
3640 )
3641 <IDENTIFIER>
3642 {
3643 WindowExpression.FrameBoundaryKind kind;
3644 if (current && isToken(ROW)) {
3645 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3646 } else if (!current && isToken(PRECEDING)) {
3647 kind = expr == null
3648 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
3649 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
3650 } else if (!current && isToken(FOLLOWING)) {
3651 kind = expr == null
3652 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
3653 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
3654 } else {
3655 throw createUnexpectedTokenError();
3656 }
3657 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
3658 }
3659}
3660
3661WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
3662{
3663 boolean current = false, no = false;
3664}
3665{
3666 <IDENTIFIER>
3667 {
3668 expectToken(EXCLUDE);
3669 }
3670 (
3671 <GROUP>
3672 {
3673 return WindowExpression.FrameExclusionKind.GROUP;
3674 }
3675 |
3676 (
3677 <IDENTIFIER>
3678 {
3679 if (isToken(TIES)) {
3680 return WindowExpression.FrameExclusionKind.TIES;
3681 } else if (isToken(CURRENT)) {
3682 current = true;
3683 } else if (isToken(NO)) {
3684 no = true;
3685 } else {
3686 throw createUnexpectedTokenError();
3687 }
3688 }
3689 <IDENTIFIER>
3690 {
3691 if (current && isToken(ROW)) {
3692 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
3693 } else if (no && isToken(OTHERS)) {
3694 return WindowExpression.FrameExclusionKind.NO_OTHERS;
3695 } else {
3696 throw createUnexpectedTokenError();
3697 }
3698 }
3699 )
3700 )
3701}
3702
Yingyi Bu391f09e2015-10-29 13:49:39 -07003703Expression ParenthesizedExpression() throws ParseException:
3704{
3705 Expression expr;
3706}
3707{
3708 (
3709 LOOKAHEAD(2)
3710 <LEFTPAREN> expr = Expression() <RIGHTPAREN>
3711 |
3712 expr = Subquery()
3713 )
3714 {
3715 return expr;
3716 }
3717}
3718
Yingyi Buc8c067c2016-07-25 23:37:19 -07003719Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003720{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003721 Token startToken = null;
3722 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003723 List<Expression> whenExprs = new ArrayList<Expression>();
3724 List<Expression> thenExprs = new ArrayList<Expression>();
3725 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003726 Expression whenExpr = null;
3727 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003728}
3729{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003730 <CASE> { startToken = token; }
3731 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07003732 (
3733 <WHEN> whenExpr = Expression()
3734 {
3735 whenExprs.add(whenExpr);
3736 }
3737 <THEN> thenExpr = Expression()
3738 {
3739 thenExprs.add(thenExpr);
3740 }
3741 )*
3742 (<ELSE> elseExpr = Expression() )?
3743 <END>
3744 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003745 if (conditionExpr == null) {
3746 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003747 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003748 }
3749 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003750 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07003751 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003752}
3753
Yingyi Buab817482016-08-19 21:29:31 -07003754SelectExpression SelectExpression(boolean subquery) throws ParseException:
3755{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003756 List<LetClause> letClauses = new ArrayList<LetClause>();
3757 SelectSetOperation selectSetOperation;
3758 OrderbyClause orderbyClause = null;
3759 LimitClause limitClause = null;
3760 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07003761}
3762{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003763 ( letClauses = LetClause() )?
3764 selectSetOperation = SelectSetOperation()
3765 (orderbyClause = OrderbyClause() {})?
3766 (limitClause = LimitClause() {})?
3767 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003768 SelectExpression selectExpr =
3769 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
3770 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
3771 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003772 }
3773}
3774
Yingyi Buab817482016-08-19 21:29:31 -07003775SelectSetOperation SelectSetOperation() throws ParseException:
3776{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003777 SetOperationInput setOperationInputLeft;
3778 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
3779}
3780{
3781 {
3782 SelectBlock selectBlockLeft = null;
3783 SelectExpression subqueryLeft = null;
3784 Expression expr = null;
3785 }
3786 selectBlockLeft = SelectBlock()
3787 {
3788 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
3789 }
3790 (
3791 {
3792 SetOpType opType = SetOpType.UNION;
3793 boolean setSemantics = true;
3794 SelectBlock selectBlockRight = null;
3795 SelectExpression subqueryRight = null;
3796 }
3797 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
3798 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
3799 {
3800 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
3801 }
3802 )*
3803 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003804 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
3805 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
3806 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003807 }
3808}
3809
Yingyi Buab817482016-08-19 21:29:31 -07003810SelectExpression Subquery() throws ParseException:
3811{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003812 SelectExpression selectExpr = null;
3813}
3814{
3815 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
3816 {
3817 return selectExpr;
3818 }
3819}
3820
Yingyi Buab817482016-08-19 21:29:31 -07003821SelectBlock SelectBlock() throws ParseException:
3822{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003823 SelectClause selectClause = null;
3824 FromClause fromClause = null;
3825 List<LetClause> fromLetClauses = null;
3826 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003827 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003828 GroupbyClause groupbyClause = null;
3829 List<LetClause> gbyLetClauses = null;
3830 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003831 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003832 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003833}
3834{
3835 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003836 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003837 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003838 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07003839 fromClause = FromClause()
3840 (
3841 fromLetClauses = LetClause()
3842 )?
3843 )
3844 |
3845 (
3846 fromLetClauses = LetClause()
3847 {
3848 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
3849 SourceLocation sourceLoc = getSourceLocation(token);
3850 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
3851 missingExpr.setSourceLocation(sourceLoc);
3852 List<Expression> list = new ArrayList<Expression>(1);
3853 list.add(missingExpr);
3854 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
3855 listExpr.setSourceLocation(sourceLoc);
3856 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
3857 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
3858 fromVar.setSourceLocation(sourceLoc);
3859 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
3860 fromClause = new FromClause(fromTerms);
3861 }
3862 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003863 )?
3864 (whereClause = WhereClause())?
3865 (
3866 groupbyClause = GroupbyClause()
3867 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003868 gbyLetClauses = LetClause()
3869 )?
3870 (havingClause = HavingClause())?
3871 )?
3872 |
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003873 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003874 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003875 fromLetClauses = LetClause()
3876 )?
3877 (whereClause = WhereClause())?
3878 (
3879 groupbyClause = GroupbyClause()
3880 (
3881 gbyLetClauses = LetClause()
3882 )?
3883 (havingClause = HavingClause())?
3884 )?
3885 selectClause = SelectClause()
3886 )
3887 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003888 if (fromLetClauses != null) {
3889 fromLetWhereClauses.addAll(fromLetClauses);
3890 }
3891 if (whereClause != null) {
3892 fromLetWhereClauses.add(whereClause);
3893 }
3894 if (gbyLetClauses != null) {
3895 gbyLetHavingClauses.addAll(gbyLetClauses);
3896 }
3897 if (havingClause != null) {
3898 gbyLetHavingClauses.add(havingClause);
3899 }
3900 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
3901 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003902 selectBlock.setSourceLocation(startSrcLoc);
3903 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003904 }
3905}
3906
Yingyi Buab817482016-08-19 21:29:31 -07003907SelectClause SelectClause() throws ParseException:
3908{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003909 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003910 SelectRegular selectRegular = null;
3911 SelectElement selectElement = null;
3912 boolean distinct = false;
3913}
3914{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003915 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003916 (
3917 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04003918 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07003919 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07003920 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003921 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003922 SourceLocation sourceLoc = getSourceLocation(startToken);
3923 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07003924 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003925 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003926 List<Projection> projections = new ArrayList<Projection>();
3927 projections.add(projection);
3928 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003929 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003930 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003931 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
3932 selectClause.setSourceLocation(sourceLoc);
3933 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003934 }
3935}
3936
Yingyi Buab817482016-08-19 21:29:31 -07003937SelectRegular SelectRegular() throws ParseException:
3938{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003939 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003940 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003941 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003942}
3943{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003944 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003945 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003946 projections.add(projection);
3947 startSrcLoc = projection.getSourceLocation();
3948 }
3949 ( LOOKAHEAD(2) <COMMA> projection = Projection()
3950 {
3951 projections.add(projection);
3952 }
3953 )*
3954 {
3955 SelectRegular selectRegular = new SelectRegular(projections);
3956 selectRegular.setSourceLocation(startSrcLoc);
3957 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003958 }
3959}
3960
Yingyi Buab817482016-08-19 21:29:31 -07003961SelectElement SelectElement() throws ParseException:
3962{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003963 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003964 Expression expr = null;
3965 String name = null;
3966}
3967{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003968 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003969 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003970 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003971 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003972 }
3973}
3974
Yingyi Buab817482016-08-19 21:29:31 -07003975Projection Projection() throws ParseException :
3976{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003977 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003978 Expression expr = null;
3979 Identifier identifier = null;
3980 String name = null;
3981 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003982 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003983}
3984{
3985 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003986 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
3987 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003988 | expr = Expression() ((<AS>)? name = Identifier())?
3989 {
3990 if (name == null) {
3991 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
3992 if (generatedColumnIdentifier != null) {
3993 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
3994 }
3995 }
3996 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003997 )
3998 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003999 Projection projection = new Projection(expr, name, star, varStar);
4000 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
4001 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004002 }
4003}
4004
4005FromClause FromClause() throws ParseException :
4006{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004007 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004008 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
4009 extendCurrentScope();
4010}
4011{
4012 {
4013 FromTerm fromTerm = null;
4014 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004015 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004016 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
4017 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004018 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004019 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004020 }
4021}
4022
4023FromTerm FromTerm() throws ParseException :
4024{
4025 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004026 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004027 VariableExpr posVar = null;
4028 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
4029}
4030{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004031 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004032 (
4033 {JoinType joinType = JoinType.INNER; }
4034 (joinType = JoinType())?
4035 {
4036 AbstractBinaryCorrelateClause correlateClause = null;
4037 }
4038 (correlateClause = JoinClause(joinType)
Yingyi Bu6d999ed2016-07-13 11:59:06 -07004039 | correlateClause = UnnestClause(joinType)
Yingyi Bu391f09e2015-10-29 13:49:39 -07004040 )
4041 {
4042 correlateClauses.add(correlateClause);
4043 }
4044 )*
4045 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004046 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07004047 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004048 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004049 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
4050 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
4051 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004052 }
4053}
4054
4055JoinClause JoinClause(JoinType joinType) throws ParseException :
4056{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004057 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004058 Expression rightExpr = null;
4059 VariableExpr rightVar = null;
4060 VariableExpr posVar = null;
4061 Expression conditionExpr = null;
4062}
4063{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004064 <JOIN> { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())? <ON> conditionExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004065 {
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004066 if(rightVar==null){
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07004067 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004068 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004069 JoinClause joinClause = new JoinClause(joinType, rightExpr, rightVar, posVar, conditionExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004070 return addSourceLocation(joinClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004071 }
4072}
4073
Yingyi Bu391f09e2015-10-29 13:49:39 -07004074UnnestClause UnnestClause(JoinType joinType) throws ParseException :
4075{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004076 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004077 Expression rightExpr;
4078 VariableExpr rightVar;
4079 VariableExpr posVar = null;
4080}
4081{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004082 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable()) (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004083 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004084 if (rightVar == null) {
4085 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004086 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004087 UnnestClause unnestClause = new UnnestClause(joinType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004088 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004089 }
4090}
4091
Yingyi Bu391f09e2015-10-29 13:49:39 -07004092JoinType JoinType() throws ParseException :
4093{
4094 JoinType joinType = JoinType.INNER;
4095}
4096{
4097 (<INNER>|<LEFT> (<OUTER>)? {joinType = JoinType.LEFTOUTER; })
4098 {
4099 return joinType;
4100 }
4101}
4102
4103List<LetClause> LetClause() throws ParseException:
4104{
4105 List<LetClause> letList = new ArrayList<LetClause>();
4106 LetClause letClause;
4107}
4108{
4109 (
4110 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
4111 |
4112 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
4113 )
4114 {
4115 return letList;
4116 }
4117}
4118
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004119WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004120{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004121 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004122 Expression whereExpr;
4123}
4124{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004125 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004126 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004127 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004128 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004129 }
4130}
4131
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004132OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004133{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004134 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004135 OrderbyClause oc = new OrderbyClause();
4136 Expression orderbyExpr;
4137 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004138 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004139 int numOfOrderby = 0;
4140}
4141{
4142 <ORDER>
4143 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004144 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004145 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
4146 if (hintToken != null) {
4147 switch (hintToken.hint) {
4148 case INMEMORY_HINT:
4149 String[] splits = hintToken.hintParams.split("\\s+");
4150 int numFrames = Integer.parseInt(splits[0]);
4151 int numTuples = Integer.parseInt(splits[1]);
4152 oc.setNumFrames(numFrames);
4153 oc.setNumTuples(numTuples);
4154 break;
4155 case RANGE_HINT:
4156 try {
4157 oc.setRangeMap(RangeMapBuilder.parseHint(parseExpression(hintToken.hintParams)));
4158 } catch (CompilationException e) {
4159 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
4160 }
4161 break;
Ali Alsuliman80225e22018-10-15 14:17:07 -07004162 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004163 }
4164 }
4165 <BY> orderbyExpr = Expression()
4166 {
4167 orderbyList.add(orderbyExpr);
4168 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
4169 }
4170 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
4171 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
4172 {
4173 modifierList.add(modif);
4174 }
4175
4176 (LOOKAHEAD(2) <COMMA> orderbyExpr = Expression()
4177 {
4178 orderbyList.add(orderbyExpr);
4179 modif = OrderbyClause.OrderModifier.ASC;
4180 }
4181 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
4182 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
4183 {
4184 modifierList.add(modif);
4185 }
4186 )*
4187
4188 {
4189 oc.setModifierList(modifierList);
4190 oc.setOrderbyList(orderbyList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004191 return addSourceLocation(oc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004192 }
4193}
4194
4195GroupbyClause GroupbyClause()throws ParseException :
4196{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004197 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004198 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004199 List<List<GbyVariableExpressionPair>> gbyList = null;
4200 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004201 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07004202 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004203 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004204}
4205{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004206 {
4207 Scope newScope = extendCurrentScopeNoPush(true);
4208 // extendCurrentScope(true);
4209 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004210 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004211 {
4212 startToken = token;
4213 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
4214 if (hintToken != null) {
4215 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004216 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004217 }
4218 <BY> groupingElementList = GroupingElementList()
4219 (
4220 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004221 {
4222 groupVar = groupVarWithFieldList.first;
4223 groupFieldList = groupVarWithFieldList.second;
4224 }
Yingyi Buacc12a92016-03-26 17:25:05 -07004225 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004226 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004227 if (groupingSetsParser == null) {
4228 groupingSetsParser = new SqlppGroupingSetsParser();
4229 }
4230 SourceLocation sourceLoc = getSourceLocation(startToken);
4231 try {
4232 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
4233 } catch (CompilationException e) {
4234 throw new SqlppParseException(sourceLoc, e.getMessage());
4235 }
4236 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004237 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07004238 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07004239 gbc.setGroupVar(groupVar);
4240 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004241 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004242 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004243 }
4244}
4245
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004246List<GroupingElement> GroupingElementList() throws ParseException:
4247{
4248 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
4249 GroupingElement groupingElement = null;
4250}
4251{
4252 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
4253 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
4254 {
4255 return groupingElementList;
4256 }
4257}
4258
4259GroupingElement GroupingElement() throws ParseException:
4260{
4261 GroupingElement groupingElement = null;
4262 List<GroupingSet> groupingSets = null;
4263 List<GroupingElement> groupingElements = null;
4264}
4265{
4266 (
4267 LOOKAHEAD(2)
4268 <LEFTPAREN> <RIGHTPAREN>
4269 {
4270 groupingElement = GroupingSet.EMPTY;
4271 }
4272 |
4273 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
4274 <IDENTIFIER> { expectToken(ROLLUP); }
4275 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4276 {
4277 groupingElement = new RollupCube(groupingSets, false);
4278 }
4279 |
4280 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
4281 <IDENTIFIER> { expectToken(CUBE); }
4282 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4283 {
4284 groupingElement = new RollupCube(groupingSets, true);
4285 }
4286 |
4287 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
4288 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
4289 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
4290 {
4291 groupingElement = new GroupingSets(groupingElements);
4292 }
4293 |
4294 groupingElement = OrdinaryGroupingSet()
4295 )
4296 {
4297 return groupingElement;
4298 }
4299}
4300
4301GroupingSet OrdinaryGroupingSet() throws ParseException:
4302{
4303 GbyVariableExpressionPair gbyExprPair = null;
4304 List<GbyVariableExpressionPair> items = null;
4305}
4306{
4307 (
4308 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
4309 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
4310 )
4311 {
4312 return new GroupingSet(items);
4313 }
4314}
4315
4316List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
4317{
4318 GroupingSet groupingSet = null;
4319 List<GroupingSet> items = new ArrayList<GroupingSet>();
4320}
4321{
4322 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
4323 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
4324 {
4325 return items;
4326 }
4327}
4328
4329List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
4330{
4331 GbyVariableExpressionPair gbyExprPair = null;
4332 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
4333}
4334{
4335 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
4336 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
4337 {
4338 return items;
4339 }
4340}
4341
4342GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
4343{
4344 Expression expr = null;
4345 VariableExpr var = null;
4346}
4347{
4348 expr = Expression() ( (<AS>)? var = Variable() )?
4349 {
4350 return new GbyVariableExpressionPair(var, expr);
4351 }
4352}
4353
Yingyi Bu391f09e2015-10-29 13:49:39 -07004354HavingClause HavingClause() throws ParseException:
4355{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004356 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004357 Expression filterExpr = null;
4358}
4359{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004360 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004361 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004362 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004363 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004364 }
4365}
4366
4367LimitClause LimitClause() throws ParseException:
4368{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004369 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004370 LimitClause lc = new LimitClause();
4371 Expression expr;
4372 pushForbiddenScope(getCurrentScope());
4373}
4374{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004375 <LIMIT> { startToken = token; } expr = Expression() { lc.setLimitExpr(expr); }
4376 (<OFFSET> expr = Expression() { lc.setOffset(expr); })?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004377
4378 {
4379 popForbiddenScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004380 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004381 }
4382}
4383
4384QuantifiedExpression QuantifiedExpression()throws ParseException:
4385{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004386 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004387 QuantifiedExpression qc = new QuantifiedExpression();
4388 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
4389 Expression satisfiesExpr;
4390 VariableExpr var;
4391 Expression inExpr;
4392 QuantifiedPair pair;
4393}
4394{
4395 {
4396 createNewScope();
4397 }
4398
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004399 ( ((<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); })
4400 | (<EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); }))
Yingyi Bu391f09e2015-10-29 13:49:39 -07004401 var = Variable() <IN> inExpr = Expression()
4402 {
4403 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004404 quantifiedList.add(pair);
4405 }
4406 (
4407 <COMMA> var = Variable() <IN> inExpr = Expression()
4408 {
4409 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004410 quantifiedList.add(pair);
4411 }
4412 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07004413 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004414 {
4415 qc.setSatisfiesExpr(satisfiesExpr);
4416 qc.setQuantifiedList(quantifiedList);
4417 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004418 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004419 }
4420}
4421
4422LetClause LetElement() throws ParseException:
4423{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004424 VariableExpr varExp;
4425 Expression beExp;
4426 extendCurrentScope();
4427}
4428{
4429 varExp = Variable() <EQ> beExp = Expression()
4430 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004431 LetClause lc = new LetClause(varExp, beExp);
4432 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004433 return lc;
4434 }
4435}
4436
4437LetClause WithElement() throws ParseException:
4438{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004439 VariableExpr varExp;
4440 Expression beExp;
4441 extendCurrentScope();
4442}
4443{
4444 varExp = Variable() <AS> beExp = Expression()
4445 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004446 LetClause lc = new LetClause(varExp, beExp);
4447 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004448 return lc;
4449 }
4450}
4451
4452TOKEN_MGR_DECLS:
4453{
4454 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07004455 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004456 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004457
4458 public void pushState() {
4459 lexerStateStack.push( curLexState );
4460 }
4461
4462 public void popState(String token) {
4463 if (lexerStateStack.size() > 0) {
4464 SwitchTo( lexerStateStack.pop() );
4465 } else {
4466 int errorLine = input_stream.getEndLine();
4467 int errorColumn = input_stream.getEndColumn();
4468 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
4469 + "\" but state stack is empty.";
4470 throw new TokenMgrError(msg, -1);
4471 }
4472 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004473
4474 void CommonTokenAction(Token token) {
4475 Token hintToken = token.specialToken;
4476 if (hintToken != null) {
4477 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
4478 String text = hintToken.image.substring(1).trim();
4479 boolean hintFound = hintToken.parseHint(text);
4480 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
4481 }
4482 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004483}
4484
4485<DEFAULT,IN_DBL_BRACE>
4486TOKEN [IGNORE_CASE]:
4487{
Ian Maxon38fe9402020-01-29 19:27:40 -08004488 <ADAPTER: "adapter">
4489 | <ALL : "all">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004490 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07004491 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004492 | <APPLY : "apply">
4493 | <AS : "as">
4494 | <ASC : "asc">
4495 | <AT : "at">
4496 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004497 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004498 | <BTREE : "btree">
4499 | <BY : "by">
4500 | <CASE : "case">
4501 | <CLOSED : "closed">
4502 | <CREATE : "create">
4503 | <COMPACTION : "compaction">
4504 | <COMPACT : "compact">
4505 | <CONNECT : "connect">
4506 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07004507 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07004508 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07004509 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004510 | <DECLARE : "declare">
4511 | <DEFINITION : "definition">
4512 | <DELETE : "delete">
4513 | <DESC : "desc">
4514 | <DISCONNECT : "disconnect">
4515 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004516 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004517 | <DROP : "drop">
4518 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07004519 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004520 | <ELSE : "else">
4521 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07004522 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004523 | <EVERY : "every">
4524 | <EXCEPT : "except">
4525 | <EXISTS : "exists">
4526 | <EXTERNAL : "external">
4527 | <FEED : "feed">
4528 | <FILTER : "filter">
4529 | <FLATTEN : "flatten">
4530 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004531 | <FROM : "from">
4532 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08004533 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004534 | <FUNCTION : "function">
4535 | <GROUP : "group">
4536 | <HAVING : "having">
4537 | <HINTS : "hints">
4538 | <IF : "if">
4539 | <INTO : "into">
4540 | <IN : "in">
4541 | <INDEX : "index">
4542 | <INGESTION : "ingestion">
4543 | <INNER : "inner">
4544 | <INSERT : "insert">
4545 | <INTERNAL : "internal">
4546 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07004547 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004548 | <JOIN : "join">
4549 | <KEYWORD : "keyword">
4550 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07004551 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004552 | <LEFT : "left">
4553 | <LETTING : "letting">
4554 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004555 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004556 | <LIMIT : "limit">
4557 | <LOAD : "load">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004558 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004559 | <NODEGROUP : "nodegroup">
4560 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07004561 | <NOT : "not">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004562 | <OFFSET : "offset">
4563 | <ON : "on">
4564 | <OPEN : "open">
4565 | <OR : "or">
4566 | <ORDER : "order">
4567 | <OUTER : "outer">
4568 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004569 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004570 | <PATH : "path">
4571 | <POLICY : "policy">
4572 | <PRESORTED : "pre-sorted">
4573 | <PRIMARY : "primary">
4574 | <RAW : "raw">
4575 | <REFRESH : "refresh">
4576 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004577 | <RETURNING : "returning">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004578 | <RTREE : "rtree">
4579 | <RUN : "run">
4580 | <SATISFIES : "satisfies">
4581 | <SECONDARY : "secondary">
4582 | <SELECT : "select">
4583 | <SET : "set">
4584 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08004585 | <START : "start">
4586 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08004587 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03004588 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07004589 | <THEN : "then">
4590 | <TYPE : "type">
4591 | <TO : "to">
4592 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08004593 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004594 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07004595 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004596 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07004597 | <USE : "use">
4598 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004599 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08004600 | <VALUED : "valued">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004601 | <WHEN : "when">
4602 | <WHERE : "where">
4603 | <WITH : "with">
4604 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004605}
4606
4607<DEFAULT,IN_DBL_BRACE>
4608TOKEN :
4609{
4610 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07004611 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004612 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004613 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004614 | <MUL : "*">
4615 | <PLUS : "+">
4616
4617 | <LEFTPAREN : "(">
4618 | <RIGHTPAREN : ")">
4619 | <LEFTBRACKET : "[">
4620 | <RIGHTBRACKET : "]">
4621
4622 | <ATT : "@">
4623 | <COLON : ":">
4624 | <COMMA : ",">
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004625 | <DOLLAR: "$">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004626 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004627 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004628 | <QUES : "?">
4629 | <SEMICOLON : ";">
4630 | <SHARP : "#">
4631
4632 | <LT : "<">
4633 | <GT : ">">
4634 | <LE : "<=">
4635 | <GE : ">=">
4636 | <EQ : "=">
4637 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07004638 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004639 | <SIMILAR : "~=">
4640}
4641
4642<DEFAULT,IN_DBL_BRACE>
4643TOKEN :
4644{
4645 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
4646}
4647
4648<DEFAULT>
4649TOKEN :
4650{
4651 <RIGHTBRACE : "}"> { popState("}"); }
4652}
4653
4654<DEFAULT,IN_DBL_BRACE>
4655TOKEN :
4656{
4657 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
4658}
4659
4660<IN_DBL_BRACE>
4661TOKEN :
4662{
4663 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
4664}
4665
4666<DEFAULT,IN_DBL_BRACE>
4667TOKEN :
4668{
4669 <INTEGER_LITERAL : (<DIGIT>)+ >
4670}
4671
4672<DEFAULT,IN_DBL_BRACE>
Yingyi Bue311a632016-06-07 18:23:16 -07004673TOKEN [IGNORE_CASE]:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004674{
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004675 <MISSING : "missing">
4676 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004677 | <TRUE : "true">
4678 | <FALSE : "false">
4679}
4680
4681<DEFAULT,IN_DBL_BRACE>
4682TOKEN :
4683{
4684 <#DIGIT : ["0" - "9"]>
4685}
4686
4687<DEFAULT,IN_DBL_BRACE>
4688TOKEN:
4689{
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07004690 < DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
4691 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
4692 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004693 >
Yingyi Bue4d919e2016-10-30 10:47:03 -07004694 | < FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
4695 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
4696 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004697 >
4698 | <DIGITS : (<DIGIT>)+ >
4699}
4700
4701<DEFAULT,IN_DBL_BRACE>
4702TOKEN :
4703{
4704 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08004705 | <#IDENTIFIER_SPECIALCHARS_START : ["_"]>
4706 | <#IDENTIFIER_SPECIALCHARS_REST : ["$"]>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004707}
4708
4709<DEFAULT,IN_DBL_BRACE>
4710TOKEN :
4711{
4712 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07004713 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004714 <EscapeQuot>
4715 | <EscapeBslash>
4716 | <EscapeSlash>
4717 | <EscapeBspace>
4718 | <EscapeFormf>
4719 | <EscapeNl>
4720 | <EscapeCr>
4721 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004722 | ~["`","\\"])* "`">
4723 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004724 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004725 | <EscapeBslash>
4726 | <EscapeSlash>
4727 | <EscapeBspace>
4728 | <EscapeFormf>
4729 | <EscapeNl>
4730 | <EscapeCr>
4731 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004732 | ~["\"","\\"])* "\"")
4733 | ("\'"(
4734 <EscapeApos>
4735 | <EscapeBslash>
4736 | <EscapeSlash>
4737 | <EscapeBspace>
4738 | <EscapeFormf>
4739 | <EscapeNl>
4740 | <EscapeCr>
4741 | <EscapeTab>
4742 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004743 | < #EscapeQuot: "\\\"" >
4744 | < #EscapeApos: "\\\'" >
4745 | < #EscapeBslash: "\\\\" >
4746 | < #EscapeSlash: "\\/" >
4747 | < #EscapeBspace: "\\b" >
4748 | < #EscapeFormf: "\\f" >
4749 | < #EscapeNl: "\\n" >
4750 | < #EscapeCr: "\\r" >
4751 | < #EscapeTab: "\\t" >
4752}
4753
4754<DEFAULT,IN_DBL_BRACE>
4755TOKEN :
4756{
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08004757 <IDENTIFIER : ( <LETTER> | <IDENTIFIER_SPECIALCHARS_START> )
4758 ( <LETTER> | <DIGIT> | <IDENTIFIER_SPECIALCHARS_START> | <IDENTIFIER_SPECIALCHARS_REST> )*>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004759}
4760
4761<DEFAULT,IN_DBL_BRACE>
4762SKIP:
4763{
4764 " "
4765 | "\t"
4766 | "\r"
4767 | "\n"
4768}
4769
4770<DEFAULT,IN_DBL_BRACE>
4771SKIP:
4772{
4773 <"//" (~["\n"])* "\n">
4774}
4775
4776<DEFAULT,IN_DBL_BRACE>
4777SKIP:
4778{
4779 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4780}
4781
4782<DEFAULT,IN_DBL_BRACE>
4783SKIP:
4784{
Yingyi Bu93846a72016-09-13 16:30:39 -07004785 <"--" (~["\n"])* "\n">
4786}
4787
4788
4789<DEFAULT,IN_DBL_BRACE>
4790SKIP:
4791{
4792 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4793}
4794
4795<DEFAULT,IN_DBL_BRACE>
4796SKIP:
4797{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004798 <"/*"> { pushState(); } : INSIDE_COMMENT
4799}
4800
4801<INSIDE_COMMENT>
4802SPECIAL_TOKEN:
4803{
4804 <"+"(" ")*(~["*"])*>
4805}
4806
4807<INSIDE_COMMENT>
4808SKIP:
4809{
4810 <"/*"> { pushState(); }
4811}
4812
4813<INSIDE_COMMENT>
4814SKIP:
4815{
4816 <"*/"> { popState("*/"); }
4817 | <~[]>
4818}