blob: fcd06eb047f487f0a942fa05f7416dbd4967b89a [file] [log] [blame]
Michael Blow82464fb2017-03-28 18:48:13 -04001//
2// Licensed to the Apache Software Foundation (ASF) under one
3// or more contributor license agreements. See the NOTICE file
4// distributed with this work for additional information
5// regarding copyright ownership. The ASF licenses this file
6// to you under the Apache License, Version 2.0 (the
7// "License"); you may not use this file except in compliance
8// with the License. You may obtain a copy of the License at
9//
10// http://www.apache.org/licenses/LICENSE-2.0
11//
12// Unless required by applicable law or agreed to in writing,
13// software distributed under the License is distributed on an
14// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15// KIND, either express or implied. See the License for the
16// specific language governing permissions and limitations
17// under the License.
18//
Yingyi Bu391f09e2015-10-29 13:49:39 -070019options {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -070020 COMMON_TOKEN_ACTION = true;
21 STATIC = false;
22 TOKEN_EXTENDS = "org.apache.asterix.lang.sqlpp.parser.SqlppToken";
Yingyi Bu391f09e2015-10-29 13:49:39 -070023}
24
Yingyi Bu391f09e2015-10-29 13:49:39 -070025PARSER_BEGIN(SQLPPParser)
26
27package org.apache.asterix.lang.sqlpp.parser;
28
29// For SQL++ ParserTokenManager
Till Westmanne9b2adf2016-10-15 12:39:01 -070030import java.util.ArrayDeque;
Yingyi Bu391f09e2015-10-29 13:49:39 -070031
32import java.io.BufferedReader;
33import java.io.File;
34import java.io.FileInputStream;
35import java.io.FileNotFoundException;
36import java.io.IOException;
37import java.io.InputStreamReader;
38import java.io.Reader;
39import java.io.StringReader;
40import java.util.ArrayList;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070041import java.util.Arrays;
42import java.util.Collection;
Yingyi Budaa549c2016-06-28 22:30:52 -070043import java.util.Collections;
Yingyi Bu391f09e2015-10-29 13:49:39 -070044import java.util.HashMap;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070045import java.util.HashSet;
Yingyi Bu391f09e2015-10-29 13:49:39 -070046import java.util.Iterator;
47import java.util.LinkedHashMap;
48import java.util.List;
49import java.util.Map;
50
51import org.apache.asterix.common.annotations.AutoDataGen;
52import org.apache.asterix.common.annotations.DateBetweenYearsDataGen;
53import org.apache.asterix.common.annotations.DatetimeAddRandHoursDataGen;
54import org.apache.asterix.common.annotations.DatetimeBetweenYearsDataGen;
55import org.apache.asterix.common.annotations.FieldIntervalDataGen;
56import org.apache.asterix.common.annotations.FieldValFileDataGen;
57import org.apache.asterix.common.annotations.FieldValFileSameIndexDataGen;
58import org.apache.asterix.common.annotations.IRecordFieldDataGen;
59import org.apache.asterix.common.annotations.InsertRandIntDataGen;
60import org.apache.asterix.common.annotations.ListDataGen;
61import org.apache.asterix.common.annotations.ListValFileDataGen;
62import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
63import org.apache.asterix.common.annotations.TypeDataGen;
64import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
65import org.apache.asterix.common.config.DatasetConfig.DatasetType;
66import org.apache.asterix.common.config.DatasetConfig.IndexType;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080067import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070068import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070069import org.apache.asterix.common.exceptions.WarningCollector;
70import org.apache.asterix.common.exceptions.WarningUtil;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080071import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070072import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin54c06012019-11-13 15:54:20 -080073import org.apache.asterix.common.metadata.DataverseName;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080074import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070075import org.apache.asterix.lang.common.base.AbstractLangExpression;
76import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070077import org.apache.asterix.lang.common.base.Expression;
78import org.apache.asterix.lang.common.base.Literal;
79import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070080import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070081import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070082import org.apache.asterix.lang.common.base.Statement;
83import org.apache.asterix.lang.common.clause.GroupbyClause;
84import org.apache.asterix.lang.common.clause.LetClause;
85import org.apache.asterix.lang.common.clause.LimitClause;
86import org.apache.asterix.lang.common.clause.OrderbyClause;
87import org.apache.asterix.lang.common.clause.UpdateClause;
88import org.apache.asterix.lang.common.clause.WhereClause;
89import org.apache.asterix.lang.common.context.RootScopeFactory;
90import org.apache.asterix.lang.common.context.Scope;
91import org.apache.asterix.lang.common.expression.AbstractAccessor;
92import org.apache.asterix.lang.common.expression.CallExpr;
93import org.apache.asterix.lang.common.expression.FieldAccessor;
94import org.apache.asterix.lang.common.expression.FieldBinding;
95import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
96import org.apache.asterix.lang.common.expression.IfExpr;
97import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -070098import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -070099import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +0300100import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700101import org.apache.asterix.lang.common.expression.LiteralExpr;
102import org.apache.asterix.lang.common.expression.OperatorExpr;
103import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
104import org.apache.asterix.lang.common.expression.QuantifiedExpression;
105import org.apache.asterix.lang.common.expression.RecordConstructor;
106import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
107import org.apache.asterix.lang.common.expression.TypeExpression;
108import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
109import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700110import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
111import org.apache.asterix.lang.common.expression.VariableExpr;
112import org.apache.asterix.lang.common.literal.DoubleLiteral;
113import org.apache.asterix.lang.common.literal.FalseLiteral;
114import org.apache.asterix.lang.common.literal.FloatLiteral;
115import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700116import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700117import org.apache.asterix.lang.common.literal.NullLiteral;
118import org.apache.asterix.lang.common.literal.StringLiteral;
119import org.apache.asterix.lang.common.literal.TrueLiteral;
120import org.apache.asterix.lang.common.parser.ScopeChecker;
121import org.apache.asterix.lang.common.statement.CompactStatement;
122import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800123import org.apache.asterix.lang.common.statement.StartFeedStatement;
124import org.apache.asterix.lang.common.statement.StopFeedStatement;
Ian Maxon38fe9402020-01-29 19:27:40 -0800125import org.apache.asterix.lang.common.statement.CreateAdapterStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700126import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
127import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
128import org.apache.asterix.lang.common.statement.CreateFeedStatement;
129import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
130import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800131import org.apache.asterix.lang.common.statement.CreateSynonymStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700132import org.apache.asterix.lang.common.statement.DatasetDecl;
133import org.apache.asterix.lang.common.statement.DataverseDecl;
134import org.apache.asterix.lang.common.statement.DataverseDropStatement;
135import org.apache.asterix.lang.common.statement.DeleteStatement;
136import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700137import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700138import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
139import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300140import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700141import org.apache.asterix.lang.common.statement.FunctionDecl;
142import org.apache.asterix.lang.common.statement.FunctionDropStatement;
143import org.apache.asterix.lang.common.statement.IndexDropStatement;
144import org.apache.asterix.lang.common.statement.InsertStatement;
145import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
146import org.apache.asterix.lang.common.statement.LoadStatement;
147import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
148import org.apache.asterix.lang.common.statement.NodegroupDecl;
149import org.apache.asterix.lang.common.statement.Query;
150import org.apache.asterix.lang.common.statement.RefreshExternalDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700151import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800152import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700153import org.apache.asterix.lang.common.statement.TypeDecl;
154import org.apache.asterix.lang.common.statement.TypeDropStatement;
155import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800156import org.apache.asterix.lang.common.statement.UpsertStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700157import org.apache.asterix.lang.common.statement.WriteStatement;
158import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800159import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700160import org.apache.asterix.lang.common.struct.QuantifiedPair;
161import org.apache.asterix.lang.common.struct.VarIdentifier;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700162import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700163import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
164import org.apache.asterix.lang.sqlpp.clause.FromClause;
165import org.apache.asterix.lang.sqlpp.clause.FromTerm;
166import org.apache.asterix.lang.sqlpp.clause.HavingClause;
167import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700168import org.apache.asterix.lang.sqlpp.clause.Projection;
169import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
170import org.apache.asterix.lang.sqlpp.clause.SelectClause;
171import org.apache.asterix.lang.sqlpp.clause.SelectElement;
172import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
173import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
174import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800175import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700176import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700177import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700178import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700179import org.apache.asterix.lang.sqlpp.optype.JoinType;
180import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700181import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700182import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
183import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700184import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700185import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700186import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Ian Maxon38fe9402020-01-29 19:27:40 -0800187import org.apache.asterix.external.dataset.adapter.AdapterIdentifier;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800188import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700189import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700190import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700191import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700192import org.apache.hyracks.algebricks.common.utils.Pair;
193import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800194import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700195import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700196import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
197import org.apache.hyracks.algebricks.core.algebra.expressions.IndexedNLJoinExpressionAnnotation;
198import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700199import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700200import org.apache.hyracks.api.exceptions.Warning;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800201import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700202import org.apache.hyracks.util.StringUtil;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700203
Yingyi Bucaea8f02015-11-16 15:12:15 -0800204class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700205
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800206 // tokens parsed as identifiers
207 private static final String CURRENT = "CURRENT";
208 private static final String EXCLUDE = "EXCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700209 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800210 private static final String FOLLOWING = "FOLLOWING";
211 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700212 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700213 private static final String LAST = "LAST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800214 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700215 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800216 private static final String OTHERS = "OTHERS";
217 private static final String PARTITION = "PARTITION";
218 private static final String PRECEDING = "PRECEDING";
219 private static final String RANGE = "RANGE";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700220 private static final String RESPECT = "RESPECT";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800221 private static final String ROW = "ROW";
222 private static final String ROWS = "ROWS";
223 private static final String TIES = "TIES";
224 private static final String UNBOUNDED = "UNBOUNDED";
Ian Maxon38fe9402020-01-29 19:27:40 -0800225 private static final String ACTION = "ACTION";
226 private static final String LANGUAGE = "LANGUAGE";
227 private static final String CALL = "CALL";
228 private static final String DETERMINISTIC = "DETERMINISTIC";
229 private static final String RETURNS = "RETURNS";
230 private static final String INLINE = "INLINE";
231
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800232
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700233 private static final String INT_TYPE_NAME = "int";
234
Till Westmann7199a562016-09-17 16:07:32 -0700235 // error configuration
236 protected static final boolean REPORT_EXPECTED_TOKENS = false;
237
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700238 private int externalVarCounter;
239
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800240 private DataverseName defaultDataverse;
241
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700242 private final WarningCollector warningCollector = new WarningCollector();
243
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700244 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
245
Yingyi Bu391f09e2015-10-29 13:49:39 -0700246 private static class IndexParams {
247 public IndexType type;
248 public int gramLength;
249
250 public IndexParams(IndexType type, int gramLength) {
251 this.type = type;
252 this.gramLength = gramLength;
253 }
254 };
255
256 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800257 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700258 public String library;
259 public String function;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700260 public SqlppHint hint;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700261 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700262 }
263
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700264 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
265 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
266 switch (hintToken.hint) {
267 case VAL_FILE_HINT:
268 File[] valFiles = new File[splits.length];
269 for (int k=0; k<splits.length; k++) {
270 valFiles[k] = new File(splits[k]);
271 }
272 return new FieldValFileDataGen(valFiles);
273 case VAL_FILE_SAME_INDEX_HINT:
274 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
275 case LIST_VAL_FILE_HINT:
276 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
277 case LIST_HINT:
278 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
279 case INTERVAL_HINT:
280 FieldIntervalDataGen.ValueType vt;
281 switch (splits[0]) {
282 case "int":
283 vt = FieldIntervalDataGen.ValueType.INT;
284 break;
285 case "long":
286 vt = FieldIntervalDataGen.ValueType.LONG;
287 break;
288 case "float":
289 vt = FieldIntervalDataGen.ValueType.FLOAT;
290 break;
291 case "double":
292 vt = FieldIntervalDataGen.ValueType.DOUBLE;
293 break;
294 default:
295 throw new SqlppParseException(getSourceLocation(hintToken),
296 "Unknown type for interval data gen: " + splits[0]);
297 }
298 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
299 case INSERT_RAND_INT_HINT:
300 return new InsertRandIntDataGen(splits[0], splits[1]);
301 case DATE_BETWEEN_YEARS_HINT:
302 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
303 case DATETIME_BETWEEN_YEARS_HINT:
304 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
305 case DATETIME_ADD_RAND_HOURS_HINT:
306 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
307 case AUTO_HINT:
308 return new AutoDataGen(splits[0]);
309 default:
310 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700311 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700312 }
313
Till Westmann7199a562016-09-17 16:07:32 -0700314 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700315 this(new StringReader(s));
316 super.setInput(s);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700317 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700318 }
319
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800320 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700321 File file = new File(args[0]);
322 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
323 SQLPPParser parser = new SQLPPParser(fis);
324 List<Statement> st = parser.parse();
325 //st.accept(new SQLPPPrintVisitor(), 0);
326 }
327
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800328 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700329 return parseImpl(new ParseFunction<List<Statement>>() {
330 @Override
331 public List<Statement> parse() throws ParseException {
332 return SQLPPParser.this.Statement();
333 }
334 });
335 }
336
337 private Expression parseExpression() throws CompilationException {
338 return parseImpl(new ParseFunction<Expression>() {
339 @Override
340 public Expression parse() throws ParseException {
341 return SQLPPParser.this.Expression();
342 }
343 });
344 }
345
346 private static Expression parseExpression(String text) throws CompilationException {
347 return new SQLPPParser(text).parseExpression();
348 }
349
350 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700351 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700352 hintCollector.clear();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700353 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700354 return parseFunction.parse();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700355 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700356 // 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 -0700357 // 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 -0700358 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700359 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700360 } catch (SqlppParseException e) {
Ali Alsulimanfe901892019-03-19 01:40:35 -0700361 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(), LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700362 } catch (ParseException e) {
Ali Alsulimanfe901892019-03-19 01:40:35 -0700363 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700364 } finally {
365 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700366 }
367 }
Till Westmann7199a562016-09-17 16:07:32 -0700368
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700369 @FunctionalInterface
370 private interface ParseFunction<T> {
371 T parse() throws ParseException;
372 }
373
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700374 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700375 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
376 warningCollector.getWarnings(outWarnings, maxWarnings);
377 }
378
379 @Override
380 public long getTotalWarningsCount() {
381 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700382 }
383
Till Westmann7199a562016-09-17 16:07:32 -0700384 protected String getMessage(ParseException pe) {
385 Token currentToken = pe.currentToken;
386 if (currentToken == null) {
387 return pe.getMessage();
388 }
389 int[][] expectedTokenSequences = pe.expectedTokenSequences;
390 String[] tokenImage = pe.tokenImage;
391 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
392 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
393 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
394 Token tok = currentToken.next;
395 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700396 StringBuilder message = new StringBuilder(128);
397 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700398 for (int i = 0; i < maxSize; i++) {
399 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700400 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700401 }
402 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700403 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700404 break;
405 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700406 final String fixedTokenImage = tokenImage[tok.kind];
407 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700408 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700409 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700410 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700411 tok = tok.next;
412 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700413 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700414 if (REPORT_EXPECTED_TOKENS) {
415 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700416 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700417 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700418 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700419 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700420 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700421 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700422 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700423 }
424
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700425 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700426 return
427 token == null ? null :
428 token.sourceLocation != null ? token.sourceLocation :
429 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700430 }
431
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700432 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
433 expr.setSourceLocation(getSourceLocation(token));
434 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700435 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800436
437 private boolean isToken(String image) {
438 return token.image.equalsIgnoreCase(image);
439 }
440
441 private void expectToken(String image) throws SqlppParseException {
442 if (!isToken(image)) {
443 throw createUnexpectedTokenError();
444 }
445 }
446
447 private SqlppParseException createUnexpectedTokenError() {
Ali Alsuliman587b7902019-01-21 14:33:50 -0800448 return new SqlppParseException(getSourceLocation(token), "Unexpected token: " + LogRedactionUtil.userData(token.image));
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800449 }
450
451 private boolean laToken(int idx, int kind, String image) {
452 Token t = getToken(idx);
453 return t.kind == kind && t.image.equalsIgnoreCase(image);
454 }
455
456 private boolean laIdentifier(int idx, String image) {
457 return laToken(idx, IDENTIFIER, image);
458 }
459
460 private boolean laIdentifier(String image) {
461 return laIdentifier(1, image);
462 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700463
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700464 private Token fetchHint(Token token, SqlppHint... expectedHints) {
465 Token hintToken = token.specialToken;
466 if (hintToken == null) {
467 return null;
468 }
469 SourceLocation sourceLoc = getSourceLocation(hintToken);
470 hintCollector.remove(sourceLoc);
471 if (hintToken.hint == null) {
472 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
473 return null;
474 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
475 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
476 return null;
477 } else {
478 return hintToken;
479 }
480 }
481
482 private void reportUnclaimedHints() {
483 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
484 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
485 }
486 }
487
488 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
489 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
490 }
491
492 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700493 if (warningCollector.shouldWarn()) {
494 warningCollector.warn(WarningUtil.forAsterix(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
495 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700496 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700497}
498
499PARSER_END(SQLPPParser)
500
Yingyi Bu391f09e2015-10-29 13:49:39 -0700501List<Statement> Statement() throws ParseException:
502{
503 scopeStack.push(RootScopeFactory.createRootScope(this));
504 List<Statement> decls = new ArrayList<Statement>();
505 Statement stmt = null;
506}
507{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300508 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700509 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300510 {
511 decls.add(stmt);
512 }
513 )?
514 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700515 )*
516 <EOF>
517 {
518 return decls;
519 }
520}
521
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700522Statement ExplainStatement() throws ParseException:
523{
524 Statement stmt = null;
525 Token explainToken = null;
526}
527{
528 ( <EXPLAIN> { explainToken = token; } )?
529 stmt = SingleStatement()
530 {
531 if (explainToken != null) {
532 if (stmt.getKind() == Statement.Kind.QUERY) {
533 ((Query)stmt).setExplain(true);
534 } else {
535 throw new SqlppParseException(getSourceLocation(explainToken),
536 "EXPLAIN is not supported for this kind of statement");
537 }
538 }
539 return stmt;
540 }
541}
542
Yingyi Bu391f09e2015-10-29 13:49:39 -0700543Statement SingleStatement() throws ParseException:
544{
545 Statement stmt = null;
546}
547{
548 (
549 stmt = DataverseDeclaration()
550 | stmt = FunctionDeclaration()
551 | stmt = CreateStatement()
552 | stmt = LoadStatement()
553 | stmt = DropStatement()
554 | stmt = WriteStatement()
555 | stmt = SetStatement()
556 | stmt = InsertStatement()
557 | stmt = DeleteStatement()
558 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800559 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700560 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700561 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700562 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700563 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700564 )
565 {
566 return stmt;
567 }
568}
569
570DataverseDecl DataverseDeclaration() throws ParseException:
571{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700572 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800573 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700574}
575{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800576 <USE> { startToken = token; } dvName = MultipartIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700577 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800578 defaultDataverse = DataverseName.create(dvName);
579 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700580 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700581 }
582}
583
584Statement CreateStatement() throws ParseException:
585{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700586 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700587 Statement stmt = null;
588}
589{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700590 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700591 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800592 stmt = CreateTypeStatement(startToken)
593 | stmt = CreateNodegroupStatement(startToken)
594 | stmt = CreateDatasetStatement(startToken)
595 | stmt = CreateIndexStatement(startToken)
596 | stmt = CreateDataverseStatement(startToken)
597 | stmt = CreateFunctionStatement(startToken)
Ian Maxon38fe9402020-01-29 19:27:40 -0800598 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800599 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800600 | stmt = CreateFeedStatement(startToken)
601 | stmt = CreateFeedPolicyStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700602 )
603 {
604 return stmt;
605 }
606}
607
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800608TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
609{
610 TypeDecl stmt = null;
611}
612{
613 <TYPE> stmt = TypeSpecification(startStmtToken)
614 {
615 return stmt;
616 }
617}
618
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700619TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700620{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800621 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700622 boolean ifNotExists = false;
623 TypeExpression typeExpr = null;
624}
625{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800626 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700627 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800628 {
629 boolean dgen = false;
630 long numValues = -1;
631 String filename = null;
632 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
633 if (hintToken != null) {
634 String hintParams = hintToken.hintParams;
635 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
636 if (splits == null || splits.length != 2) {
637 throw new SqlppParseException(getSourceLocation(hintToken),
638 "Expecting /*+ dgen <filename> <numberOfItems> */");
639 }
640 dgen = true;
641 filename = splits[0];
642 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700643 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800644 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
645 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
646 return addSourceLocation(stmt, startStmtToken);
647 }
648}
649
650NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
651{
652 NodegroupDecl stmt = null;
653}
654{
655 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
656 {
657 return stmt;
658 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700659}
660
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700661NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700662{
663 String name = null;
664 String tmp = null;
665 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800666 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700667}
668{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800669 name = Identifier() ifNotExists = IfNotExists()
670 <ON> tmp = Identifier()
671 {
672 ncNames.add(new Identifier(tmp));
673 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700674 ( <COMMA> tmp = Identifier()
675 {
676 ncNames.add(new Identifier(tmp));
677 }
678 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800679 {
680 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
681 return addSourceLocation(stmt, startStmtToken);
682 }
683}
684
685void Dataset() throws ParseException:
686{
687}
688{
689 (<DATASET>|<COLLECTION>)
690}
691
692DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
693{
694 DatasetDecl stmt = null;
695}
696{
697 (
698 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
699 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
700 )
701 {
702 return stmt;
703 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700704}
705
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700706DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700707{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800708 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700709 boolean ifNotExists = false;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800710 Pair<DataverseName,Identifier> typeComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700711 String adapterName = null;
712 Map<String,String> properties = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700713 FunctionSignature appliedFunction = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800714 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700715 String nodeGroupName = null;
716 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700717 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700718 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800719 Pair<Integer, List<String>> filterField = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800720 Pair<DataverseName,Identifier> metaTypeComponents = new Pair<DataverseName, Identifier>(null, null);
Till Westmannf3aa19f2017-12-01 17:42:35 -0800721 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700722}
723{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800724 nameComponents = QualifiedName()
725 <LEFTPAREN> typeComponents = TypeName() <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -0700726 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800727 { String name; }
728 <WITH>
729 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700730 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800731 if (!name.equalsIgnoreCase("meta")){
732 throw new SqlppParseException(getSourceLocation(startStmtToken),
733 "We can only support one additional associated field called \"meta\".");
734 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700735 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800736 <LEFTPAREN> metaTypeComponents = TypeName() <RIGHTPAREN>
737 )?
738 ifNotExists = IfNotExists()
739 primaryKeyFields = PrimaryKey()
740 (<AUTOGENERATED> { autogenerated = true; } )?
741 (<ON> nodeGroupName = Identifier() )?
742 ( <HINTS> hints = Properties() )?
743 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
744 ( <WITH> withRecord = RecordConstructor() )?
745 {
746 if(filterField!=null && filterField.first!=0){
747 throw new SqlppParseException(getSourceLocation(startStmtToken),
748 "A filter field can only be a field in the main record of the dataset.");
749 }
750 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
751 filterField == null? null : filterField.second);
752 try {
753 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeComponents.first, typeComponents.second,
754 metaTypeComponents.first, metaTypeComponents.second,
755 nodeGroupName != null ? new Identifier(nodeGroupName) : null, hints, DatasetType.INTERNAL, idd, withRecord,
756 ifNotExists);
757 return addSourceLocation(stmt, startStmtToken);
758 } catch (CompilationException e) {
759 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
760 }
761 }
762}
763
764DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
765{
766 Pair<DataverseName,Identifier> nameComponents = null;
767 boolean ifNotExists = false;
768 Pair<DataverseName,Identifier> typeComponents = null;
769 String adapterName = null;
770 Map<String,String> properties = null;
771 String nodeGroupName = null;
772 Map<String,String> hints = new HashMap<String,String>();
773 DatasetDecl stmt = null;
774 Pair<DataverseName,Identifier> metaTypeComponents = new Pair<DataverseName, Identifier>(null, null);
775 RecordConstructor withRecord = null;
776}
777{
778 nameComponents = QualifiedName()
779 <LEFTPAREN> typeComponents = TypeName() <RIGHTPAREN>
780 ifNotExists = IfNotExists()
781 <USING> adapterName = AdapterName() properties = Configuration()
782 ( <ON> nodeGroupName = Identifier() )?
783 ( <HINTS> hints = Properties() )?
784 ( <WITH> withRecord = RecordConstructor() )?
785 {
786 ExternalDetailsDecl edd = new ExternalDetailsDecl();
787 edd.setAdapter(adapterName);
788 edd.setProperties(properties);
789 try {
790 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeComponents.first, typeComponents.second,
791 metaTypeComponents.first, metaTypeComponents.second,
792 nodeGroupName != null? new Identifier(nodeGroupName): null, hints, DatasetType.EXTERNAL, edd, withRecord,
793 ifNotExists);
794 return addSourceLocation(stmt, startStmtToken);
795 } catch (CompilationException e) {
796 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
797 }
798 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700799}
800
801RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
802{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700803 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800804 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700805 String datasetName = null;
806}
807{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800808 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
809 {
810 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
811 stmt.setDataverseName(nameComponents.first);
812 stmt.setDatasetName(nameComponents.second);
813 return addSourceLocation(stmt, startToken);
814 }
815}
816
817CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
818{
819 CreateIndexStatement stmt = null;
820}
821{
822 (
823 <INDEX> stmt = IndexSpecification(startStmtToken)
824 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
825 )
826 {
827 return stmt;
828 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700829}
830
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700831CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700832{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700833 CreateIndexStatement stmt = new CreateIndexStatement();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700834 String indexName = null;
835 boolean ifNotExists = false;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800836 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700837 Pair<Integer, Pair<List<String>, IndexedTypeExpression>> fieldPair = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700838 IndexParams indexType = null;
839 boolean enforced = false;
840}
841{
Ali Alsuliman8351d252017-09-24 00:43:15 -0700842 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800843 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -0700844 <ON> nameComponents = QualifiedName()
845 <LEFTPAREN> ( fieldPair = OpenField()
846 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700847 stmt.addFieldExprPair(fieldPair.second);
848 stmt.addFieldIndexIndicator(fieldPair.first);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700849 }
Ali Alsuliman8351d252017-09-24 00:43:15 -0700850 ) (<COMMA> fieldPair = OpenField()
851 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700852 stmt.addFieldExprPair(fieldPair.second);
853 stmt.addFieldIndexIndicator(fieldPair.first);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700854 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800855 )* <RIGHTPAREN> ( <TYPE> indexType = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman8351d252017-09-24 00:43:15 -0700856 )
857 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700858 stmt.setIndexName(new Identifier(indexName));
859 stmt.setIfNotExists(ifNotExists);
860 stmt.setDataverseName(nameComponents.first);
861 stmt.setDatasetName(nameComponents.second);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700862 if (indexType != null) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700863 stmt.setIndexType(indexType.type);
864 stmt.setGramLength(indexType.gramLength);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700865 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700866 stmt.setEnforced(enforced);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700867 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -0700868 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700869}
870
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800871CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
872{
873 CreateIndexStatement stmt = new CreateIndexStatement();
874 String indexName = null;
875 boolean ifNotExists = false;
876 Pair<DataverseName,Identifier> nameComponents = null;
877}
878{
879 (indexName = Identifier())? ifNotExists = IfNotExists()
880 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
881 {
882 if (indexName == null) {
883 indexName = "primary_idx_" + nameComponents.second;
884 }
885 stmt.setIndexName(new Identifier(indexName));
886 stmt.setIfNotExists(ifNotExists);
887 stmt.setDataverseName(nameComponents.first);
888 stmt.setDatasetName(nameComponents.second);
889 return addSourceLocation(stmt, startStmtToken);
890 }
891}
892
Yingyi Bu391f09e2015-10-29 13:49:39 -0700893String CompactionPolicy() throws ParseException :
894{
895 String compactionPolicy = null;
896}
897{
898 compactionPolicy = Identifier()
899 {
900 return compactionPolicy;
901 }
902}
903
904String FilterField() throws ParseException :
905{
906 String filterField = null;
907}
908{
909 filterField = Identifier()
910 {
911 return filterField;
912 }
913}
914
915IndexParams IndexType() throws ParseException:
916{
917 IndexType type = null;
918 int gramLength = 0;
919}
920{
921 (<BTREE>
922 {
923 type = IndexType.BTREE;
924 }
925 | <RTREE>
926 {
927 type = IndexType.RTREE;
928 }
929 | <KEYWORD>
930 {
931 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
932 }
Taewoo Kimc49405a2017-01-04 00:30:43 -0800933 |<FULLTEXT>
934 {
935 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
936 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700937 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
938 {
939 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
940 gramLength = Integer.valueOf(token.image);
941 }
942 <RIGHTPAREN>)
943 {
944 return new IndexParams(type, gramLength);
945 }
946}
947
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800948CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
949{
950 CreateDataverseStatement stmt = null;
951}
952{
953 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
954 {
955 return stmt;
956 }
957}
958
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700959CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -0700960{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800961 List<String> dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700962 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700963}
964{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800965 dvName = MultipartIdentifier() ifNotExists = IfNotExists()
966 {
967 CreateDataverseStatement stmt = new CreateDataverseStatement(DataverseName.create(dvName), null, ifNotExists);
968 return addSourceLocation(stmt, startStmtToken);
969 }
970}
971
972CreateFunctionStatement CreateFunctionStatement(Token startStmtToken) throws ParseException:
973{
974 CreateFunctionStatement stmt = null;
975}
976{
977 <FUNCTION> stmt = FunctionSpecification(startStmtToken)
978 {
979 return stmt;
980 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700981}
982
Ian Maxon38fe9402020-01-29 19:27:40 -0800983
984CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
985{
986 CreateAdapterStatement stmt = null;
987}
988{
989 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
990 {
991 return stmt;
992 }
993}
994
995CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
996{
997 AdapterIdentifier signature = null;
998 Token beginPos;
999 Token endPos;
1000 Pair<DataverseName,Identifier> adaptName = null;
1001 DataverseName currentDataverse = defaultDataverse;
1002 String lang = null;
1003 String libName ="";
1004 String externalIdent = "";
1005 ListConstructor resources = null;
1006 boolean ifNotExists = false;
1007}
1008{
1009 adaptName = QualifiedName()
1010 <IDENTIFIER> {expectToken(LANGUAGE);} lang = Identifier() <AS> libName = ConstantString() <COMMA> externalIdent = ConstantString() ifNotExists = IfNotExists()
1011 {
1012 signature = new AdapterIdentifier(adaptName.getFirst(),adaptName.getSecond().getValue());
1013 CreateAdapterStatement stmt =
1014 new CreateAdapterStatement(signature, lang, libName, externalIdent, ifNotExists);
1015 return addSourceLocation(stmt, startStmtToken);
1016 }
1017
1018}
1019
1020
1021List<Pair<VarIdentifier,IndexedTypeExpression>> FunctionParameters() :
1022{
1023 List<Pair<VarIdentifier,IndexedTypeExpression>> paramList = new ArrayList<Pair<VarIdentifier,IndexedTypeExpression>>();
1024 Pair<VarIdentifier,IndexedTypeExpression> varPair = new Pair<VarIdentifier,IndexedTypeExpression>(null,null);
1025 VarIdentifier var = null;
1026 IndexedTypeExpression type = null;
1027 boolean typed = false;
1028}
1029{
1030 <LEFTPAREN>( (<IDENTIFIER>
1031 {
1032 var = SqlppVariableUtil.toInternalVariableIdentifier(token.image);
1033 varPair.setFirst(new VarIdentifier(var.toString()));
1034 paramList.add(varPair);
1035 } (LOOKAHEAD(3)(<COLON>)? type = IndexedTypeExpr() {varPair.setSecond(type);} )? (<COMMA> <IDENTIFIER>{
1036 var = SqlppVariableUtil.toInternalVariableIdentifier(token.image);
1037 varPair = new Pair<VarIdentifier,IndexedTypeExpression>(null,null);
1038 varPair.setFirst(new VarIdentifier(var.toString()));
1039 paramList.add(varPair);
1040 } (LOOKAHEAD(3)(<COLON>)? type = IndexedTypeExpr() {varPair.setSecond(type);})?)*)? ) <RIGHTPAREN>
1041
1042 {
1043 return paramList;
1044 }
1045}
1046
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001047CreateFunctionStatement FunctionSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001048{
Ian Maxon38fe9402020-01-29 19:27:40 -08001049 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001050 boolean ifNotExists = false;
Ian Maxon38fe9402020-01-29 19:27:40 -08001051 String functionBody = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001052 VarIdentifier var = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001053 Expression functionBodyExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001054 Token beginPos;
1055 Token endPos;
1056 FunctionName fctName = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001057 DataverseName currentDataverse = defaultDataverse;
Ian Maxon38fe9402020-01-29 19:27:40 -08001058 IndexedTypeExpression returnType = null;
1059 boolean deterministic = false;
1060 boolean nullCall = false;
1061 String lang = null;
1062 String libName ="";
1063 String externalIdent = "";
1064 List<Pair<VarIdentifier,IndexedTypeExpression>> args = new ArrayList<Pair<VarIdentifier,IndexedTypeExpression>>();
1065 RecordConstructor resources = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001066}
1067{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001068 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001069 {
1070 defaultDataverse = fctName.dataverse;
1071 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001072 ifNotExists = IfNotExists()
Ian Maxon38fe9402020-01-29 19:27:40 -08001073 args = FunctionParameters()
1074 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = IndexedTypeExpr() )?
1075 (
1076 (
1077 <LEFTBRACE>
1078 {
1079 createNewScope();
1080 beginPos = token;
1081 }
1082 (functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression())
1083 <RIGHTBRACE>{
1084 endPos = token;
1085 functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine, endPos.beginColumn);
1086 signature = new FunctionSignature(fctName.dataverse, fctName.function, args.size());
1087 getCurrentScope().addFunctionDescriptor(signature, false);
1088 removeCurrentScope();
1089 defaultDataverse = currentDataverse;
1090 CreateFunctionStatement stmt = new CreateFunctionStatement(signature, args, returnType, functionBody, functionBodyExpr, ifNotExists);
1091 return addSourceLocation(stmt, startStmtToken);
1092 }
1093 )
1094 |
1095 <IDENTIFIER> {expectToken(LANGUAGE);}
1096 (
1097 LOOKAHEAD({laIdentifier(INLINE)})
1098 (
1099 <IDENTIFIER> <AS> {
1100 createNewScope();
1101 beginPos = token;
1102 }
1103 (functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression())
1104 {
1105 endPos = token;
1106 functionBody = extractFragment(beginPos.endLine, beginPos.beginColumn+1, endPos.endLine, endPos.endColumn+1);
1107 signature = new FunctionSignature(fctName.dataverse, fctName.function, args.size());
1108 getCurrentScope().addFunctionDescriptor(signature, false);
1109 removeCurrentScope();
1110 defaultDataverse = currentDataverse;
1111 CreateFunctionStatement stmt = new CreateFunctionStatement(signature, args, returnType, functionBody, functionBodyExpr, ifNotExists);
1112 return addSourceLocation(stmt, startStmtToken);
1113 }
1114 )
1115 |
1116 (
1117 lang = Identifier()
1118 ( (<NOT> <IDENTIFIER> {expectToken(DETERMINISTIC); deterministic = false;}) | (<IDENTIFIER> {expectToken(DETERMINISTIC); deterministic = true;}) )?
1119 (<NULL> <IDENTIFIER> { expectToken(CALL); nullCall = true;})?
1120 <AS> libName = ConstantString() <COMMA> externalIdent = ConstantString()
1121 (<WITH> resources = RecordConstructor())?
1122 {
1123 signature = new FunctionSignature(fctName.dataverse, fctName.function, args.size());
1124 defaultDataverse = currentDataverse;
1125 CreateFunctionStatement stmt = null;
1126 try{
1127 stmt =
1128 new CreateFunctionStatement(signature, args, returnType, deterministic, nullCall,
1129 lang, libName, externalIdent, resources, ifNotExists);
1130 } catch (AlgebricksException e) {
1131 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1132 }
1133 return addSourceLocation(stmt, startStmtToken);
1134 }
1135 )
1136 )
1137 )
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001138}
1139
1140CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1141{
1142 CreateFeedStatement stmt = null;
1143}
1144{
1145 <FEED> stmt = FeedSpecification(startStmtToken)
1146 {
1147 return stmt;
1148 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001149}
1150
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001151CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001152{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001153 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001154 boolean ifNotExists = false;
1155 String adapterName = null;
1156 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001157 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001158 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001159 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001160}
1161{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001162 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001163 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001164 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001165 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001166 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001167 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001168 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001169 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001170 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001171 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001172}
1173
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001174CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1175{
1176 CreateFeedPolicyStatement stmt = null;
1177}
1178{
1179 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1180 {
1181 return stmt;
1182 }
1183}
1184
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001185CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001186{
Michael Blowd6cf6412016-06-30 02:44:35 -04001187 String policyName = null;
1188 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001189 String sourcePolicyFile = null;
1190 String definition = null;
1191 boolean ifNotExists = false;
1192 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001193 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001194}
1195{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001196 policyName = Identifier() ifNotExists = IfNotExists()
1197 <FROM>
1198 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001199 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001200 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001201 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001202 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1203 {
1204 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1205 }
1206 )
1207 {
1208 return addSourceLocation(stmt, startStmtToken);
1209 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001210}
1211
Yingyi Bu391f09e2015-10-29 13:49:39 -07001212
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001213CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
1214{
1215 CreateSynonymStatement stmt = null;
1216}
1217{
1218 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
1219 {
1220 return stmt;
1221 }
1222}
1223
1224CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
1225{
1226 Pair<DataverseName,Identifier> nameComponents = null;
1227 Pair<DataverseName,Identifier> objectNameComponents = null;
1228 boolean ifNotExists = false;
1229}
1230{
1231 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1232 <FOR> objectNameComponents = QualifiedName()
1233 {
1234 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
1235 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
1236 return addSourceLocation(stmt, startStmtToken);
1237 }
1238}
1239
Yingyi Bu391f09e2015-10-29 13:49:39 -07001240boolean IfNotExists() throws ParseException:
1241{
1242}
1243{
Yingyi Budaa549c2016-06-28 22:30:52 -07001244 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001245 {
1246 return true;
1247 }
1248 )?
1249 {
1250 return false;
1251 }
1252}
1253
Xikui Wang9d63f622017-05-18 17:50:44 -07001254void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001255{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001256 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07001257 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001258}
1259{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001260 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001261 {
1262 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1263 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
1264 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07001265 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001266 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07001267 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001268 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1269 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07001270 }
1271 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07001272}
1273
1274String GetPolicy() throws ParseException:
1275{
1276 String policy = null;
1277}
1278{
1279 <USING> <POLICY> policy = Identifier()
1280 {
1281 return policy;
1282 }
1283
1284}
1285
1286FunctionSignature FunctionSignature() throws ParseException:
1287{
1288 FunctionName fctName = null;
1289 int arity = 0;
1290}
1291{
1292 fctName = FunctionName() <ATT> <INTEGER_LITERAL>
1293 {
1294 arity = new Integer(token.image);
1295 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001296 throw new SqlppParseException(getSourceLocation(token), "Invalid arity:" + arity);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001297 }
1298
1299 // TODO use fctName.library
1300 String fqFunctionName = fctName.library == null ? fctName.function : fctName.library + "#" + fctName.function;
1301 return new FunctionSignature(fctName.dataverse, fqFunctionName, arity);
1302 }
1303}
1304
Yingyi Buc9bfe252016-03-01 00:02:40 -08001305Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001306{
Yingyi Buc9bfe252016-03-01 00:02:40 -08001307 Pair<Integer, List<String>> tmp = null;
1308 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001309 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
1310}
1311{
1312 <PRIMARY> <KEY> tmp = NestedField()
1313 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001314 keyFieldSourceIndicators.add(tmp.first);
1315 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001316 }
1317 ( <COMMA> tmp = NestedField()
1318 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001319 keyFieldSourceIndicators.add(tmp.first);
1320 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001321 }
1322 )*
1323 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001324 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001325 }
1326}
1327
1328Statement DropStatement() throws ParseException:
1329{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001330 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001331 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001332}
1333{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001334 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001335 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001336 stmt = DropDatasetStatement(startToken)
1337 | stmt = DropIndexStatement(startToken)
1338 | stmt = DropNodeGroupStatement(startToken)
1339 | stmt = DropTypeStatement(startToken)
1340 | stmt = DropDataverseStatement(startToken)
1341 | stmt = DropFunctionStatement(startToken)
1342 | stmt = DropFeedStatement(startToken)
1343 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001344 | stmt = DropSynonymStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07001345 )
1346 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001347 return stmt;
1348 }
1349}
1350
1351DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
1352{
1353 DropDatasetStatement stmt = null;
1354}
1355{
1356 Dataset() stmt = DropDatasetSpecification(startStmtToken)
1357 {
1358 return stmt;
1359 }
1360}
1361
1362DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
1363{
1364 Pair<DataverseName,Identifier> pairId = null;
1365 boolean ifExists = false;
1366}
1367{
1368 pairId = QualifiedName() ifExists = IfExists()
1369 {
1370 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
1371 return addSourceLocation(stmt, startStmtToken);
1372 }
1373}
1374
1375IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
1376{
1377 IndexDropStatement stmt = null;
1378}
1379{
1380 <INDEX> stmt = DropIndexSpecification(startStmtToken)
1381 {
1382 return stmt;
1383 }
1384}
1385
1386IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
1387{
1388 Triple<DataverseName,Identifier,Identifier> tripleId = null;
1389 boolean ifExists = false;
1390}
1391{
1392 tripleId = DoubleQualifiedName() ifExists = IfExists()
1393 {
1394 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
1395 return addSourceLocation(stmt, startStmtToken);
1396 }
1397}
1398
1399NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
1400{
1401 NodeGroupDropStatement stmt = null;
1402}
1403{
1404 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
1405 {
1406 return stmt;
1407 }
1408}
1409
1410NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
1411{
1412 String id = null;
1413 boolean ifExists = false;
1414}
1415{
1416 id = Identifier() ifExists = IfExists()
1417 {
1418 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
1419 return addSourceLocation(stmt, startStmtToken);
1420 }
1421}
1422
1423TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
1424{
1425 TypeDropStatement stmt = null;
1426}
1427{
1428 <TYPE> stmt = DropTypeSpecification(startStmtToken)
1429 {
1430 return stmt;
1431 }
1432}
1433
1434TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
1435{
1436 Pair<DataverseName,Identifier> pairId = null;
1437 boolean ifExists = false;
1438}
1439{
1440 pairId = TypeName() ifExists = IfExists()
1441 {
1442 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
1443 return addSourceLocation(stmt, startStmtToken);
1444 }
1445}
1446
1447DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
1448{
1449 DataverseDropStatement stmt = null;
1450}
1451{
1452 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
1453 {
1454 return stmt;
1455 }
1456}
1457
1458DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
1459{
1460 List<String> multipartId = null;
1461 boolean ifExists = false;
1462}
1463{
1464 multipartId = MultipartIdentifier() ifExists = IfExists()
1465 {
1466 DataverseDropStatement stmt = new DataverseDropStatement(DataverseName.create(multipartId), ifExists);
1467 return addSourceLocation(stmt, startStmtToken);
1468 }
1469}
1470
1471FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
1472{
1473 FunctionDropStatement stmt = null;
1474}
1475{
1476 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
1477 {
1478 return stmt;
1479 }
1480}
1481
1482FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
1483{
1484 FunctionSignature funcSig = null;
1485 boolean ifExists = false;
1486}
1487{
1488 funcSig = FunctionSignature() ifExists = IfExists()
1489 {
1490 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
1491 return addSourceLocation(stmt, startStmtToken);
1492 }
1493}
1494
1495FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
1496{
1497 FeedDropStatement stmt = null;
1498}
1499{
1500 <FEED> stmt = DropFeedSpecification(startStmtToken)
1501 {
1502 return stmt;
1503 }
1504}
1505
1506FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
1507{
1508 Pair<DataverseName,Identifier> pairId = null;
1509 boolean ifExists = false;
1510}
1511{
1512 pairId = QualifiedName() ifExists = IfExists()
1513 {
1514 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
1515 return addSourceLocation(stmt, startStmtToken);
1516 }
1517}
1518
1519FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
1520{
1521 FeedPolicyDropStatement stmt = null;
1522}
1523{
1524 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
1525 {
1526 return stmt;
1527 }
1528}
1529
1530FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
1531{
1532 Pair<DataverseName,Identifier> pairId = null;
1533 boolean ifExists = false;
1534}
1535{
1536 pairId = QualifiedName() ifExists = IfExists()
1537 {
1538 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
1539 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001540 }
1541}
1542
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001543SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
1544{
1545 SynonymDropStatement stmt = null;
1546}
1547{
1548 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
1549 {
1550 return stmt;
1551 }
1552}
1553
1554SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
1555{
1556 Pair<DataverseName,Identifier> pairId = null;
1557 boolean ifExists = false;
1558}
1559{
1560 pairId = QualifiedName() ifExists = IfExists()
1561 {
1562 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
1563 return addSourceLocation(stmt, startStmtToken);
1564 }
1565}
1566
Yingyi Bu391f09e2015-10-29 13:49:39 -07001567boolean IfExists() throws ParseException :
1568{
1569}
1570{
1571 ( LOOKAHEAD(1) <IF> <EXISTS>
1572 {
1573 return true;
1574 }
1575 )?
1576 {
1577 return false;
1578 }
1579}
1580
1581InsertStatement InsertStatement() throws ParseException:
1582{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001583 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001584 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001585 VariableExpr var = null;
1586 Query query = null;
1587 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001588}
1589{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001590 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001591 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001592 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001593 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001594 if (var == null && returnExpression != null) {
1595 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1596 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001597 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08001598 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001599 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1600 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001601 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001602 }
1603}
1604
1605UpsertStatement UpsertStatement() throws ParseException:
1606{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001607 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001608 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08001609 VariableExpr var = null;
1610 Query query = null;
1611 Expression returnExpression = null;
1612}
1613{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001614 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07001615 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08001616 ( <RETURNING> returnExpression = Expression())?
1617 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001618 if (var == null && returnExpression != null) {
1619 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1620 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08001621 }
1622 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001623 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
1624 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001625 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001626 }
1627}
1628
1629DeleteStatement DeleteStatement() throws ParseException:
1630{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001631 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001632 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001633 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001634 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001635}
1636{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001637 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001638 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001639 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07001640 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07001641 if (var == null) {
1642 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
1643 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001644 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001645 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07001646 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001647 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07001648 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001649}
1650
1651UpdateStatement UpdateStatement() throws ParseException:
1652{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001653 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001654 VariableExpr vars;
1655 Expression target;
1656 Expression condition;
1657 UpdateClause uc;
1658 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
1659}
1660{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001661 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001662 <WHERE> condition = Expression()
1663 <LEFTPAREN> (uc = UpdateClause()
1664 {
1665 ucs.add(uc);
1666 }
1667 (<COMMA> uc = UpdateClause()
1668 {
1669 ucs.add(uc);
1670 }
1671 )*) <RIGHTPAREN>
1672 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001673 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001674 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001675 }
1676}
1677
1678UpdateClause UpdateClause() throws ParseException:
1679{
1680 Expression target = null;
1681 Expression value = null ;
1682 InsertStatement is = null;
1683 DeleteStatement ds = null;
1684 UpdateStatement us = null;
1685 Expression condition = null;
1686 UpdateClause ifbranch = null;
1687 UpdateClause elsebranch = null;
1688}
1689{
1690 (<SET> target = Expression() <EQ> value = Expression()
1691 | is = InsertStatement()
1692 | ds = DeleteStatement()
1693 | us = UpdateStatement()
1694 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
1695 <THEN> ifbranch = UpdateClause()
1696 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
1697 {
1698 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
1699 }
1700 )
1701}
1702
1703Statement SetStatement() throws ParseException:
1704{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001705 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001706 String pn = null;
1707 String pv = null;
1708}
1709{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001710 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001711 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001712 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001713 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001714 }
1715}
1716
1717Statement WriteStatement() throws ParseException:
1718{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001719 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001720 String nodeName = null;
1721 String fileName = null;
1722 Query query;
1723 String writerClass = null;
1724 Pair<Identifier,Identifier> nameComponents = null;
1725}
1726{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001727 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07001728 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001729 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001730 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001731 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001732 }
1733}
1734
1735LoadStatement LoadStatement() throws ParseException:
1736{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001737 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001738 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001739 Identifier datasetName = null;
1740 boolean alreadySorted = false;
1741 String adapterName;
1742 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001743 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001744}
1745{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001746 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001747 {
1748 dataverseName = nameComponents.first;
1749 datasetName = nameComponents.second;
1750 }
1751 <USING> adapterName = AdapterName() properties = Configuration()
1752 (<PRESORTED>
1753 {
1754 alreadySorted = true;
1755 }
1756 )?
1757 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001758 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
1759 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001760 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001761 }
1762}
1763
1764
1765String AdapterName() throws ParseException :
1766{
1767 String adapterName = null;
1768}
1769{
1770 adapterName = Identifier()
1771 {
1772 return adapterName;
1773 }
1774}
1775
1776Statement CompactStatement() throws ParseException:
1777{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001778 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001779 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001780}
1781{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001782 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001783 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001784 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001785 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001786 }
1787}
1788
Yingyi Buab817482016-08-19 21:29:31 -07001789Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001790{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001791 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001792 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001793}
1794{
1795 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001796 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
1797 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
1798 | <START> { startToken = token; } stmt = StartStatement(startToken)
1799 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07001800 )
1801 {
1802 return stmt;
1803 }
1804}
1805
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001806Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001807{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001808 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001809 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001810}
1811{
1812 <FEED> feedNameComponents = QualifiedName()
1813 {
1814 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001815 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001816 }
1817}
1818
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001819AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001820{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001821 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001822 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001823}
1824{
1825 <FEED> feedNameComponents = QualifiedName()
1826 {
1827 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001828 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001829 }
1830}
1831
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001832AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001833{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001834 Pair<DataverseName,Identifier> feedNameComponents = null;
1835 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07001836
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001837 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07001838}
1839{
1840 (
1841 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001842 {
1843 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
1844 }
1845 )
Yingyi Buab817482016-08-19 21:29:31 -07001846 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001847 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07001848 }
1849}
1850
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001851AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07001852{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001853 Pair<DataverseName,Identifier> feedNameComponents = null;
1854 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07001855
1856 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07001857 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001858 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07001859 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08001860 String whereClauseBody = null;
1861 WhereClause whereClause = null;
1862 Token beginPos = null;
1863 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07001864}
1865{
1866 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001867 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08001868 (ApplyFunction(appliedFunctions))?
1869 (policy = GetPolicy())?
1870 (
1871 <WHERE>
1872 {
1873 beginPos = token;
1874 whereClause = new WhereClause();
1875 Expression whereExpr;
1876 }
1877 whereExpr = Expression()
1878 {
1879 whereClause.setWhereExpr(whereExpr);
1880 }
1881 )?
1882 {
1883 if (whereClause != null) {
1884 endPos = token;
1885 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
1886 }
1887 }
Yingyi Buab817482016-08-19 21:29:31 -07001888 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07001889 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08001890 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07001891 }
1892 )
1893 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001894 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07001895 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001896}
1897
1898Map<String,String> Configuration() throws ParseException :
1899{
1900 Map<String,String> configuration = new LinkedHashMap<String,String>();
1901 Pair<String, String> keyValuePair = null;
1902}
1903{
1904 <LEFTPAREN> ( keyValuePair = KeyValuePair()
1905 {
1906 configuration.put(keyValuePair.first, keyValuePair.second);
1907 }
1908 ( <COMMA> keyValuePair = KeyValuePair()
1909 {
1910 configuration.put(keyValuePair.first, keyValuePair.second);
1911 }
1912 )* )? <RIGHTPAREN>
1913 {
1914 return configuration;
1915 }
1916}
1917
1918Pair<String, String> KeyValuePair() throws ParseException:
1919{
1920 String key;
1921 String value;
1922}
1923{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001924 <LEFTPAREN> key = ConstantString() <EQ> value = ConstantString() <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001925 {
1926 return new Pair<String, String>(key, value);
1927 }
1928}
1929
1930Map<String,String> Properties() throws ParseException:
1931{
1932 Map<String,String> properties = new HashMap<String,String>();
1933 Pair<String, String> property;
1934}
1935{
1936 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
1937 {
1938 properties.put(property.first, property.second);
1939 }
1940 ( <COMMA> property = Property()
1941 {
1942 properties.put(property.first, property.second);
1943 }
1944 )* <RIGHTPAREN> )?
1945 {
1946 return properties;
1947 }
1948}
1949
1950Pair<String, String> Property() throws ParseException:
1951{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001952 String key = null;
1953 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001954}
1955{
Yingyi Bu6d57e492016-06-06 21:24:42 -07001956 (key = Identifier() | key = StringLiteral())
1957 <EQ>
1958 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001959 {
1960 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001961 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07001962 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001963 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001964 }
1965 }
1966 )
1967 {
1968 return new Pair<String, String>(key.toUpperCase(), value);
1969 }
1970}
1971
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001972IndexedTypeExpression IndexedTypeExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001973{
1974 TypeExpression typeExpr = null;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001975 boolean isUnknownable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001976}
1977{
1978 (
1979 typeExpr = TypeReference()
1980 | typeExpr = OrderedListTypeDef()
1981 | typeExpr = UnorderedListTypeDef()
1982 )
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001983 ( <QUES> { isUnknownable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001984 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07001985 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001986 }
1987}
1988
1989TypeExpression TypeExpr() throws ParseException:
1990{
1991 TypeExpression typeExpr = null;
1992}
1993{
1994 (
1995 typeExpr = RecordTypeDef()
1996 | typeExpr = TypeReference()
1997 | typeExpr = OrderedListTypeDef()
1998 | typeExpr = UnorderedListTypeDef()
1999 )
2000 {
2001 return typeExpr;
2002 }
2003}
2004
2005RecordTypeDefinition RecordTypeDef() throws ParseException:
2006{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002007 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002008 RecordTypeDefinition recType = new RecordTypeDefinition();
2009 RecordTypeDefinition.RecordKind recordKind = null;
2010}
2011{
2012 ( <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
2013 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; } )?
2014 <LEFTBRACE>
2015 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002016 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002017 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
2018 if (hintToken != null) {
2019 String hintParams = hintToken.hintParams;
2020 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
2021 if (splits == null || splits.length != 4) {
2022 throw new SqlppParseException(getSourceLocation(hintToken),
2023 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002024 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002025 if (!splits[0].equals("int")) {
2026 throw new SqlppParseException(getSourceLocation(hintToken),
2027 "The only supported type for gen-fields is int.");
2028 }
2029 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
2030 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
2031 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002032 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002033 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002034 (
2035 RecordField(recType)
2036 ( <COMMA> RecordField(recType) )*
2037 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002038 <RIGHTBRACE>
2039 {
2040 if (recordKind == null) {
2041 recordKind = RecordTypeDefinition.RecordKind.OPEN;
2042 }
2043 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002044 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002045 }
2046}
2047
2048void RecordField(RecordTypeDefinition recType) throws ParseException:
2049{
2050 String fieldName;
2051 TypeExpression type = null;
2052 boolean nullable = false;
2053}
2054{
2055 fieldName = Identifier()
2056 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002057 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
2058 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
2059 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
2060 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002061 }
2062 <COLON> type = TypeExpr() (<QUES> { nullable = true; } )?
2063 {
2064 recType.addField(fieldName, type, nullable, rfdg);
2065 }
2066}
2067
2068TypeReferenceExpression TypeReference() throws ParseException:
2069{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002070 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002071}
2072{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002073 id = QualifiedName()
2074 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07002075 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
2076 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002077 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002078
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002079 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002080 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002081 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002082}
2083
2084OrderedListTypeDefinition OrderedListTypeDef() throws ParseException:
2085{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002086 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002087 TypeExpression type = null;
2088}
2089{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002090 <LEFTBRACKET> { startToken = token; }
2091 ( type = TypeExpr() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002092 <RIGHTBRACKET>
2093 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002094 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002095 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002096 }
2097}
2098
Yingyi Bu391f09e2015-10-29 13:49:39 -07002099UnorderedListTypeDefinition UnorderedListTypeDef() throws ParseException:
2100{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002101 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002102 TypeExpression type = null;
2103}
2104{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002105 <LEFTDBLBRACE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002106 ( type = TypeExpr() )
2107 <RIGHTDBLBRACE>
2108 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002109 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002110 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002111 }
2112}
2113
2114FunctionName FunctionName() throws ParseException:
2115{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002116 Triple<List<String>, SourceLocation, SqlppHint> prefix = null;
2117 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002118}
2119{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002120 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
2121 // that copy must be kept in sync with this code
2122 prefix = MultipartIdentifierWithHints(SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
2123 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT)
2124 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002125 {
2126 FunctionName result = new FunctionName();
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002127 result.sourceLoc = prefix.second;
2128 result.hint = prefix.third;
2129 List<String> list = prefix.first;
2130 int ln = list.size();
2131 String last = list.get(ln - 1);
2132 if (suffix == null) {
2133 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
2134 // no library name
2135 result.function = last;
2136 } else {
2137 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
2138 // suffix = func_name
2139 result.library = last;
2140 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002141 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002142 if (ln > 1) {
2143 result.dataverse = DataverseName.create(list, 0, ln - 1);
2144 } else {
2145 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002146 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002147
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002148 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
2149 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002150 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002151 return result;
2152 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002153}
2154
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002155Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002156{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002157 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002158}
2159{
2160 name = QualifiedName()
2161 {
2162 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002163 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002164 }
2165 return name;
2166 }
2167}
2168
2169String Identifier() throws ParseException:
2170{
2171 String lit = null;
2172}
2173{
2174 (<IDENTIFIER>
2175 {
2176 return token.image;
2177 }
2178 | lit = QuotedString()
2179 {
2180 return lit;
2181 }
2182 )
2183}
2184
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002185Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002186{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002187 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002188 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002189}
2190{
2191 fieldList = NestedField()
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002192 ( <COLON> fieldType = IndexedTypeExpr() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002193 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002194 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
2195 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002196 }
2197}
2198
Yingyi Buc9bfe252016-03-01 00:02:40 -08002199Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002200{
2201 List<String> exprList = new ArrayList<String>();
2202 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002203 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002204 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002205}
2206{
2207 lit = Identifier()
2208 {
Yingyi Bub9169b62016-02-26 21:21:49 -08002209 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002210 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08002211 }
2212 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08002213 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08002214 <LEFTPAREN><RIGHTPAREN>
2215 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002216 if(!lit.equalsIgnoreCase("meta")){
2217 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08002218 }
2219 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002220 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08002221 }
2222 )?
2223 {
2224 if(!meetParens){
2225 exprList.add(lit);
2226 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002227 }
2228 (<DOT>
2229 lit = Identifier()
2230 {
2231 exprList.add(lit);
2232 }
2233 )*
2234 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002235 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002236 }
2237}
2238
Yingyi Bu6d57e492016-06-06 21:24:42 -07002239String ConstantString() throws ParseException:
2240{
2241 String value = null;
2242}
2243{
2244 (value = QuotedString() | value = StringLiteral())
2245 {
2246 return value;
2247 }
2248}
2249
Yingyi Bu391f09e2015-10-29 13:49:39 -07002250String QuotedString() throws ParseException:
2251{
2252}
2253{
2254 <QUOTED_STRING>
2255 {
2256 return removeQuotesAndEscapes(token.image);
2257 }
2258}
2259
Yingyi Bu391f09e2015-10-29 13:49:39 -07002260String StringLiteral() throws ParseException:
2261{
2262}
2263{
2264 <STRING_LITERAL>
2265 {
2266 return removeQuotesAndEscapes(token.image);
2267 }
2268}
2269
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002270List<String> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002271{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002272 Triple<List<String>, SourceLocation, SqlppHint> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002273}
2274{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002275 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002276 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002277 return result.first;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002278 }
2279}
2280
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002281Triple<List<String>, SourceLocation, SqlppHint> MultipartIdentifierWithHints(SqlppHint... expectedHints)
2282 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002283{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002284 List<String> list = new ArrayList<String>();
2285 SourceLocation sourceLoc = null;
2286 SqlppHint hint = null;
2287 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002288}
2289{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002290 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002291 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002292 list.add(item);
2293 sourceLoc = getSourceLocation(token);
2294 if (expectedHints != null && expectedHints.length > 0) {
2295 Token hintToken = fetchHint(token, expectedHints);
2296 if (hintToken != null) {
2297 hint = hintToken.hint;
2298 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002299 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002300 }
2301 (<DOT> item = Identifier() { list.add(item); } )*
2302 {
2303 return new Triple<List<String>, SourceLocation, SqlppHint>(list, sourceLoc, hint);
2304 }
2305}
2306
2307Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
2308{
2309 List<String> list = null;
2310}
2311{
2312 list = MultipartIdentifier()
2313 {
2314 int len = list.size();
2315 DataverseName id1 = len > 1 ? DataverseName.create(list, 0, len - 1) : null;
2316 Identifier id2 = new Identifier(list.get(len - 1));
2317 return new Pair<DataverseName,Identifier>(id1, id2);
2318 }
2319}
2320
2321Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
2322{
2323 List<String> list = new ArrayList<String>();
2324 String item = null;
2325}
2326{
2327 item = Identifier() { list.add(item); }
2328 (<DOT> item = Identifier() { list.add(item); } )+
2329 {
2330 int len = list.size();
2331 DataverseName id1 = len > 2 ? DataverseName.create(list, 0, len - 2) : null;
2332 Identifier id2 = new Identifier(list.get(len - 2));
2333 Identifier id3 = new Identifier(list.get(len - 1));
2334 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002335 }
2336}
2337
2338FunctionDecl FunctionDeclaration() throws ParseException:
2339{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002340 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002341 String functionName;
Ian Maxon38fe9402020-01-29 19:27:40 -08002342 List<Pair<VarIdentifier,IndexedTypeExpression>> paramList = new ArrayList<Pair<VarIdentifier,IndexedTypeExpression>>();
2343 List<VarIdentifier> paramVarOnly = new ArrayList<VarIdentifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002344 Expression funcBody;
2345 createNewScope();
2346}
2347{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002348 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07002349 functionName = Identifier()
Ian Maxon38fe9402020-01-29 19:27:40 -08002350 paramList = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07002351 <LEFTBRACE>
2352 (funcBody = SelectExpression(true) | funcBody = Expression())
2353 <RIGHTBRACE>
2354 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002355 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, paramList.size());
Xikui Wang3de700a2018-03-15 16:32:55 -07002356 getCurrentScope().addFunctionDescriptor(signature, false);
Ian Maxon38fe9402020-01-29 19:27:40 -08002357 for(Pair<VarIdentifier,IndexedTypeExpression> p: paramList){
2358 paramVarOnly.add(p.getFirst());
2359 }
2360 FunctionDecl stmt = new FunctionDecl(signature, paramVarOnly, funcBody);
Xikui Wang3de700a2018-03-15 16:32:55 -07002361 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002362 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07002363 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002364}
2365
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002366Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002367{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002368 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002369 Expression expr;
2370}
2371{
2372 (
2373 expr = Expression()
2374 |
2375 expr = SelectExpression(false)
2376 )
2377 {
2378 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002379 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07002380 return query;
2381 }
2382}
2383
2384
Yingyi Bu391f09e2015-10-29 13:49:39 -07002385Expression Expression():
2386{
2387 Expression expr = null;
2388 Expression exprP = null;
2389}
2390{
2391(
2392 LOOKAHEAD(2)
2393 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002394 | expr = QuantifiedExpression()
2395)
2396 {
2397 return (exprP==null) ? expr : exprP;
2398 }
2399}
2400
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002401Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002402{
2403 OperatorExpr op = null;
2404 Expression operand = null;
2405}
2406{
2407 operand = AndExpr()
2408 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002409 <OR>
2410 {
2411 if (op == null) {
2412 op = new OperatorExpr();
2413 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07002414 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002415 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002416 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002417 try{
2418 op.addOperator(token.image.toLowerCase());
2419 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002420 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002421 }
Xikui Wang3de700a2018-03-15 16:32:55 -07002422 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002423
Xikui Wang3de700a2018-03-15 16:32:55 -07002424 operand = AndExpr()
2425 {
2426 op.addOperand(operand);
2427 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002428
2429 )*
2430
2431 {
2432 return op==null? operand: op;
2433 }
2434}
2435
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002436Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002437{
2438 OperatorExpr op = null;
2439 Expression operand = null;
2440}
2441{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002442 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002443 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07002444 <AND>
2445 {
2446 if (op == null) {
2447 op = new OperatorExpr();
2448 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002449 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002450 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002451 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002452 try{
2453 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002454 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002455 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002456 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002457 }
2458
Yingyi Bu196db5d2016-07-15 19:07:20 -07002459 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002460 {
2461 op.addOperand(operand);
2462 }
2463
2464 )*
2465
2466 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002467 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002468 }
2469}
2470
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002471Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07002472{
2473 Expression inputExpr;
2474 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002475 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07002476}
2477{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07002478 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07002479 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002480 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002481 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002482 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002483 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07002484 } else {
2485 return inputExpr;
2486 }
2487 }
2488}
Yingyi Bu391f09e2015-10-29 13:49:39 -07002489
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002490Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002491{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002492 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002493 OperatorExpr op = null;
2494 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002495 IExpressionAnnotation annotation = null;
2496}
2497{
Yingyi Bu6c638342016-09-02 17:54:34 -07002498 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002499
2500 (
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002501 LOOKAHEAD(2)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN>)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002502 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002503 Token hintToken = fetchHint(token, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
Shiva2ea73232019-10-09 18:04:05 -07002504 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.HASH_BROADCAST_JOIN_HINT);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002505 if (hintToken != null) {
2506 switch (hintToken.hint) {
2507 case INDEXED_NESTED_LOOP_JOIN_HINT:
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002508 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002509 break;
2510 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002511 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002512 break;
Shiva2ea73232019-10-09 18:04:05 -07002513 case HASH_BROADCAST_JOIN_HINT:
2514 annotation = new BroadcastExpressionAnnotation();
2515 annotation.setObject(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002516 break;
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002517 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002518 }
Yingyi Buea4ec722016-11-04 01:26:16 -07002519
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002520 String operator = token.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07002521 if (operator.equals("<>")){
2522 operator = "!=";
2523 }
2524 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002525 operator = "not_" + operator;
2526 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002527 if (op == null) {
2528 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07002529 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002530 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002531 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002532 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07002533 try{
2534 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002535 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002536 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07002537 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07002538 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002539
Yingyi Bu6c638342016-09-02 17:54:34 -07002540 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07002541 {
Shiva2ea73232019-10-09 18:04:05 -07002542 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07002543 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002544 )?
2545
2546 {
2547 if (annotation != null) {
2548 op.addHint(annotation);
2549 }
2550 return op==null? operand: op;
2551 }
2552}
2553
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002554Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07002555{
2556 boolean not = false;
2557 OperatorExpr op = null;
2558 Expression operand = null;
2559 IExpressionAnnotation annotation = null;
2560}
2561{
2562 operand = IsExpr()
2563 (
2564 LOOKAHEAD(2)
2565 (<NOT> { not = true; })? <BETWEEN>
2566 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002567 Token hintToken = fetchHint(token, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
2568 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT);
2569 if (hintToken != null) {
2570 switch (hintToken.hint) {
2571 case INDEXED_NESTED_LOOP_JOIN_HINT:
Yingyi Bu6c638342016-09-02 17:54:34 -07002572 annotation = IndexedNLJoinExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002573 break;
2574 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
Yingyi Bu6c638342016-09-02 17:54:34 -07002575 annotation = SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002576 break;
Yingyi Bu6c638342016-09-02 17:54:34 -07002577 }
2578 }
2579 String operator = token.image.toLowerCase();
2580 if(not){
2581 operator = "not_" + operator;
2582 }
2583 if (op == null) {
2584 op = new OperatorExpr();
2585 op.addOperand(operand);
2586 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002587 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07002588 }
2589 try{
2590 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002591 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002592 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07002593 }
2594 }
2595
2596 operand = IsExpr()
2597 {
2598 op.addOperand(operand);
2599 }
2600
2601 <AND>
2602 operand = IsExpr()
2603 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08002604 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07002605 op.addOperand(operand);
2606 }
2607 )?
2608
2609 {
2610 if (annotation != null) {
2611 op.addHint(annotation);
2612 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002613 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07002614 }
2615}
2616
Yingyi Budaa549c2016-06-28 22:30:52 -07002617Expression IsExpr() throws ParseException:
2618{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002619 Token notToken = null;
2620 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002621 Expression operand = null;
2622 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002623 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07002624}
2625{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002626 operand = LikeExpr()
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002627 ( <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002628 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002629 (
2630 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
2631 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08002632 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07002633 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002634 )
Yingyi Budaa549c2016-06-28 22:30:52 -07002635 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002636 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07002637 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002638 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002639 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08002640 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07002641 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002642 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07002643 }
2644 }
2645 )?
2646 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002647 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07002648 }
2649}
2650
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002651Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002652{
2653 boolean not = false;
2654 OperatorExpr op = null;
2655 Expression operand = null;
2656}
2657{
2658 operand = ConcatExpr()
2659 (
2660 LOOKAHEAD(2)
2661 (<NOT> { not = true; })? <LIKE>
2662 {
2663 op = new OperatorExpr();
2664 op.addOperand(operand);
2665 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002666 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002667
2668 String operator = token.image.toLowerCase();
2669 if (not) {
2670 operator = "not_" + operator;
2671 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002672 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002673 op.addOperator(operator);
2674 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002675 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08002676 }
2677 }
2678
2679 operand = ConcatExpr()
2680 {
2681 op.addOperand(operand);
2682 }
2683 )?
2684
2685 {
2686 return op == null ? operand : op;
2687 }
2688}
2689
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002690Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002691{
2692 OperatorExpr op = null;
2693 Expression operand = null;
2694}
2695{
2696 operand = AddExpr()
2697 (
2698 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002699 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002700 {
2701 if (op == null) {
2702 op = new OperatorExpr();
2703 op.addOperand(operand);
2704 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002705 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002706 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002707 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002708 }
2709 operand = AddExpr()
2710 {
2711 op.addOperand(operand);
2712 }
2713 )*
2714
2715 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002716 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002717 }
2718}
Yingyi Budaa549c2016-06-28 22:30:52 -07002719
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002720Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002721{
2722 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002723 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002724 Expression operand = null;
2725}
2726{
2727 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07002728 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07002729 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002730 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002731 {
2732 if (op == null) {
2733 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002734 op.addOperand(operand);
2735 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002736 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002737 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002738 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002739 }
2740
2741 operand = MultExpr()
2742 {
2743 op.addOperand(operand);
2744 }
2745 )*
2746
2747 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002748 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002749 }
2750}
2751
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002752Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002753{
2754 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002755 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002756 Expression operand = null;
2757}
2758{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002759 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002760
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002761 ( (
2762 <MUL> { opType = OperatorType.MUL; } |
2763 <DIVIDE> { opType = OperatorType.DIVIDE; } |
2764 <DIV> { opType = OperatorType.DIV; } |
2765 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
2766 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002767 {
2768 if (op == null) {
2769 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002770 op.addOperand(operand);
2771 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002772 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002773 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07002774 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002775 }
2776 operand = ExponentExpr()
2777 {
2778 op.addOperand(operand);
2779 }
2780 )*
2781
2782 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002783 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002784 }
2785}
2786
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002787Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002788{
2789 OperatorExpr op = null;
2790 Expression operand = null;
2791}
2792{
2793 operand = UnaryExpr()
2794 (<CARET>
2795 {
2796 if (op == null) {
2797 op = new OperatorExpr();
2798 op.addOperand(operand);
2799 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002800 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002801 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002802 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002803 }
2804 operand = UnaryExpr()
2805 {
2806 op.addOperand(operand);
2807 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002808 )?
2809 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002810 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07002811 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002812}
2813
2814Expression UnaryExpr() throws ParseException:
2815{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002816 boolean not = false;
2817 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002818 Expression expr = null;
2819}
Yingyi Budaa549c2016-06-28 22:30:52 -07002820{
Yingyi Bu196db5d2016-07-15 19:07:20 -07002821 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002822 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002823 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002824 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002825 exprType = "not_" + exprType;
2826 }
2827 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002828 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002829 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07002830 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08002831 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002832 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07002833 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002834 }
2835 )?
2836
2837 expr = ValueExpr()
2838 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002839 if (uexpr == null) {
2840 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002841 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002842 uexpr.setExpr(expr);
2843 return uexpr;
2844 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002845 }
2846}
2847
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002848Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002849{
2850 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002851 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002852}
2853{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002854 expr = PrimaryExpr()
2855 (
2856 accessor = FieldAccessor(accessor != null ? accessor : expr)
2857 |
2858 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002859 )*
2860 {
2861 return accessor == null ? expr : accessor;
2862 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002863}
2864
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002865FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002866{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002867 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002868 String ident = null;
2869}
2870{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002871 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002872 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002873 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002874 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002875 }
2876}
2877
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002878AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002879{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002880 Token startToken = null;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002881 boolean isListSliceExpression = false;
2882 AbstractAccessor resultExpression = null;
2883 Expression argumentExpression1 = null;
2884 Expression argumentExpression2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002885}
2886{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002887 <LEFTBRACKET> { startToken = token; }
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002888 ( argumentExpression1 = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002889 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002890 if (argumentExpression1.getKind() == Expression.Kind.LITERAL_EXPRESSION)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002891 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002892 Literal lit = ((LiteralExpr)argumentExpression1).getValue();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002893 if (lit.getLiteralType() != Literal.Type.INTEGER &&
2894 lit.getLiteralType() != Literal.Type.LONG) {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002895 throw new SqlppParseException(argumentExpression1.getSourceLocation(), "Index should be an INTEGER");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002896 }
2897 }
2898 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002899 )
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002900 (<COLON>
2901 {
2902 isListSliceExpression = true;
2903 }
2904 ( argumentExpression2 = Expression()
2905 {
2906 if (argumentExpression2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
2907 Literal lit = ((LiteralExpr)argumentExpression2).getValue();
2908 if (lit.getLiteralType() != Literal.Type.INTEGER &&
2909 lit.getLiteralType() != Literal.Type.LONG) {
2910 throw new SqlppParseException(argumentExpression2.getSourceLocation(), "Index should be an INTEGER");
2911 }
2912 }
2913 })?
2914 )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002915 <RIGHTBRACKET>
2916 {
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03002917 if (!isListSliceExpression) {
2918 resultExpression = new IndexAccessor(inputExpr, argumentExpression1);
2919 } else {
2920 resultExpression = new ListSliceExpression(inputExpr, argumentExpression1, argumentExpression2);
2921 }
2922
2923 return addSourceLocation(resultExpression, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002924 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002925}
2926
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002927Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002928{
2929 Expression expr = null;
2930}
2931{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002932 (
2933 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07002934 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002935 | expr = Literal()
2936 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07002937 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002938 | expr = ListConstructor()
2939 | expr = RecordConstructor()
2940 | expr = ParenthesizedExpression()
2941 )
2942 {
2943 return expr;
2944 }
2945}
2946
2947Expression Literal() throws ParseException:
2948{
2949 LiteralExpr lit = new LiteralExpr();
2950 String str = null;
2951}
2952{
2953 ( str = StringLiteral()
2954 {
2955 lit.setValue(new StringLiteral(str));
2956 }
2957 | <INTEGER_LITERAL>
2958 {
Till Westmann68c6a992016-09-30 00:19:12 -07002959 try {
2960 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
2961 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002962 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002963 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002964 }
2965 | <FLOAT_LITERAL>
2966 {
Till Westmann68c6a992016-09-30 00:19:12 -07002967 try {
2968 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
2969 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002970 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002971 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002972 }
2973 | <DOUBLE_LITERAL>
2974 {
Till Westmann68c6a992016-09-30 00:19:12 -07002975 try {
2976 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
2977 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08002978 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07002979 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002980 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07002981 | <MISSING>
2982 {
2983 lit.setValue(MissingLiteral.INSTANCE);
2984 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002985 | <NULL>
2986 {
2987 lit.setValue(NullLiteral.INSTANCE);
2988 }
2989 | <TRUE>
2990 {
2991 lit.setValue(TrueLiteral.INSTANCE);
2992 }
2993 | <FALSE>
2994 {
2995 lit.setValue(FalseLiteral.INSTANCE);
2996 }
2997 )
2998 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002999 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003000 }
3001}
3002
Yingyi Bu391f09e2015-10-29 13:49:39 -07003003VariableExpr VariableRef() throws ParseException:
3004{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003005 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003006}
3007{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003008 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3009 {
Yingyi Buacc12a92016-03-26 17:25:05 -07003010 id = SqlppVariableUtil.toInternalVariableName(id); // Prefix user-defined variables with "$"
Yingyi Bu391f09e2015-10-29 13:49:39 -07003011 Identifier ident = lookupSymbol(id);
3012 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003013 throw new SqlppParseException(getSourceLocation(token),
3014 "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 -07003015 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003016 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003017 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003018 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003019 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003020 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07003021 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003022 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003023 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003024 }
3025}
3026
Yingyi Bu391f09e2015-10-29 13:49:39 -07003027VariableExpr Variable() throws ParseException:
3028{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003029 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003030}
3031{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003032 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3033 {
Yingyi Buacc12a92016-03-26 17:25:05 -07003034 id = SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
Yingyi Bu391f09e2015-10-29 13:49:39 -07003035 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003036 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
3037 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07003038 varExp.setIsNewVar(false);
3039 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003040 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003041 }
3042}
3043
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003044Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
3045{
3046 VariableExpr var = null;
3047 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
3048}
3049{
3050 var = Variable()
3051 ( LOOKAHEAD(1)
3052 {
3053 VariableExpr fieldVarExpr = null;
3054 String fieldIdentifierStr = null;
3055 }
3056 <LEFTPAREN>
3057 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3058 {
3059 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3060 }
3061 (<COMMA>
3062 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3063 {
3064 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3065 }
3066 )*
3067 <RIGHTPAREN>
3068 )?
3069 {
3070 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
3071 }
3072}
3073
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003074VariableExpr ExternalVariableRef() throws ParseException:
3075{
3076 String name = null;
3077}
3078{
3079 (
3080 (
3081 <DOLLAR>
3082 (
3083 <INTEGER_LITERAL> { name = token.image; } |
3084 <IDENTIFIER> { name = token.image; } |
3085 name = QuotedString()
3086 )
3087 )
3088 |
3089 (
3090 <QUES> { name = String.valueOf(++externalVarCounter); }
3091 )
3092 )
3093 {
3094 String idName = SqlppVariableUtil.toExternalVariableName(name);
3095 VarIdentifier id = new VarIdentifier(idName);
3096 VariableExpr varExp = new VariableExpr(id);
3097 return addSourceLocation(varExp, token);
3098 }
3099}
3100
Yingyi Bu391f09e2015-10-29 13:49:39 -07003101Expression ListConstructor() throws ParseException:
3102{
3103 Expression expr = null;
3104}
3105{
3106 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003107 expr = OrderedListConstructor() |
3108 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003109 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003110 {
3111 return expr;
3112 }
3113}
3114
Yingyi Bu391f09e2015-10-29 13:49:39 -07003115ListConstructor OrderedListConstructor() throws ParseException:
3116{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003117 Token startToken = null;
3118 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003119}
3120{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003121 <LEFTBRACKET> { startToken = token; }
3122 exprList = ExpressionList()
3123 <RIGHTBRACKET>
3124 {
3125 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003126 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003127 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003128}
3129
3130ListConstructor UnorderedListConstructor() throws ParseException:
3131{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003132 Token startToken = null;
3133 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003134}
3135{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003136 <LEFTDBLBRACE> { startToken = token; }
3137 exprList = ExpressionList()
3138 <RIGHTDBLBRACE>
3139 {
3140 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003141 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003142 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003143}
3144
3145List<Expression> ExpressionList() throws ParseException:
3146{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003147 Expression expr = null;
3148 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003149}
3150{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003151 (
3152 expr = Expression()
3153 {
3154 exprList.add(expr);
3155 }
3156 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07003157 {
3158 exprList.add(expr);
3159 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003160 )*
3161 )?
3162 {
3163 return exprList;
3164 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003165}
3166
Yingyi Bu391f09e2015-10-29 13:49:39 -07003167RecordConstructor RecordConstructor() throws ParseException:
3168{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003169 Token startToken = null;
3170 FieldBinding fb = null;
3171 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003172}
3173{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003174 <LEFTBRACE> { startToken = token; }
3175 (
3176 fb = FieldBinding() { fbList.add(fb); }
3177 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
3178 )?
3179 <RIGHTBRACE>
3180 {
3181 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003182 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003183 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003184}
3185
3186FieldBinding FieldBinding() throws ParseException:
3187{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003188 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003189}
3190{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003191 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003192 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003193 if (right == null) {
3194 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
3195 if (generatedIdentifier == null) {
3196 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
3197 }
3198 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07003199 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
3200 generatedNameExpr.setSourceLocation(left.getSourceLocation());
3201 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003202 } else {
3203 return new FieldBinding(left, right);
3204 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003205 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003206}
3207
Yingyi Bu391f09e2015-10-29 13:49:39 -07003208Expression FunctionCallExpr() throws ParseException:
3209{
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003210 Expression resultExpr;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003211 CallExpr callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003212 List<Expression> argList = new ArrayList<Expression>();
Yingyi Buf4d09842016-08-26 00:03:52 -07003213 Expression tmp = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003214 int arity = 0;
3215 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07003216 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003217 boolean distinct = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003218}
3219{
3220 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003221 <LEFTPAREN> (
3222 ( <DISTINCT> { distinct = true; } )?
3223 ( tmp = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003224 {
Yingyi Buf4d09842016-08-26 00:03:52 -07003225 if(star){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003226 if(!funcName.function.equalsIgnoreCase("count")){
3227 throw new SqlppParseException(getSourceLocation(token), "The parameter * can only be used in COUNT().");
Yingyi Buf4d09842016-08-26 00:03:52 -07003228 }
3229 argList.add(new LiteralExpr(new LongIntegerLiteral(1L)));
3230 } else {
3231 argList.add(tmp);
3232 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003233 arity ++;
3234 }
3235 (<COMMA> tmp = Expression()
3236 {
3237 argList.add(tmp);
3238 arity++;
3239 }
3240 )*)? <RIGHTPAREN>
3241 {
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003242 String name = funcName.function;
3243 if (distinct) {
Dmitry Lychagine5ea42d2019-03-15 11:38:09 -07003244 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003245 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003246 // TODO use funcName.library
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003247 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003248 FunctionSignature signature
3249 = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
3250 if (signature == null) {
3251 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
3252 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003253 callExpr = FunctionMapUtil.normalizedListInputFunctions(new CallExpr(signature,argList));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003254 if (funcName.hint != null) {
3255 switch (funcName.hint) {
3256 case INDEXED_NESTED_LOOP_JOIN_HINT:
3257 callExpr.addHint(IndexedNLJoinExpressionAnnotation.INSTANCE);
3258 break;
3259 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
3260 callExpr.addHint(SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE);
3261 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003262 }
3263 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003264 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003265 resultExpr = callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003266 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003267
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003268 ( LOOKAHEAD(5) resultExpr = WindowExpr(callExpr.getFunctionSignature(), callExpr.getExprList()) )?
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07003269
3270 {
3271 return resultExpr;
3272 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003273}
3274
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003275WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList) throws ParseException:
3276{
3277 Boolean fromLast = null, ignoreNulls = null;
3278}
3279{
3280 (
3281 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
3282 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
3283 <FROM> <IDENTIFIER>
3284 {
3285 if (isToken(FIRST)) {
3286 fromLast = false;
3287 } else if (isToken(LAST)) {
3288 fromLast = true;
3289 } else {
3290 throw createUnexpectedTokenError();
3291 }
3292 }
3293 )?
3294 (
3295 // ( RESPECT | IGNORE ) NULLS OVER
3296 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
3297 <IDENTIFIER>
3298 {
3299 if (isToken(RESPECT)) {
3300 ignoreNulls = false;
3301 } else if (isToken(IGNORE)) {
3302 ignoreNulls = true;
3303 } else {
3304 throw createUnexpectedTokenError();
3305 }
3306 }
3307 <IDENTIFIER>
3308 {
3309 if (!isToken(NULLS)) {
3310 throw createUnexpectedTokenError();
3311 }
3312 }
3313 )?
3314 <OVER>
3315 {
3316 return OverClause(signature, argList, token, fromLast, ignoreNulls);
3317 }
3318}
3319
3320WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Token startToken, Boolean fromLast,
3321 Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003322{
3323 Expression partitionExpr = null;
3324 List<Expression> partitionExprs = new ArrayList<Expression>();
3325 OrderbyClause orderByClause = null;
3326 List<Expression> orderbyList = null;
3327 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
3328 WindowExpression.FrameMode frameMode = null;
3329 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
3330 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
3331 Expression frameStartExpr = null, frameEndExpr = null;
3332 WindowExpression.FrameExclusionKind frameExclusionKind = null;
3333 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
3334 VariableExpr windowVar = null;
3335 List<Pair<Expression, Identifier>> windowFieldList = null;
3336}
3337{
3338 (
3339 windowVarWithFieldList = VariableWithFieldMap() <AS>
3340 {
3341 windowVar = windowVarWithFieldList.first;
3342 windowFieldList = windowVarWithFieldList.second;
3343 }
3344 )?
3345 <LEFTPAREN>
3346 (
3347 <IDENTIFIER> { expectToken(PARTITION); } <BY>
3348 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
3349 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
3350 )?
3351 (
3352 orderByClause = OrderbyClause()
3353 {
3354 orderbyList = orderByClause.getOrderbyList();
3355 orderbyModifierList = orderByClause.getModifierList();
3356 }
3357 (
3358 frameMode = WindowFrameMode()
3359 (
3360 frameStart = WindowFrameBoundary() |
3361 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
3362 )
3363 ( frameExclusionKind = WindowFrameExclusion() )?
3364 {
3365 frameStartKind = frameStart.first;
3366 frameStartExpr = frameStart.second;
3367 if (frameEnd == null) {
3368 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3369 } else {
3370 frameEndKind = frameEnd.first;
3371 frameEndExpr = frameEnd.second;
3372 }
3373 if (frameExclusionKind == null) {
3374 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
3375 }
3376 }
3377 )?
3378 )?
3379 <RIGHTPAREN>
3380 {
3381 WindowExpression winExp = new WindowExpression(signature, argList, partitionExprs, orderbyList, orderbyModifierList,
3382 frameMode, frameStartKind, frameStartExpr, frameEndKind, frameEndExpr, frameExclusionKind, windowVar,
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003383 windowFieldList, ignoreNulls, fromLast);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003384 return addSourceLocation(winExp, startToken);
3385 }
3386}
3387
3388WindowExpression.FrameMode WindowFrameMode() throws ParseException:
3389{
3390}
3391{
3392 <IDENTIFIER>
3393 {
3394 if (isToken(RANGE)) {
3395 return WindowExpression.FrameMode.RANGE;
3396 } else if (isToken(ROWS)) {
3397 return WindowExpression.FrameMode.ROWS;
3398 } else if (isToken(GROUPS)) {
3399 return WindowExpression.FrameMode.GROUPS;
3400 } else {
3401 throw createUnexpectedTokenError();
3402 }
3403 }
3404}
3405
3406Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
3407{
3408 boolean current = false;
3409 Expression expr = null;
3410}
3411{
3412 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003413 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003414 | expr = Expression()
3415 )
3416 <IDENTIFIER>
3417 {
3418 WindowExpression.FrameBoundaryKind kind;
3419 if (current && isToken(ROW)) {
3420 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
3421 } else if (!current && isToken(PRECEDING)) {
3422 kind = expr == null
3423 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
3424 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
3425 } else if (!current && isToken(FOLLOWING)) {
3426 kind = expr == null
3427 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
3428 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
3429 } else {
3430 throw createUnexpectedTokenError();
3431 }
3432 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
3433 }
3434}
3435
3436WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
3437{
3438 boolean current = false, no = false;
3439}
3440{
3441 <IDENTIFIER>
3442 {
3443 expectToken(EXCLUDE);
3444 }
3445 (
3446 <GROUP>
3447 {
3448 return WindowExpression.FrameExclusionKind.GROUP;
3449 }
3450 |
3451 (
3452 <IDENTIFIER>
3453 {
3454 if (isToken(TIES)) {
3455 return WindowExpression.FrameExclusionKind.TIES;
3456 } else if (isToken(CURRENT)) {
3457 current = true;
3458 } else if (isToken(NO)) {
3459 no = true;
3460 } else {
3461 throw createUnexpectedTokenError();
3462 }
3463 }
3464 <IDENTIFIER>
3465 {
3466 if (current && isToken(ROW)) {
3467 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
3468 } else if (no && isToken(OTHERS)) {
3469 return WindowExpression.FrameExclusionKind.NO_OTHERS;
3470 } else {
3471 throw createUnexpectedTokenError();
3472 }
3473 }
3474 )
3475 )
3476}
3477
Yingyi Bu391f09e2015-10-29 13:49:39 -07003478Expression ParenthesizedExpression() throws ParseException:
3479{
3480 Expression expr;
3481}
3482{
3483 (
3484 LOOKAHEAD(2)
3485 <LEFTPAREN> expr = Expression() <RIGHTPAREN>
3486 |
3487 expr = Subquery()
3488 )
3489 {
3490 return expr;
3491 }
3492}
3493
Yingyi Buc8c067c2016-07-25 23:37:19 -07003494Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003495{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003496 Token startToken = null;
3497 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003498 List<Expression> whenExprs = new ArrayList<Expression>();
3499 List<Expression> thenExprs = new ArrayList<Expression>();
3500 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07003501 Expression whenExpr = null;
3502 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003503}
3504{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003505 <CASE> { startToken = token; }
3506 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07003507 (
3508 <WHEN> whenExpr = Expression()
3509 {
3510 whenExprs.add(whenExpr);
3511 }
3512 <THEN> thenExpr = Expression()
3513 {
3514 thenExprs.add(thenExpr);
3515 }
3516 )*
3517 (<ELSE> elseExpr = Expression() )?
3518 <END>
3519 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003520 if (conditionExpr == null) {
3521 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003522 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003523 }
3524 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003525 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07003526 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003527}
3528
Yingyi Buab817482016-08-19 21:29:31 -07003529SelectExpression SelectExpression(boolean subquery) throws ParseException:
3530{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003531 List<LetClause> letClauses = new ArrayList<LetClause>();
3532 SelectSetOperation selectSetOperation;
3533 OrderbyClause orderbyClause = null;
3534 LimitClause limitClause = null;
3535 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07003536}
3537{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003538 ( letClauses = LetClause() )?
3539 selectSetOperation = SelectSetOperation()
3540 (orderbyClause = OrderbyClause() {})?
3541 (limitClause = LimitClause() {})?
3542 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003543 SelectExpression selectExpr =
3544 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
3545 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
3546 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003547 }
3548}
3549
Yingyi Buab817482016-08-19 21:29:31 -07003550SelectSetOperation SelectSetOperation() throws ParseException:
3551{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003552 SetOperationInput setOperationInputLeft;
3553 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
3554}
3555{
3556 {
3557 SelectBlock selectBlockLeft = null;
3558 SelectExpression subqueryLeft = null;
3559 Expression expr = null;
3560 }
3561 selectBlockLeft = SelectBlock()
3562 {
3563 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
3564 }
3565 (
3566 {
3567 SetOpType opType = SetOpType.UNION;
3568 boolean setSemantics = true;
3569 SelectBlock selectBlockRight = null;
3570 SelectExpression subqueryRight = null;
3571 }
3572 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
3573 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
3574 {
3575 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
3576 }
3577 )*
3578 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003579 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
3580 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
3581 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003582 }
3583}
3584
Yingyi Buab817482016-08-19 21:29:31 -07003585SelectExpression Subquery() throws ParseException:
3586{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003587 SelectExpression selectExpr = null;
3588}
3589{
3590 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
3591 {
3592 return selectExpr;
3593 }
3594}
3595
Yingyi Buab817482016-08-19 21:29:31 -07003596SelectBlock SelectBlock() throws ParseException:
3597{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003598 SelectClause selectClause = null;
3599 FromClause fromClause = null;
3600 List<LetClause> fromLetClauses = null;
3601 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003602 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003603 GroupbyClause groupbyClause = null;
3604 List<LetClause> gbyLetClauses = null;
3605 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003606 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003607 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003608}
3609{
3610 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003611 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003612 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003613 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07003614 fromClause = FromClause()
3615 (
3616 fromLetClauses = LetClause()
3617 )?
3618 )
3619 |
3620 (
3621 fromLetClauses = LetClause()
3622 {
3623 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
3624 SourceLocation sourceLoc = getSourceLocation(token);
3625 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
3626 missingExpr.setSourceLocation(sourceLoc);
3627 List<Expression> list = new ArrayList<Expression>(1);
3628 list.add(missingExpr);
3629 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
3630 listExpr.setSourceLocation(sourceLoc);
3631 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
3632 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
3633 fromVar.setSourceLocation(sourceLoc);
3634 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
3635 fromClause = new FromClause(fromTerms);
3636 }
3637 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003638 )?
3639 (whereClause = WhereClause())?
3640 (
3641 groupbyClause = GroupbyClause()
3642 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003643 gbyLetClauses = LetClause()
3644 )?
3645 (havingClause = HavingClause())?
3646 )?
3647 |
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003648 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003649 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003650 fromLetClauses = LetClause()
3651 )?
3652 (whereClause = WhereClause())?
3653 (
3654 groupbyClause = GroupbyClause()
3655 (
3656 gbyLetClauses = LetClause()
3657 )?
3658 (havingClause = HavingClause())?
3659 )?
3660 selectClause = SelectClause()
3661 )
3662 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08003663 if (fromLetClauses != null) {
3664 fromLetWhereClauses.addAll(fromLetClauses);
3665 }
3666 if (whereClause != null) {
3667 fromLetWhereClauses.add(whereClause);
3668 }
3669 if (gbyLetClauses != null) {
3670 gbyLetHavingClauses.addAll(gbyLetClauses);
3671 }
3672 if (havingClause != null) {
3673 gbyLetHavingClauses.add(havingClause);
3674 }
3675 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
3676 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003677 selectBlock.setSourceLocation(startSrcLoc);
3678 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003679 }
3680}
3681
Yingyi Buab817482016-08-19 21:29:31 -07003682SelectClause SelectClause() throws ParseException:
3683{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003684 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003685 SelectRegular selectRegular = null;
3686 SelectElement selectElement = null;
3687 boolean distinct = false;
3688}
3689{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003690 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003691 (
3692 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04003693 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07003694 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07003695 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003696 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003697 SourceLocation sourceLoc = getSourceLocation(startToken);
3698 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07003699 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003700 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003701 List<Projection> projections = new ArrayList<Projection>();
3702 projections.add(projection);
3703 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003704 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07003705 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003706 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
3707 selectClause.setSourceLocation(sourceLoc);
3708 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003709 }
3710}
3711
Yingyi Buab817482016-08-19 21:29:31 -07003712SelectRegular SelectRegular() throws ParseException:
3713{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003714 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003715 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003716 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003717}
3718{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003719 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003720 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003721 projections.add(projection);
3722 startSrcLoc = projection.getSourceLocation();
3723 }
3724 ( LOOKAHEAD(2) <COMMA> projection = Projection()
3725 {
3726 projections.add(projection);
3727 }
3728 )*
3729 {
3730 SelectRegular selectRegular = new SelectRegular(projections);
3731 selectRegular.setSourceLocation(startSrcLoc);
3732 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003733 }
3734}
3735
Yingyi Buab817482016-08-19 21:29:31 -07003736SelectElement SelectElement() throws ParseException:
3737{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003738 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003739 Expression expr = null;
3740 String name = null;
3741}
3742{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003743 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003744 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003745 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003746 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003747 }
3748}
3749
Yingyi Buab817482016-08-19 21:29:31 -07003750Projection Projection() throws ParseException :
3751{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003752 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003753 Expression expr = null;
3754 Identifier identifier = null;
3755 String name = null;
3756 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003757 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003758}
3759{
3760 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003761 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
3762 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08003763 | expr = Expression() ((<AS>)? name = Identifier())?
3764 {
3765 if (name == null) {
3766 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
3767 if (generatedColumnIdentifier != null) {
3768 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
3769 }
3770 }
3771 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003772 )
3773 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003774 Projection projection = new Projection(expr, name, star, varStar);
3775 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
3776 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003777 }
3778}
3779
3780FromClause FromClause() throws ParseException :
3781{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003782 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003783 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
3784 extendCurrentScope();
3785}
3786{
3787 {
3788 FromTerm fromTerm = null;
3789 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003790 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003791 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
3792 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003793 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003794 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003795 }
3796}
3797
3798FromTerm FromTerm() throws ParseException :
3799{
3800 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04003801 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003802 VariableExpr posVar = null;
3803 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
3804}
3805{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003806 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003807 (
3808 {JoinType joinType = JoinType.INNER; }
3809 (joinType = JoinType())?
3810 {
3811 AbstractBinaryCorrelateClause correlateClause = null;
3812 }
3813 (correlateClause = JoinClause(joinType)
Yingyi Bu6d999ed2016-07-13 11:59:06 -07003814 | correlateClause = UnnestClause(joinType)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003815 )
3816 {
3817 correlateClauses.add(correlateClause);
3818 }
3819 )*
3820 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003821 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003822 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003823 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003824 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
3825 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
3826 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003827 }
3828}
3829
3830JoinClause JoinClause(JoinType joinType) throws ParseException :
3831{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003832 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003833 Expression rightExpr = null;
3834 VariableExpr rightVar = null;
3835 VariableExpr posVar = null;
3836 Expression conditionExpr = null;
3837}
3838{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003839 <JOIN> { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())? <ON> conditionExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003840 {
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003841 if(rightVar==null){
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07003842 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003843 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003844 JoinClause joinClause = new JoinClause(joinType, rightExpr, rightVar, posVar, conditionExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003845 return addSourceLocation(joinClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003846 }
3847}
3848
Yingyi Bu391f09e2015-10-29 13:49:39 -07003849UnnestClause UnnestClause(JoinType joinType) throws ParseException :
3850{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003851 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003852 Expression rightExpr;
3853 VariableExpr rightVar;
3854 VariableExpr posVar = null;
3855}
3856{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003857 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable()) (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003858 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003859 if (rightVar == null) {
3860 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07003861 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003862 UnnestClause unnestClause = new UnnestClause(joinType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003863 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003864 }
3865}
3866
Yingyi Bu391f09e2015-10-29 13:49:39 -07003867JoinType JoinType() throws ParseException :
3868{
3869 JoinType joinType = JoinType.INNER;
3870}
3871{
3872 (<INNER>|<LEFT> (<OUTER>)? {joinType = JoinType.LEFTOUTER; })
3873 {
3874 return joinType;
3875 }
3876}
3877
3878List<LetClause> LetClause() throws ParseException:
3879{
3880 List<LetClause> letList = new ArrayList<LetClause>();
3881 LetClause letClause;
3882}
3883{
3884 (
3885 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
3886 |
3887 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
3888 )
3889 {
3890 return letList;
3891 }
3892}
3893
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003894WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07003895{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003896 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003897 Expression whereExpr;
3898}
3899{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003900 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003901 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003902 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003903 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003904 }
3905}
3906
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003907OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07003908{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003909 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003910 OrderbyClause oc = new OrderbyClause();
3911 Expression orderbyExpr;
3912 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003913 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003914 int numOfOrderby = 0;
3915}
3916{
3917 <ORDER>
3918 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003919 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003920 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
3921 if (hintToken != null) {
3922 switch (hintToken.hint) {
3923 case INMEMORY_HINT:
3924 String[] splits = hintToken.hintParams.split("\\s+");
3925 int numFrames = Integer.parseInt(splits[0]);
3926 int numTuples = Integer.parseInt(splits[1]);
3927 oc.setNumFrames(numFrames);
3928 oc.setNumTuples(numTuples);
3929 break;
3930 case RANGE_HINT:
3931 try {
3932 oc.setRangeMap(RangeMapBuilder.parseHint(parseExpression(hintToken.hintParams)));
3933 } catch (CompilationException e) {
3934 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
3935 }
3936 break;
Ali Alsuliman80225e22018-10-15 14:17:07 -07003937 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003938 }
3939 }
3940 <BY> orderbyExpr = Expression()
3941 {
3942 orderbyList.add(orderbyExpr);
3943 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
3944 }
3945 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
3946 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
3947 {
3948 modifierList.add(modif);
3949 }
3950
3951 (LOOKAHEAD(2) <COMMA> orderbyExpr = Expression()
3952 {
3953 orderbyList.add(orderbyExpr);
3954 modif = OrderbyClause.OrderModifier.ASC;
3955 }
3956 ( (<ASC> { modif = OrderbyClause.OrderModifier.ASC; })
3957 | (<DESC> { modif = OrderbyClause.OrderModifier.DESC; }))?
3958 {
3959 modifierList.add(modif);
3960 }
3961 )*
3962
3963 {
3964 oc.setModifierList(modifierList);
3965 oc.setOrderbyList(orderbyList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003966 return addSourceLocation(oc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003967 }
3968}
3969
3970GroupbyClause GroupbyClause()throws ParseException :
3971{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003972 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003973 GroupbyClause gbc = new GroupbyClause();
3974 List<GbyVariableExpressionPair> vePairList = new ArrayList<GbyVariableExpressionPair>();
3975 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003976 Expression expr = null;
3977 VariableExpr decorVar = null;
3978 Expression decorExpr = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003979 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07003980 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003981 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003982}
3983{
3984 {
3985 Scope newScope = extendCurrentScopeNoPush(true);
3986 // extendCurrentScope(true);
3987 }
3988 <GROUP>
3989 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003990 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003991 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
3992 if (hintToken != null) {
3993 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003994 }
3995 }
3996 <BY> (
3997 expr = Expression()
3998 (LOOKAHEAD(1) (<AS>)?
3999 var = Variable()
Yingyi Bucaea8f02015-11-16 15:12:15 -08004000 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004001 {
4002 GbyVariableExpressionPair pair1 = new GbyVariableExpressionPair(var, expr);
4003 vePairList.add(pair1);
4004 }
4005 ( LOOKAHEAD(1) <COMMA>
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004006 {
4007 var = null;
4008 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004009 expr = Expression()
4010 (LOOKAHEAD(1) (<AS>)?
4011 var = Variable()
Yingyi Bucaea8f02015-11-16 15:12:15 -08004012 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004013 {
4014 GbyVariableExpressionPair pair2 = new GbyVariableExpressionPair(var, expr);
4015 vePairList.add(pair2);
4016 }
4017 )*
4018 )
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004019 (<GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
4020 {
4021 groupVar = groupVarWithFieldList.first;
4022 groupFieldList = groupVarWithFieldList.second;
4023 }
Yingyi Buacc12a92016-03-26 17:25:05 -07004024 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004025 {
4026 gbc.setGbyPairList(vePairList);
4027 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07004028 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07004029 gbc.setGroupVar(groupVar);
4030 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004031 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004032 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004033 }
4034}
4035
4036HavingClause HavingClause() throws ParseException:
4037{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004038 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004039 Expression filterExpr = null;
4040}
4041{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004042 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004043 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004044 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004045 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004046 }
4047}
4048
4049LimitClause LimitClause() throws ParseException:
4050{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004051 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004052 LimitClause lc = new LimitClause();
4053 Expression expr;
4054 pushForbiddenScope(getCurrentScope());
4055}
4056{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004057 <LIMIT> { startToken = token; } expr = Expression() { lc.setLimitExpr(expr); }
4058 (<OFFSET> expr = Expression() { lc.setOffset(expr); })?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004059
4060 {
4061 popForbiddenScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004062 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004063 }
4064}
4065
4066QuantifiedExpression QuantifiedExpression()throws ParseException:
4067{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004068 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004069 QuantifiedExpression qc = new QuantifiedExpression();
4070 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
4071 Expression satisfiesExpr;
4072 VariableExpr var;
4073 Expression inExpr;
4074 QuantifiedPair pair;
4075}
4076{
4077 {
4078 createNewScope();
4079 }
4080
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004081 ( ((<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); })
4082 | (<EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); }))
Yingyi Bu391f09e2015-10-29 13:49:39 -07004083 var = Variable() <IN> inExpr = Expression()
4084 {
4085 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004086 quantifiedList.add(pair);
4087 }
4088 (
4089 <COMMA> var = Variable() <IN> inExpr = Expression()
4090 {
4091 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004092 quantifiedList.add(pair);
4093 }
4094 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07004095 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004096 {
4097 qc.setSatisfiesExpr(satisfiesExpr);
4098 qc.setQuantifiedList(quantifiedList);
4099 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004100 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004101 }
4102}
4103
4104LetClause LetElement() throws ParseException:
4105{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004106 VariableExpr varExp;
4107 Expression beExp;
4108 extendCurrentScope();
4109}
4110{
4111 varExp = Variable() <EQ> beExp = Expression()
4112 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004113 LetClause lc = new LetClause(varExp, beExp);
4114 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004115 return lc;
4116 }
4117}
4118
4119LetClause WithElement() throws ParseException:
4120{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004121 VariableExpr varExp;
4122 Expression beExp;
4123 extendCurrentScope();
4124}
4125{
4126 varExp = Variable() <AS> beExp = Expression()
4127 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004128 LetClause lc = new LetClause(varExp, beExp);
4129 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004130 return lc;
4131 }
4132}
4133
4134TOKEN_MGR_DECLS:
4135{
4136 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07004137 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004138 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004139
4140 public void pushState() {
4141 lexerStateStack.push( curLexState );
4142 }
4143
4144 public void popState(String token) {
4145 if (lexerStateStack.size() > 0) {
4146 SwitchTo( lexerStateStack.pop() );
4147 } else {
4148 int errorLine = input_stream.getEndLine();
4149 int errorColumn = input_stream.getEndColumn();
4150 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
4151 + "\" but state stack is empty.";
4152 throw new TokenMgrError(msg, -1);
4153 }
4154 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07004155
4156 void CommonTokenAction(Token token) {
4157 Token hintToken = token.specialToken;
4158 if (hintToken != null) {
4159 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
4160 String text = hintToken.image.substring(1).trim();
4161 boolean hintFound = hintToken.parseHint(text);
4162 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
4163 }
4164 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004165}
4166
4167<DEFAULT,IN_DBL_BRACE>
4168TOKEN [IGNORE_CASE]:
4169{
Ian Maxon38fe9402020-01-29 19:27:40 -08004170 <ADAPTER: "adapter">
4171 | <ALL : "all">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004172 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07004173 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004174 | <APPLY : "apply">
4175 | <AS : "as">
4176 | <ASC : "asc">
4177 | <AT : "at">
4178 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004179 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004180 | <BTREE : "btree">
4181 | <BY : "by">
4182 | <CASE : "case">
4183 | <CLOSED : "closed">
4184 | <CREATE : "create">
4185 | <COMPACTION : "compaction">
4186 | <COMPACT : "compact">
4187 | <CONNECT : "connect">
4188 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07004189 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07004190 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07004191 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004192 | <DECLARE : "declare">
4193 | <DEFINITION : "definition">
4194 | <DELETE : "delete">
4195 | <DESC : "desc">
4196 | <DISCONNECT : "disconnect">
4197 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004198 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004199 | <DROP : "drop">
4200 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07004201 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004202 | <ELSE : "else">
4203 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07004204 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004205 | <EVERY : "every">
4206 | <EXCEPT : "except">
4207 | <EXISTS : "exists">
4208 | <EXTERNAL : "external">
4209 | <FEED : "feed">
4210 | <FILTER : "filter">
4211 | <FLATTEN : "flatten">
4212 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004213 | <FROM : "from">
4214 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08004215 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004216 | <FUNCTION : "function">
4217 | <GROUP : "group">
4218 | <HAVING : "having">
4219 | <HINTS : "hints">
4220 | <IF : "if">
4221 | <INTO : "into">
4222 | <IN : "in">
4223 | <INDEX : "index">
4224 | <INGESTION : "ingestion">
4225 | <INNER : "inner">
4226 | <INSERT : "insert">
4227 | <INTERNAL : "internal">
4228 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07004229 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004230 | <JOIN : "join">
4231 | <KEYWORD : "keyword">
4232 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07004233 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004234 | <LEFT : "left">
4235 | <LETTING : "letting">
4236 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07004237 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004238 | <LIMIT : "limit">
4239 | <LOAD : "load">
Murtadha Hubail29f63912019-04-21 14:23:57 +03004240 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004241 | <NODEGROUP : "nodegroup">
4242 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07004243 | <NOT : "not">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004244 | <OFFSET : "offset">
4245 | <ON : "on">
4246 | <OPEN : "open">
4247 | <OR : "or">
4248 | <ORDER : "order">
4249 | <OUTER : "outer">
4250 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004251 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004252 | <PATH : "path">
4253 | <POLICY : "policy">
4254 | <PRESORTED : "pre-sorted">
4255 | <PRIMARY : "primary">
4256 | <RAW : "raw">
4257 | <REFRESH : "refresh">
4258 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004259 | <RETURNING : "returning">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004260 | <RTREE : "rtree">
4261 | <RUN : "run">
4262 | <SATISFIES : "satisfies">
4263 | <SECONDARY : "secondary">
4264 | <SELECT : "select">
4265 | <SET : "set">
4266 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08004267 | <START : "start">
4268 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08004269 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03004270 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07004271 | <THEN : "then">
4272 | <TYPE : "type">
4273 | <TO : "to">
4274 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08004275 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004276 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07004277 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08004278 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07004279 | <USE : "use">
4280 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004281 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08004282 | <VALUED : "valued">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004283 | <WHEN : "when">
4284 | <WHERE : "where">
4285 | <WITH : "with">
4286 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004287}
4288
4289<DEFAULT,IN_DBL_BRACE>
4290TOKEN :
4291{
4292 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07004293 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004294 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004295 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004296 | <MUL : "*">
4297 | <PLUS : "+">
4298
4299 | <LEFTPAREN : "(">
4300 | <RIGHTPAREN : ")">
4301 | <LEFTBRACKET : "[">
4302 | <RIGHTBRACKET : "]">
4303
4304 | <ATT : "@">
4305 | <COLON : ":">
4306 | <COMMA : ",">
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004307 | <DOLLAR: "$">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004308 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004309 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004310 | <QUES : "?">
4311 | <SEMICOLON : ";">
4312 | <SHARP : "#">
4313
4314 | <LT : "<">
4315 | <GT : ">">
4316 | <LE : "<=">
4317 | <GE : ">=">
4318 | <EQ : "=">
4319 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07004320 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004321 | <SIMILAR : "~=">
4322}
4323
4324<DEFAULT,IN_DBL_BRACE>
4325TOKEN :
4326{
4327 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
4328}
4329
4330<DEFAULT>
4331TOKEN :
4332{
4333 <RIGHTBRACE : "}"> { popState("}"); }
4334}
4335
4336<DEFAULT,IN_DBL_BRACE>
4337TOKEN :
4338{
4339 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
4340}
4341
4342<IN_DBL_BRACE>
4343TOKEN :
4344{
4345 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
4346}
4347
4348<DEFAULT,IN_DBL_BRACE>
4349TOKEN :
4350{
4351 <INTEGER_LITERAL : (<DIGIT>)+ >
4352}
4353
4354<DEFAULT,IN_DBL_BRACE>
Yingyi Bue311a632016-06-07 18:23:16 -07004355TOKEN [IGNORE_CASE]:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004356{
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004357 <MISSING : "missing">
4358 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07004359 | <TRUE : "true">
4360 | <FALSE : "false">
4361}
4362
4363<DEFAULT,IN_DBL_BRACE>
4364TOKEN :
4365{
4366 <#DIGIT : ["0" - "9"]>
4367}
4368
4369<DEFAULT,IN_DBL_BRACE>
4370TOKEN:
4371{
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07004372 < DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
4373 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
4374 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004375 >
Yingyi Bue4d919e2016-10-30 10:47:03 -07004376 | < FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
4377 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
4378 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004379 >
4380 | <DIGITS : (<DIGIT>)+ >
4381}
4382
4383<DEFAULT,IN_DBL_BRACE>
4384TOKEN :
4385{
4386 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08004387 | <#IDENTIFIER_SPECIALCHARS_START : ["_"]>
4388 | <#IDENTIFIER_SPECIALCHARS_REST : ["$"]>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004389}
4390
4391<DEFAULT,IN_DBL_BRACE>
4392TOKEN :
4393{
4394 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07004395 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004396 <EscapeQuot>
4397 | <EscapeBslash>
4398 | <EscapeSlash>
4399 | <EscapeBspace>
4400 | <EscapeFormf>
4401 | <EscapeNl>
4402 | <EscapeCr>
4403 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004404 | ~["`","\\"])* "`">
4405 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004406 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004407 | <EscapeBslash>
4408 | <EscapeSlash>
4409 | <EscapeBspace>
4410 | <EscapeFormf>
4411 | <EscapeNl>
4412 | <EscapeCr>
4413 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07004414 | ~["\"","\\"])* "\"")
4415 | ("\'"(
4416 <EscapeApos>
4417 | <EscapeBslash>
4418 | <EscapeSlash>
4419 | <EscapeBspace>
4420 | <EscapeFormf>
4421 | <EscapeNl>
4422 | <EscapeCr>
4423 | <EscapeTab>
4424 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004425 | < #EscapeQuot: "\\\"" >
4426 | < #EscapeApos: "\\\'" >
4427 | < #EscapeBslash: "\\\\" >
4428 | < #EscapeSlash: "\\/" >
4429 | < #EscapeBspace: "\\b" >
4430 | < #EscapeFormf: "\\f" >
4431 | < #EscapeNl: "\\n" >
4432 | < #EscapeCr: "\\r" >
4433 | < #EscapeTab: "\\t" >
4434}
4435
4436<DEFAULT,IN_DBL_BRACE>
4437TOKEN :
4438{
Dmitry Lychagin9c83efb2018-01-16 14:31:58 -08004439 <IDENTIFIER : ( <LETTER> | <IDENTIFIER_SPECIALCHARS_START> )
4440 ( <LETTER> | <DIGIT> | <IDENTIFIER_SPECIALCHARS_START> | <IDENTIFIER_SPECIALCHARS_REST> )*>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004441}
4442
4443<DEFAULT,IN_DBL_BRACE>
4444SKIP:
4445{
4446 " "
4447 | "\t"
4448 | "\r"
4449 | "\n"
4450}
4451
4452<DEFAULT,IN_DBL_BRACE>
4453SKIP:
4454{
4455 <"//" (~["\n"])* "\n">
4456}
4457
4458<DEFAULT,IN_DBL_BRACE>
4459SKIP:
4460{
4461 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4462}
4463
4464<DEFAULT,IN_DBL_BRACE>
4465SKIP:
4466{
Yingyi Bu93846a72016-09-13 16:30:39 -07004467 <"--" (~["\n"])* "\n">
4468}
4469
4470
4471<DEFAULT,IN_DBL_BRACE>
4472SKIP:
4473{
4474 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
4475}
4476
4477<DEFAULT,IN_DBL_BRACE>
4478SKIP:
4479{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004480 <"/*"> { pushState(); } : INSIDE_COMMENT
4481}
4482
4483<INSIDE_COMMENT>
4484SPECIAL_TOKEN:
4485{
4486 <"+"(" ")*(~["*"])*>
4487}
4488
4489<INSIDE_COMMENT>
4490SKIP:
4491{
4492 <"/*"> { pushState(); }
4493}
4494
4495<INSIDE_COMMENT>
4496SKIP:
4497{
4498 <"*/"> { popState("*/"); }
4499 | <~[]>
4500}