blob: 728b2409fb83305112713bfde3583c86694904a6 [file] [log] [blame]
Michael Blow82464fb2017-03-28 18:48:13 -04001//
2// Licensed to the Apache Software Foundation (ASF) under one
3// or more contributor license agreements. See the NOTICE file
4// distributed with this work for additional information
5// regarding copyright ownership. The ASF licenses this file
6// to you under the Apache License, Version 2.0 (the
7// "License"); you may not use this file except in compliance
8// with the License. You may obtain a copy of the License at
9//
10// http://www.apache.org/licenses/LICENSE-2.0
11//
12// Unless required by applicable law or agreed to in writing,
13// software distributed under the License is distributed on an
14// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15// KIND, either express or implied. See the License for the
16// specific language governing permissions and limitations
17// under the License.
18//
Yingyi Bu391f09e2015-10-29 13:49:39 -070019options {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -070020 COMMON_TOKEN_ACTION = true;
21 STATIC = false;
22 TOKEN_EXTENDS = "org.apache.asterix.lang.sqlpp.parser.SqlppToken";
Yingyi Bu391f09e2015-10-29 13:49:39 -070023}
24
Yingyi Bu391f09e2015-10-29 13:49:39 -070025PARSER_BEGIN(SQLPPParser)
26
27package org.apache.asterix.lang.sqlpp.parser;
28
29// For SQL++ ParserTokenManager
Till Westmanne9b2adf2016-10-15 12:39:01 -070030import java.util.ArrayDeque;
Yingyi Bu391f09e2015-10-29 13:49:39 -070031
32import java.io.BufferedReader;
33import java.io.File;
34import java.io.FileInputStream;
35import java.io.FileNotFoundException;
36import java.io.IOException;
37import java.io.InputStreamReader;
38import java.io.Reader;
39import java.io.StringReader;
40import java.util.ArrayList;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070041import java.util.Arrays;
42import java.util.Collection;
Yingyi Budaa549c2016-06-28 22:30:52 -070043import java.util.Collections;
Yingyi Bu391f09e2015-10-29 13:49:39 -070044import java.util.HashMap;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070045import java.util.HashSet;
Yingyi Bu391f09e2015-10-29 13:49:39 -070046import java.util.Iterator;
47import java.util.LinkedHashMap;
48import java.util.List;
49import java.util.Map;
50
51import org.apache.asterix.common.annotations.AutoDataGen;
52import org.apache.asterix.common.annotations.DateBetweenYearsDataGen;
53import org.apache.asterix.common.annotations.DatetimeAddRandHoursDataGen;
54import org.apache.asterix.common.annotations.DatetimeBetweenYearsDataGen;
55import org.apache.asterix.common.annotations.FieldIntervalDataGen;
56import org.apache.asterix.common.annotations.FieldValFileDataGen;
57import org.apache.asterix.common.annotations.FieldValFileSameIndexDataGen;
58import org.apache.asterix.common.annotations.IRecordFieldDataGen;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080059import org.apache.asterix.common.annotations.IndexedNLJoinExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070060import org.apache.asterix.common.annotations.InsertRandIntDataGen;
61import org.apache.asterix.common.annotations.ListDataGen;
62import org.apache.asterix.common.annotations.ListValFileDataGen;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -060063import org.apache.asterix.common.annotations.RangeAnnotation;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080064import org.apache.asterix.common.annotations.SecondaryIndexSearchPreferenceAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070065import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
Tin Vu4f1090d2021-09-21 02:06:32 -070066import org.apache.asterix.common.annotations.SpatialJoinAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070067import org.apache.asterix.common.annotations.TypeDataGen;
68import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
69import org.apache.asterix.common.config.DatasetConfig.DatasetType;
70import org.apache.asterix.common.config.DatasetConfig.IndexType;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -070071import org.apache.asterix.common.exceptions.AsterixException;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080072import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070073import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070074import org.apache.asterix.common.exceptions.WarningCollector;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080075import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070076import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin54c06012019-11-13 15:54:20 -080077import org.apache.asterix.common.metadata.DataverseName;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -070078import org.apache.asterix.common.metadata.DatasetFullyQualifiedName;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080079import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070080import org.apache.asterix.lang.common.base.AbstractLangExpression;
81import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070082import org.apache.asterix.lang.common.base.Expression;
83import org.apache.asterix.lang.common.base.Literal;
84import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070085import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070086import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070087import org.apache.asterix.lang.common.base.Statement;
88import org.apache.asterix.lang.common.clause.GroupbyClause;
89import org.apache.asterix.lang.common.clause.LetClause;
90import org.apache.asterix.lang.common.clause.LimitClause;
91import org.apache.asterix.lang.common.clause.OrderbyClause;
92import org.apache.asterix.lang.common.clause.UpdateClause;
93import org.apache.asterix.lang.common.clause.WhereClause;
94import org.apache.asterix.lang.common.context.RootScopeFactory;
95import org.apache.asterix.lang.common.context.Scope;
96import org.apache.asterix.lang.common.expression.AbstractAccessor;
97import org.apache.asterix.lang.common.expression.CallExpr;
98import org.apache.asterix.lang.common.expression.FieldAccessor;
99import org.apache.asterix.lang.common.expression.FieldBinding;
100import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
101import org.apache.asterix.lang.common.expression.IfExpr;
102import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700103import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700104import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +0300105import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700106import org.apache.asterix.lang.common.expression.LiteralExpr;
107import org.apache.asterix.lang.common.expression.OperatorExpr;
108import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
109import org.apache.asterix.lang.common.expression.QuantifiedExpression;
110import org.apache.asterix.lang.common.expression.RecordConstructor;
111import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
112import org.apache.asterix.lang.common.expression.TypeExpression;
113import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
114import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700115import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
116import org.apache.asterix.lang.common.expression.VariableExpr;
117import org.apache.asterix.lang.common.literal.DoubleLiteral;
118import org.apache.asterix.lang.common.literal.FalseLiteral;
119import org.apache.asterix.lang.common.literal.FloatLiteral;
120import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700121import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700122import org.apache.asterix.lang.common.literal.NullLiteral;
123import org.apache.asterix.lang.common.literal.StringLiteral;
124import org.apache.asterix.lang.common.literal.TrueLiteral;
125import org.apache.asterix.lang.common.parser.ScopeChecker;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -0700126import org.apache.asterix.lang.common.statement.AdapterDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700127import org.apache.asterix.lang.common.statement.CompactStatement;
128import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800129import org.apache.asterix.lang.common.statement.StartFeedStatement;
130import org.apache.asterix.lang.common.statement.StopFeedStatement;
Ian Maxon38fe9402020-01-29 19:27:40 -0800131import org.apache.asterix.lang.common.statement.CreateAdapterStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700132import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
133import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
134import org.apache.asterix.lang.common.statement.CreateFeedStatement;
135import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
136import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800137import org.apache.asterix.lang.common.statement.CreateSynonymStatement;
Rui Guoe6986dd2020-11-25 19:50:06 -0800138import org.apache.asterix.lang.common.statement.CreateFullTextFilterStatement;
139import org.apache.asterix.lang.common.statement.CreateFullTextConfigStatement;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700140import org.apache.asterix.lang.common.statement.CreateViewStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700141import org.apache.asterix.lang.common.statement.DatasetDecl;
142import org.apache.asterix.lang.common.statement.DataverseDecl;
143import org.apache.asterix.lang.common.statement.DataverseDropStatement;
144import org.apache.asterix.lang.common.statement.DeleteStatement;
145import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700146import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700147import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
148import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300149import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700150import org.apache.asterix.lang.common.statement.FunctionDecl;
151import org.apache.asterix.lang.common.statement.FunctionDropStatement;
152import org.apache.asterix.lang.common.statement.IndexDropStatement;
Rui Guoe6986dd2020-11-25 19:50:06 -0800153import org.apache.asterix.lang.common.statement.FullTextFilterDropStatement;
154import org.apache.asterix.lang.common.statement.FullTextConfigDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700155import org.apache.asterix.lang.common.statement.InsertStatement;
156import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
157import org.apache.asterix.lang.common.statement.LoadStatement;
158import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
159import org.apache.asterix.lang.common.statement.NodegroupDecl;
160import org.apache.asterix.lang.common.statement.Query;
161import org.apache.asterix.lang.common.statement.RefreshExternalDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700162import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800163import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700164import org.apache.asterix.lang.common.statement.TypeDecl;
165import org.apache.asterix.lang.common.statement.TypeDropStatement;
166import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800167import org.apache.asterix.lang.common.statement.UpsertStatement;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700168import org.apache.asterix.lang.common.statement.ViewDecl;
169import org.apache.asterix.lang.common.statement.ViewDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700170import org.apache.asterix.lang.common.statement.WriteStatement;
171import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800172import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700173import org.apache.asterix.lang.common.struct.QuantifiedPair;
174import org.apache.asterix.lang.common.struct.VarIdentifier;
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -0800175import org.apache.asterix.lang.common.util.DatasetDeclParametersUtil;
Tin Vu4f1090d2021-09-21 02:06:32 -0700176import org.apache.asterix.lang.common.util.ExpressionUtils;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700177import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700178import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
179import org.apache.asterix.lang.sqlpp.clause.FromClause;
180import org.apache.asterix.lang.sqlpp.clause.FromTerm;
181import org.apache.asterix.lang.sqlpp.clause.HavingClause;
182import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700183import org.apache.asterix.lang.sqlpp.clause.Projection;
184import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
185import org.apache.asterix.lang.sqlpp.clause.SelectClause;
186import org.apache.asterix.lang.sqlpp.clause.SelectElement;
187import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
188import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
189import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800190import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700191import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700192import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700193import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700194import org.apache.asterix.lang.sqlpp.optype.JoinType;
195import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -0700196import org.apache.asterix.lang.sqlpp.optype.UnnestType;
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700197import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser;
198import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingElement;
199import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSet;
200import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSets;
201import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.RollupCube;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700202import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700203import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
204import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700205import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700206import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700207import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Ian Maxon38fe9402020-01-29 19:27:40 -0800208import org.apache.asterix.external.dataset.adapter.AdapterIdentifier;
Tin Vu4f1090d2021-09-21 02:06:32 -0700209import org.apache.asterix.om.exceptions.TypeMismatchException;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800210import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700211import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700212import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700213import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700214import org.apache.hyracks.algebricks.common.utils.Pair;
215import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800216import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700217import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700218import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700219import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800220import org.apache.hyracks.api.exceptions.IWarningCollector;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700221import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700222import org.apache.hyracks.api.exceptions.Warning;
Dmitry Lychagin70bf47a2020-10-22 13:25:57 -0700223import org.apache.hyracks.dataflow.common.data.partition.range.RangeMap;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800224import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700225import org.apache.hyracks.util.StringUtil;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700226
Yingyi Bucaea8f02015-11-16 15:12:15 -0800227class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700228
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800229 // tokens parsed as identifiers
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700230 private static final String CUBE = "CUBE";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800231 private static final String CURRENT = "CURRENT";
Dmitry Lychagin33c77f92021-07-21 15:59:04 -0700232 private static final String DEFAULT = "DEFAULT";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800233 private static final String EXCLUDE = "EXCLUDE";
Ali Alsuliman931e7382021-08-08 21:55:59 +0300234 private static final String INCLUDE = "INCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700235 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800236 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychagin914983f2021-09-30 15:11:11 -0700237 private static final String FOREIGN = "FOREIGN";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700238 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800239 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700240 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700241 private static final String LAST = "LAST";
Glenn67fd1f32021-02-25 16:04:49 -0800242 private static final String META = "META";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800243 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700244 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800245 private static final String OTHERS = "OTHERS";
246 private static final String PARTITION = "PARTITION";
247 private static final String PRECEDING = "PRECEDING";
248 private static final String RANGE = "RANGE";
Dmitry Lychagin914983f2021-09-30 15:11:11 -0700249 private static final String REFERENCES = "REFERENCES";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700250 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700251 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800252 private static final String ROW = "ROW";
253 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700254 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800255 private static final String TIES = "TIES";
256 private static final String UNBOUNDED = "UNBOUNDED";
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700257 private static final String REPLACE = "REPLACE";
Ian Maxon38fe9402020-01-29 19:27:40 -0800258 private static final String RETURNS = "RETURNS";
Rui Guoe6986dd2020-11-25 19:50:06 -0800259 private static final String CONFIG = "CONFIG";
260
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800261
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700262 private static final String INT_TYPE_NAME = "int";
Dmitry Lychagin541652082020-11-09 14:04:53 -0800263 private static final String UDF_VARARGS_PARAM_NAME = "args"; // Note: this value is stored in the function metadata
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700264
Till Westmann7199a562016-09-17 16:07:32 -0700265 // error configuration
266 protected static final boolean REPORT_EXPECTED_TOKENS = false;
267
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700268 private int externalVarCounter;
269
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800270 private DataverseName defaultDataverse;
271
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700272 private SqlppGroupingSetsParser groupingSetsParser;
273
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700274 private final WarningCollector warningCollector = new WarningCollector();
275
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700276 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
277
Yingyi Bu391f09e2015-10-29 13:49:39 -0700278 private static class IndexParams {
279 public IndexType type;
280 public int gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800281 public String fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700282
Rui Guoe6986dd2020-11-25 19:50:06 -0800283 public IndexParams(IndexType type, int gramLength, String fullTextConfig) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700284 this.type = type;
285 this.gramLength = gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800286 this.fullTextConfig = fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700287 }
Ali Alsuliman69ce7d82021-11-01 15:00:16 -0700288 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700289
290 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800291 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700292 public String library;
293 public String function;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -0600294 public Token hintToken;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700295 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700296 }
297
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700298 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
299 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
300 switch (hintToken.hint) {
301 case VAL_FILE_HINT:
302 File[] valFiles = new File[splits.length];
303 for (int k=0; k<splits.length; k++) {
304 valFiles[k] = new File(splits[k]);
305 }
306 return new FieldValFileDataGen(valFiles);
307 case VAL_FILE_SAME_INDEX_HINT:
308 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
309 case LIST_VAL_FILE_HINT:
310 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
311 case LIST_HINT:
312 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
313 case INTERVAL_HINT:
314 FieldIntervalDataGen.ValueType vt;
315 switch (splits[0]) {
316 case "int":
317 vt = FieldIntervalDataGen.ValueType.INT;
318 break;
319 case "long":
320 vt = FieldIntervalDataGen.ValueType.LONG;
321 break;
322 case "float":
323 vt = FieldIntervalDataGen.ValueType.FLOAT;
324 break;
325 case "double":
326 vt = FieldIntervalDataGen.ValueType.DOUBLE;
327 break;
328 default:
329 throw new SqlppParseException(getSourceLocation(hintToken),
330 "Unknown type for interval data gen: " + splits[0]);
331 }
332 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
333 case INSERT_RAND_INT_HINT:
334 return new InsertRandIntDataGen(splits[0], splits[1]);
335 case DATE_BETWEEN_YEARS_HINT:
336 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
337 case DATETIME_BETWEEN_YEARS_HINT:
338 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
339 case DATETIME_ADD_RAND_HOURS_HINT:
340 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
341 case AUTO_HINT:
342 return new AutoDataGen(splits[0]);
343 default:
344 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700345 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700346 }
347
Till Westmann7199a562016-09-17 16:07:32 -0700348 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700349 this(new StringReader(s));
350 super.setInput(s);
351 }
352
Ali Alsuliman69ce7d82021-11-01 15:00:16 -0700353 public static void main(String[] args) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700354 File file = new File(args[0]);
355 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
356 SQLPPParser parser = new SQLPPParser(fis);
357 List<Statement> st = parser.parse();
358 //st.accept(new SQLPPPrintVisitor(), 0);
359 }
360
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800361 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800362 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700363 return parseImpl(new ParseFunction<List<Statement>>() {
364 @Override
365 public List<Statement> parse() throws ParseException {
366 return SQLPPParser.this.Statement();
367 }
368 });
369 }
370
Dmitry Lychagin62f0beb2020-09-08 11:44:19 -0700371 @Override
372 public Expression parseExpression() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700373 return parseImpl(new ParseFunction<Expression>() {
374 @Override
375 public Expression parse() throws ParseException {
376 return SQLPPParser.this.Expression();
377 }
378 });
379 }
380
381 private static Expression parseExpression(String text) throws CompilationException {
382 return new SQLPPParser(text).parseExpression();
383 }
384
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800385 @Override
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700386 public List<String> parseMultipartIdentifier() throws CompilationException {
387 return parseImpl(new ParseFunction<List<String>>() {
388 @Override
389 public List<String> parse() throws ParseException {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700390 return SQLPPParser.this.MultipartIdentifier().first;
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700391 }
392 });
393 }
394
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800395 private List<String> parseParenthesizedIdentifierList() throws CompilationException {
396 return parseImpl(new ParseFunction<List<String>>() {
397 @Override
398 public List<String> parse() throws ParseException {
399 return SQLPPParser.this.ParenthesizedIdentifierList();
400 }
401 });
402 }
403
404 private static List<String> parseParenthesizedIdentifierList(String text) throws CompilationException {
405 return new SQLPPParser(text).parseParenthesizedIdentifierList();
406 }
407
Tin Vu4f1090d2021-09-21 02:06:32 -0700408 private List<Literal> parseParenthesizedLiteralList() throws CompilationException {
409 return parseImpl(new ParseFunction<List<Literal>>() {
410 @Override
411 public List<Literal> parse() throws ParseException {
412 return SQLPPParser.this.ParenthesizedLiteralList();
413 }
414 });
415 }
416
417 private static List<Literal> parseParenthesizedLiteralList(String text) throws CompilationException {
418 return new SQLPPParser(text).parseParenthesizedLiteralList();
419 }
420
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700421 @Override
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700422 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames, boolean isStored)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800423 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800424 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800425 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800426 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800427 DataverseName dataverse = defaultDataverse;
428 defaultDataverse = signature.getDataverseName();
429 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800430 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
431 for (String paramName : paramNames) {
432 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
433 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800434 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
435 removeCurrentScope();
436 defaultDataverse = dataverse;
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700437 return new FunctionDecl(signature, paramVars, functionBodyExpr, isStored);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800438 }
439 });
440 }
441
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700442 @Override
443 public ViewDecl parseViewBody(DatasetFullyQualifiedName viewName) throws CompilationException {
444 return parseImpl(new ParseFunction<ViewDecl>() {
445 @Override
446 public ViewDecl parse() throws ParseException {
447 DataverseName dataverse = defaultDataverse;
448 defaultDataverse = viewName.getDataverseName();
449 createNewScope();
450 Expression viewBodyExpr = SQLPPParser.this.ViewBody();
451 removeCurrentScope();
452 defaultDataverse = dataverse;
453 return new ViewDecl(viewName, viewBodyExpr);
454 }
455 });
456 }
457
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700458 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700459 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700460 hintCollector.clear();
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800461 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700462 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700463 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700464 } catch (SqlppParseException e) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700465 if (e.getCause() instanceof AlgebricksException) {
466 AlgebricksException cause = (AlgebricksException) e.getCause();
467 if (cause.getError().isPresent() && cause.getError().get() instanceof ErrorCode) {
468 throw new CompilationException((ErrorCode) cause.getError().get(), e.getSourceLocation(),
469 cause.getParams());
470 }
471 }
472 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(),
473 LogRedactionUtil.userData(getMessage(e)));
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700474 } catch (ParseException e) {
475 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700476 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700477 // this is here as the JavaCharStream that's below the lexer sometimes throws Errors that are not handled
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -0700478 // by the generated lexer or parser (e.g it does this for invalid backslash u + 4 hex digits escapes)
Till Westmann60f89982017-08-11 18:14:20 -0700479 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700480 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700481 } finally {
482 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700483 }
484 }
Till Westmann7199a562016-09-17 16:07:32 -0700485
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700486 @FunctionalInterface
487 private interface ParseFunction<T> {
488 T parse() throws ParseException;
489 }
490
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700491 @Override
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800492 public void getWarnings(IWarningCollector outWarningCollector) {
493 warningCollector.getWarnings(outWarningCollector);
494 }
495
496 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700497 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
498 warningCollector.getWarnings(outWarnings, maxWarnings);
499 }
500
501 @Override
502 public long getTotalWarningsCount() {
503 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700504 }
505
Till Westmann7199a562016-09-17 16:07:32 -0700506 protected String getMessage(ParseException pe) {
507 Token currentToken = pe.currentToken;
508 if (currentToken == null) {
509 return pe.getMessage();
510 }
511 int[][] expectedTokenSequences = pe.expectedTokenSequences;
512 String[] tokenImage = pe.tokenImage;
513 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
514 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
515 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
516 Token tok = currentToken.next;
517 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700518 StringBuilder message = new StringBuilder(128);
519 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700520 for (int i = 0; i < maxSize; i++) {
521 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700522 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700523 }
524 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700525 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700526 break;
527 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700528 final String fixedTokenImage = tokenImage[tok.kind];
529 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700530 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700531 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700532 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700533 tok = tok.next;
534 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700535 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700536 if (REPORT_EXPECTED_TOKENS) {
537 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700538 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700539 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700540 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700541 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700542 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700543 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700544 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700545 }
546
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700547 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700548 return
549 token == null ? null :
550 token.sourceLocation != null ? token.sourceLocation :
551 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700552 }
553
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700554 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
555 expr.setSourceLocation(getSourceLocation(token));
556 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700557 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800558
559 private boolean isToken(String image) {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700560 return isToken(token, image);
561 }
562
563 private static boolean isToken(Token token, String image) {
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800564 return token.image.equalsIgnoreCase(image);
565 }
566
567 private void expectToken(String image) throws SqlppParseException {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700568 expectToken(token, image);
569 }
570
571 private static void expectToken(Token token, String image) throws SqlppParseException {
572 if (!isToken(token, image)) {
573 throw createUnexpectedTokenError(token);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800574 }
575 }
576
577 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700578 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700579 }
580
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700581 private static SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700582 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800583 }
584
585 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700586 return createUnexpectedTokenError(token, expected);
587 }
588
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700589 private static SqlppParseException createUnexpectedTokenError(Token t, String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700590 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800591 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700592 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800593 }
594
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700595 private boolean laToken(int idx, int kind) {
596 Token t = getToken(idx);
597 return t.kind == kind;
598 }
599
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800600 private boolean laToken(int idx, int kind, String image) {
601 Token t = getToken(idx);
602 return t.kind == kind && t.image.equalsIgnoreCase(image);
603 }
604
605 private boolean laIdentifier(int idx, String image) {
606 return laToken(idx, IDENTIFIER, image);
607 }
608
609 private boolean laIdentifier(String image) {
610 return laIdentifier(1, image);
611 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700612
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700613 private Token fetchHint(Token token, SqlppHint... expectedHints) {
614 Token hintToken = token.specialToken;
615 if (hintToken == null) {
616 return null;
617 }
618 SourceLocation sourceLoc = getSourceLocation(hintToken);
619 hintCollector.remove(sourceLoc);
620 if (hintToken.hint == null) {
621 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
622 return null;
623 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
624 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
625 return null;
626 } else {
627 return hintToken;
628 }
629 }
630
631 private void reportUnclaimedHints() {
632 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
633 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
634 }
635 }
636
637 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
638 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
639 }
640
641 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700642 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500643 warningCollector.warn(Warning.of(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
Ali Alsulimanaa118862019-09-10 20:55:45 -0700644 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700645 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700646
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800647 private IExpressionAnnotation parseExpressionAnnotation(Token hintToken) {
648 // placeholder for the annotation that should be returned if this hint's parameters cannot be parsed
649 IExpressionAnnotation onParseErrorReturn = null;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800650 try {
651 switch (hintToken.hint) {
652 case HASH_BROADCAST_JOIN_HINT:
653 return new BroadcastExpressionAnnotation(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
654 case INDEXED_NESTED_LOOP_JOIN_HINT:
655 if (hintToken.hintParams == null) {
656 return IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
657 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800658 // if parameter parsing fails then return hint annotation without parameters
659 onParseErrorReturn = IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800660 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
661 return IndexedNLJoinExpressionAnnotation.newInstance(indexNames);
662 }
663 case RANGE_HINT:
664 Expression rangeExpr = parseExpression(hintToken.hintParams);
665 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
666 return new RangeAnnotation(rangeMap);
667 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
668 if (hintToken.hintParams == null) {
669 return SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE_ANY_INDEX;
670 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800671 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800672 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
673 return SkipSecondaryIndexSearchExpressionAnnotation.newInstance(indexNames);
674 }
Tin Vu4f1090d2021-09-21 02:06:32 -0700675 case SPATIAL_JOIN_HINT:
676 List<Literal> hintValues = parseParenthesizedLiteralList(hintToken.hintParams);
677
678 // Handle exceptions
679 if (hintValues.size() != 6) {
680 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. 6 arguments are required.",
681 hintToken.hint.toString()));
682 }
683
684 for (int i = 0; i < 4; i++) {
685 Literal lit = hintValues.get(i);
686 if ((lit.getLiteralType() != Literal.Type.DOUBLE)
687 && (lit.getLiteralType() != Literal.Type.FLOAT)
688 && (lit.getLiteralType() != Literal.Type.LONG)
689 && (lit.getLiteralType() != Literal.Type.INTEGER)){
690 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. Numeric value is required for first 4 arguments.",
691 hintToken.hint.toString()));
692 }
693 }
694
695 for (int i = 4; i < 6; i++) {
696 Literal lit = hintValues.get(i);
697 if ((lit.getLiteralType() != Literal.Type.LONG)
698 && (lit.getLiteralType() != Literal.Type.INTEGER)) {
699 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. Long/int is required for last 2 arguments.",
700 hintToken.hint.toString()));
701 }
702 }
703
704 try {
705 double minX = ExpressionUtils.getDoubleValue(hintValues.get(0));
706 double minY = ExpressionUtils.getDoubleValue(hintValues.get(1));
707 double maxX = ExpressionUtils.getDoubleValue(hintValues.get(2));
708 double maxY = ExpressionUtils.getDoubleValue(hintValues.get(3));
709 int numRows = (int) ExpressionUtils.getLongValue(hintValues.get(4));
710 int numColumns = (int) ExpressionUtils.getLongValue(hintValues.get(5));
711 SpatialJoinAnnotation spatialJoinAnn = new SpatialJoinAnnotation(minX, minY, maxX, maxY, numRows, numColumns);
712 return spatialJoinAnn;
713 } catch (TypeMismatchException e) {
714 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
715 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800716 case USE_SECONDARY_INDEX_SEARCH_HINT:
717 if (hintToken.hintParams == null) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800718 throw new SqlppParseException(getSourceLocation(hintToken), "Expected index name(s)");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800719 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800720 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800721 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
722 return SecondaryIndexSearchPreferenceAnnotation.newInstance(indexNames);
723 }
724 default:
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800725 throw new SqlppParseException(getSourceLocation(hintToken), "Unexpected hint");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800726 }
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800727 } catch (SqlppParseException e) {
728 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500729 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800730 hintToken.hint.toString(), e.getMessage()));
731 }
732 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800733 } catch (CompilationException e) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800734 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500735 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800736 hintToken.hint.toString(), e.getMessage()));
737 }
738 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800739 }
740 }
741
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700742 private void ensureNoTypeDeclsInFunction(String fnName, List<Pair<VarIdentifier, TypeExpression>> paramList,
743 TypeExpression returnType, Token startToken) throws SqlppParseException {
744 for (Pair<VarIdentifier, TypeExpression> p : paramList) {
745 if (p.second != null) {
746 String paramName = SqlppVariableUtil.toUserDefinedName(p.first.getValue());
747 throw new SqlppParseException(getSourceLocation(startToken),
748 "Unexpected type declaration for parameter " + paramName + " in function " + fnName);
749 }
750 }
751 if (returnType != null) {
752 throw new SqlppParseException(getSourceLocation(startToken),
753 "Unexpected return type declaration for function " + fnName);
754 }
755 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -0700756
757 private void ensureIntegerLiteral(LiteralExpr expr, String errorPrefix) throws SqlppParseException {
758 Literal lit = expr.getValue();
759 if (lit.getLiteralType() != Literal.Type.INTEGER && lit.getLiteralType() != Literal.Type.LONG) {
760 throw new SqlppParseException(expr.getSourceLocation(), errorPrefix + " should be an INTEGER");
761 }
762 }
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700763
764 private DataverseName createDataverseName(List<String> parts, int fromIndex, int toIndex, Token startToken)
765 throws SqlppParseException {
766 try {
767 return DataverseName.create(parts, fromIndex, toIndex);
768 } catch (AsterixException e) {
769 SqlppParseException pe = new SqlppParseException(getSourceLocation(startToken), e.getMessage());
770 pe.initCause(e);
771 throw pe;
772 }
773 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700774}
775
776PARSER_END(SQLPPParser)
777
Yingyi Bu391f09e2015-10-29 13:49:39 -0700778List<Statement> Statement() throws ParseException:
779{
780 scopeStack.push(RootScopeFactory.createRootScope(this));
781 List<Statement> decls = new ArrayList<Statement>();
782 Statement stmt = null;
783}
784{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300785 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700786 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300787 {
788 decls.add(stmt);
789 }
790 )?
791 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700792 )*
793 <EOF>
794 {
795 return decls;
796 }
797}
798
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700799Statement ExplainStatement() throws ParseException:
800{
801 Statement stmt = null;
802 Token explainToken = null;
803}
804{
805 ( <EXPLAIN> { explainToken = token; } )?
806 stmt = SingleStatement()
807 {
808 if (explainToken != null) {
809 if (stmt.getKind() == Statement.Kind.QUERY) {
810 ((Query)stmt).setExplain(true);
811 } else {
812 throw new SqlppParseException(getSourceLocation(explainToken),
813 "EXPLAIN is not supported for this kind of statement");
814 }
815 }
816 return stmt;
817 }
818}
819
Yingyi Bu391f09e2015-10-29 13:49:39 -0700820Statement SingleStatement() throws ParseException:
821{
822 Statement stmt = null;
823}
824{
825 (
826 stmt = DataverseDeclaration()
827 | stmt = FunctionDeclaration()
828 | stmt = CreateStatement()
829 | stmt = LoadStatement()
830 | stmt = DropStatement()
831 | stmt = WriteStatement()
832 | stmt = SetStatement()
833 | stmt = InsertStatement()
834 | stmt = DeleteStatement()
835 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800836 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700837 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700838 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700839 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700840 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700841 )
842 {
843 return stmt;
844 }
845}
846
847DataverseDecl DataverseDeclaration() throws ParseException:
848{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700849 Token startToken = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700850 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700851}
852{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700853 <USE> { startToken = token; } dvName = DataverseName()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700854 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700855 defaultDataverse = dvName;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800856 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700857 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700858 }
859}
860
861Statement CreateStatement() throws ParseException:
862{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700863 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700864 Statement stmt = null;
865}
866{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700867 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700868 (
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700869 stmt = CreateOrReplaceStatement(startToken)
870 | stmt = CreateTypeStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800871 | stmt = CreateNodegroupStatement(startToken)
872 | stmt = CreateDatasetStatement(startToken)
873 | stmt = CreateIndexStatement(startToken)
874 | stmt = CreateDataverseStatement(startToken)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700875 | stmt = CreateFunctionStatement(startToken, false)
Ian Maxon38fe9402020-01-29 19:27:40 -0800876 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800877 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800878 | stmt = CreateFeedStatement(startToken)
879 | stmt = CreateFeedPolicyStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -0800880 | stmt = CreateFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700881 | stmt = CreateViewStatement(startToken, false)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700882 )
883 {
884 return stmt;
885 }
886}
887
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700888Statement CreateOrReplaceStatement(Token startStmtToken) throws ParseException:
889{
890 Statement stmt = null;
891 Token replaceToken = null;
892}
893{
894 <OR> <IDENTIFIER> { replaceToken = token; }
895 (
896 stmt = CreateFunctionStatement(startStmtToken, true)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700897 | stmt = CreateViewStatement(startStmtToken, true)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700898 )
899 {
900 // check expected token here to make the grammar extension plugin happy
901 expectToken(replaceToken, REPLACE);
902 return stmt;
903 }
904}
905
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800906TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
907{
908 TypeDecl stmt = null;
909}
910{
911 <TYPE> stmt = TypeSpecification(startStmtToken)
912 {
913 return stmt;
914 }
915}
916
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700917TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700918{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800919 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700920 boolean ifNotExists = false;
921 TypeExpression typeExpr = null;
922}
923{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800924 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700925 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800926 {
927 boolean dgen = false;
928 long numValues = -1;
929 String filename = null;
930 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
931 if (hintToken != null) {
932 String hintParams = hintToken.hintParams;
933 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
934 if (splits == null || splits.length != 2) {
935 throw new SqlppParseException(getSourceLocation(hintToken),
936 "Expecting /*+ dgen <filename> <numberOfItems> */");
937 }
938 dgen = true;
939 filename = splits[0];
940 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700941 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800942 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
943 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
944 return addSourceLocation(stmt, startStmtToken);
945 }
946}
947
948NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
949{
950 NodegroupDecl stmt = null;
951}
952{
953 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
954 {
955 return stmt;
956 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700957}
958
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700959NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700960{
961 String name = null;
962 String tmp = null;
963 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800964 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700965}
966{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800967 name = Identifier() ifNotExists = IfNotExists()
968 <ON> tmp = Identifier()
969 {
970 ncNames.add(new Identifier(tmp));
971 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700972 ( <COMMA> tmp = Identifier()
973 {
974 ncNames.add(new Identifier(tmp));
975 }
976 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800977 {
978 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
979 return addSourceLocation(stmt, startStmtToken);
980 }
981}
982
983void Dataset() throws ParseException:
984{
985}
986{
987 (<DATASET>|<COLLECTION>)
988}
989
990DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
991{
992 DatasetDecl stmt = null;
993}
994{
995 (
996 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
997 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
998 )
999 {
1000 return stmt;
1001 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001002}
1003
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001004DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001005{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001006 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001007 boolean ifNotExists = false;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001008 TypeExpression typeExpr = null;
1009 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001010 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001011 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001012 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001013 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001014 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -08001015 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001016}
1017{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001018 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001019 typeExpr = DatasetTypeSpecification()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001020 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001021 { String name; }
1022 <WITH>
1023 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001024 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001025 if (!name.equalsIgnoreCase("meta")){
1026 throw new SqlppParseException(getSourceLocation(startStmtToken),
1027 "We can only support one additional associated field called \"meta\".");
1028 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001029 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001030 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001031 )?
1032 ifNotExists = IfNotExists()
1033 primaryKeyFields = PrimaryKey()
1034 (<AUTOGENERATED> { autogenerated = true; } )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001035 ( <HINTS> hints = Properties() )?
1036 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
1037 ( <WITH> withRecord = RecordConstructor() )?
1038 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001039 try {
Xikui Wangca7927f2020-09-03 11:27:57 -07001040 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
1041 filterField == null? null : filterField.first, filterField == null? null : filterField.second);
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -08001042 DatasetDeclParametersUtil.adjustInlineTypeDecl(typeExpr, primaryKeyFields.second, primaryKeyFields.first, false);
1043 if (metaTypeExpr != null) {
1044 DatasetDeclParametersUtil.adjustInlineTypeDecl(metaTypeExpr, primaryKeyFields.second, primaryKeyFields.first,
1045 true);
1046 }
Murtadha Hubail353e95f2020-08-24 22:18:04 +03001047 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr, hints,
1048 DatasetType.INTERNAL, idd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001049 return addSourceLocation(stmt, startStmtToken);
1050 } catch (CompilationException e) {
1051 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1052 }
1053 }
1054}
1055
1056DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
1057{
1058 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001059 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001060 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001061 String adapterName = null;
1062 Map<String,String> properties = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001063 Map<String,String> hints = new HashMap<String,String>();
1064 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001065 RecordConstructor withRecord = null;
1066}
1067{
1068 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001069 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001070 ifNotExists = IfNotExists()
1071 <USING> adapterName = AdapterName() properties = Configuration()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001072 ( <HINTS> hints = Properties() )?
1073 ( <WITH> withRecord = RecordConstructor() )?
1074 {
1075 ExternalDetailsDecl edd = new ExternalDetailsDecl();
1076 edd.setAdapter(adapterName);
1077 edd.setProperties(properties);
1078 try {
Murtadha Hubail353e95f2020-08-24 22:18:04 +03001079 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null, hints, DatasetType.EXTERNAL,
1080 edd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001081 return addSourceLocation(stmt, startStmtToken);
1082 } catch (CompilationException e) {
1083 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1084 }
1085 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001086}
1087
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001088TypeExpression DatasetTypeSpecification() throws ParseException:
1089{
1090 TypeExpression typeExpr = null;
1091}
1092{
1093 (
1094 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
1095 | typeExpr = DatasetReferenceTypeSpecification()
1096 )
1097 {
1098 return typeExpr;
1099 }
1100}
1101
1102TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
1103{
1104 TypeExpression typeExpr = null;
1105}
1106{
1107 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
1108 {
1109 return typeExpr;
1110 }
1111}
1112
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001113RecordTypeDefinition DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001114{
1115 RecordTypeDefinition recordTypeDef = null;
1116 RecordTypeDefinition.RecordKind recordKind = null;
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001117 Token startToken = null, recordKindToken = null;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001118}
1119{
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001120 <LEFTPAREN> { startToken = token; } recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001121 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
1122 {
1123 if (recordKind == null) {
1124 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
1125 } else if (!allowRecordKindModifier) {
1126 throw createUnexpectedTokenError(recordKindToken);
1127 }
1128 recordTypeDef.setRecordKind(recordKind);
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001129 return addSourceLocation(recordTypeDef, startToken);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001130 }
1131}
1132
1133RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
1134{
1135 RecordTypeDefinition recType = new RecordTypeDefinition();
1136}
1137{
1138 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
1139 {
1140 return recType;
1141 }
1142}
1143
1144void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
1145{
1146 String fieldName;
1147 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001148 boolean nullable = true, missable = true;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001149}
1150{
1151 fieldName = Identifier()
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001152 type = TypeReference() ( <NOT> <UNKNOWN> { nullable = false; missable = false; } )?
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001153 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001154 recType.addField(fieldName, type, nullable, missable);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001155 }
1156}
1157
Yingyi Bu391f09e2015-10-29 13:49:39 -07001158RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
1159{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001160 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001161 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001162 String datasetName = null;
1163}
1164{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001165 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
1166 {
1167 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
1168 stmt.setDataverseName(nameComponents.first);
1169 stmt.setDatasetName(nameComponents.second);
1170 return addSourceLocation(stmt, startToken);
1171 }
1172}
1173
1174CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
1175{
1176 CreateIndexStatement stmt = null;
1177}
1178{
1179 (
1180 <INDEX> stmt = IndexSpecification(startStmtToken)
1181 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
1182 )
1183 {
1184 return stmt;
1185 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001186}
1187
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001188CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001189{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001190 Pair<DataverseName,Identifier> nameComponents = null;
Glenn67fd1f32021-02-25 16:04:49 -08001191 String indexName = null;
1192 IndexParams indexParams = null;
1193 CreateIndexStatement.IndexedElement indexedElement = null;
1194 List<CreateIndexStatement.IndexedElement> indexedElementList = new ArrayList<CreateIndexStatement.IndexedElement>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001195 boolean enforced = false;
Glenn67fd1f32021-02-25 16:04:49 -08001196 boolean ifNotExists = false;
1197 boolean hasUnnest = false;
1198 String fullTextConfigName = null;
1199 Token startElementToken = null;
Ali Alsuliman931e7382021-08-08 21:55:59 +03001200 Boolean excludeUnknown = null;
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001201 Boolean castDefaultNull = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001202}
1203{
Ali Alsuliman8351d252017-09-24 00:43:15 -07001204 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001205 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -07001206 <ON> nameComponents = QualifiedName()
Glenn67fd1f32021-02-25 16:04:49 -08001207 <LEFTPAREN> { startElementToken = token; }
1208 indexedElement = IndexedElement(startElementToken) {
1209 indexedElementList.add(indexedElement);
1210 hasUnnest |= indexedElement.hasUnnest();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001211 }
Glenn67fd1f32021-02-25 16:04:49 -08001212 (<COMMA> { startElementToken = token; }
1213 indexedElement = IndexedElement(startElementToken) {
1214 indexedElementList.add(indexedElement);
1215 hasUnnest |= indexedElement.hasUnnest();
1216 }
1217 )*
1218 <RIGHTPAREN>
1219 ( <TYPE> indexParams = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001220 ( LOOKAHEAD({laIdentifier(EXCLUDE) || laIdentifier(INCLUDE)}) <IDENTIFIER>
1221 {
1222 if (isToken(EXCLUDE)) {
1223 excludeUnknown = true;
1224 } else if (isToken(INCLUDE)) {
1225 excludeUnknown = false;
1226 } else {
1227 throw createUnexpectedTokenError();
1228 }
1229 } <UNKNOWN> <KEY>
Ali Alsuliman931e7382021-08-08 21:55:59 +03001230 )?
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001231
1232 ( <CAST><LEFTPAREN><IDENTIFIER> { expectToken(DEFAULT); } <NULL><RIGHTPAREN> { castDefaultNull = true; })?
Ali Alsuliman8351d252017-09-24 00:43:15 -07001233 )
1234 {
Glenn67fd1f32021-02-25 16:04:49 -08001235 IndexType indexType;
1236 int gramLength;
1237 if (indexParams != null) {
1238 indexType = indexParams.type;
1239 gramLength = indexParams.gramLength;
1240 fullTextConfigName = indexParams.fullTextConfig;
1241 } else {
1242 indexType = hasUnnest ? IndexType.ARRAY : IndexType.BTREE;
1243 gramLength = -1;
1244 fullTextConfigName = null;
Ali Alsuliman8351d252017-09-24 00:43:15 -07001245 }
Glenn67fd1f32021-02-25 16:04:49 -08001246 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman931e7382021-08-08 21:55:59 +03001247 new Identifier(indexName), indexType, indexedElementList, enforced, gramLength, fullTextConfigName, ifNotExists,
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001248 excludeUnknown, castDefaultNull);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001249 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001250 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001251}
1252
Glenn67fd1f32021-02-25 16:04:49 -08001253CreateIndexStatement.IndexedElement IndexedElement(Token startElementToken) throws ParseException:
1254{
1255 Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1256 Pair<List<String>, IndexedTypeExpression> elementSimple = null;
1257 int elementSimpleSource = 0;
1258}
1259{
1260 (
1261 element = IndexedElementUnnestSelect()
1262 | (
1263 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1264 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1265 <DOT> elementSimple = IndexedField()
1266 { elementSimpleSource = 1; }
1267 )
1268 | elementSimple = IndexedField()
1269 | <LEFTPAREN> ( element = IndexedElementUnnestSelect() | elementSimple = IndexedField() ) <RIGHTPAREN>
1270 )
1271 {
1272 int source;
1273 List<List<String>> unnestList;
1274 List<Pair<List<String>, IndexedTypeExpression>> projectList;
1275 if (elementSimple != null) {
1276 source = elementSimpleSource;
1277 unnestList = null;
1278 projectList = Collections.singletonList(elementSimple);
1279 } else {
1280 source = element.first;
1281 unnestList = element.second;
1282 projectList = element.third;
1283 }
1284 CreateIndexStatement.IndexedElement ie = new CreateIndexStatement.IndexedElement(source, unnestList, projectList);
1285 ie.setSourceLocation(getSourceLocation(startElementToken));
1286 return ie;
1287 }
1288}
1289
1290Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelect()
1291 throws ParseException:
1292{
1293 int source = 0;
1294 Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1295}
1296{
1297 <UNNEST>
1298 (
1299 (
1300 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1301 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1302 <DOT> element = IndexedElementUnnestSelectBody() { source = 1; }
1303 ) | element = IndexedElementUnnestSelectBody()
1304 )
1305 {
1306 return new Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(
1307 source, element.first, element.second
1308 );
1309 }
1310}
1311
1312Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelectBody()
1313 throws ParseException:
1314{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001315 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001316 IndexedTypeExpression type = null;
1317 List<List<String>> unnestList = new ArrayList();
Glenn67fd1f32021-02-25 16:04:49 -08001318 List<Pair<List<String>, IndexedTypeExpression>> projectList = new ArrayList();
1319}
1320{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001321 path = MultipartIdentifier() { unnestList.add(path.first); }
1322 ( <UNNEST> path = MultipartIdentifier() { unnestList.add(path.first); })*
Glenn67fd1f32021-02-25 16:04:49 -08001323 (
Glenn58329202021-04-09 12:03:46 -07001324 ( <COLON> type = IndexedTypeExpr(false)
Glenn67fd1f32021-02-25 16:04:49 -08001325 {
1326 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, type));
1327 }
1328 ) |
1329 (
Glenn58329202021-04-09 12:03:46 -07001330 <SELECT> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1331 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001332 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001333 }
1334 ( <COMMA> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1335 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001336 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001337 }
1338 )*
Glenn67fd1f32021-02-25 16:04:49 -08001339 )
1340 )?
1341 {
1342 if (projectList.isEmpty()) {
1343 // To support the case (<UNNEST> IDENTIFIER)* IDENTIFIER w/o any type specification.
1344 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, null));
1345 }
1346
1347 return new Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(unnestList, projectList);
1348 }
1349}
1350
1351Pair<List<String>, IndexedTypeExpression> IndexedField() throws ParseException:
1352{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001353 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001354 IndexedTypeExpression type = null;
1355}
1356{
Glenn58329202021-04-09 12:03:46 -07001357 path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(true) )?
Glenn67fd1f32021-02-25 16:04:49 -08001358 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001359 return new Pair<List<String>, IndexedTypeExpression>(path.first, type);
Glenn67fd1f32021-02-25 16:04:49 -08001360 }
1361}
1362
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001363CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
1364{
Glenn67fd1f32021-02-25 16:04:49 -08001365 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001366 String indexName = null;
1367 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001368}
1369{
1370 (indexName = Identifier())? ifNotExists = IfNotExists()
1371 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
1372 {
1373 if (indexName == null) {
1374 indexName = "primary_idx_" + nameComponents.second;
1375 }
Glenn67fd1f32021-02-25 16:04:49 -08001376 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001377 new Identifier(indexName), IndexType.BTREE, Collections.emptyList(), false, -1, null, ifNotExists, null, null);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001378 return addSourceLocation(stmt, startStmtToken);
1379 }
1380}
1381
Yingyi Bu391f09e2015-10-29 13:49:39 -07001382String FilterField() throws ParseException :
1383{
1384 String filterField = null;
1385}
1386{
1387 filterField = Identifier()
1388 {
1389 return filterField;
1390 }
1391}
1392
1393IndexParams IndexType() throws ParseException:
1394{
1395 IndexType type = null;
1396 int gramLength = 0;
Rui Guoe6986dd2020-11-25 19:50:06 -08001397 String fullTextConfig = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001398}
1399{
1400 (<BTREE>
1401 {
1402 type = IndexType.BTREE;
1403 }
1404 | <RTREE>
1405 {
1406 type = IndexType.RTREE;
1407 }
1408 | <KEYWORD>
1409 {
1410 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1411 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001412 | <FULLTEXT>
Taewoo Kimc49405a2017-01-04 00:30:43 -08001413 {
1414 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1415 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001416 // For now we don't allow inverted index creation using a full-text config in another data verse.
1417 // We may want to support corss-dataverse full-text config access later
1418 // If so, replace the Identifier() with QualifiedName() to get the dataverse name
1419 ( <USING> Identifier()
1420 {
1421 fullTextConfig = token.image;
1422 }
1423 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001424 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1425 {
1426 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1427 gramLength = Integer.valueOf(token.image);
1428 }
1429 <RIGHTPAREN>)
1430 {
Rui Guoe6986dd2020-11-25 19:50:06 -08001431 return new IndexParams(type, gramLength, fullTextConfig);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001432 }
1433}
1434
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001435CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1436{
1437 CreateDataverseStatement stmt = null;
1438}
1439{
1440 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1441 {
1442 return stmt;
1443 }
1444}
1445
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001446CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001447{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001448 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001449 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001450}
1451{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001452 dvName = DataverseName() ifNotExists = IfNotExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001453 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001454 CreateDataverseStatement stmt = new CreateDataverseStatement(dvName, null, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001455 return addSourceLocation(stmt, startStmtToken);
1456 }
1457}
1458
Ian Maxon38fe9402020-01-29 19:27:40 -08001459CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1460{
1461 CreateAdapterStatement stmt = null;
1462}
1463{
1464 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1465 {
1466 return stmt;
1467 }
1468}
1469
1470CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1471{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001472 Pair<DataverseName,Identifier> adapterName = null;
1473 Pair<DataverseName,Identifier> libraryName = null;
1474 List<String> externalIdentifier = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001475 boolean ifNotExists = false;
1476}
1477{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001478 adapterName = QualifiedName()
1479 ifNotExists = IfNotExists()
1480 <AS> externalIdentifier = FunctionExternalIdentifier()
1481 <AT> libraryName = QualifiedName()
1482 {
1483 CreateAdapterStatement stmt = new CreateAdapterStatement(adapterName.first, adapterName.second.getValue(),
1484 libraryName.first, libraryName.second.getValue(), externalIdentifier, ifNotExists);
1485 return addSourceLocation(stmt, startStmtToken);
1486 }
1487}
Ian Maxon38fe9402020-01-29 19:27:40 -08001488
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001489CreateViewStatement CreateViewStatement(Token startStmtToken, boolean orReplace) throws ParseException:
1490{
1491 CreateViewStatement stmt = null;
1492}
1493{
1494 <VIEW> stmt = ViewSpecification(startStmtToken, orReplace)
1495 {
1496 return stmt;
1497 }
1498}
1499
1500CreateViewStatement ViewSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
1501{
1502 Pair<DataverseName, Identifier> nameComponents = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001503 TypeExpression typeExpr = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001504 boolean ifNotExists = false;
1505 Token beginPos = null, endPos = null;
1506 Expression viewBodyExpr = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001507 Boolean defaultNull = null;
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001508 Map<String, String> viewConfig = null;
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001509 String propertyName = null, propertyValue = null;
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001510 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001511 Pair<List<Integer>, List<List<String>>> foreignKeyFields = null;
1512 Pair<DataverseName, Identifier> refNameComponents = null;
1513 List<CreateViewStatement.ForeignKeyDecl> foreignKeyDecls = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001514 DataverseName currentDataverse = defaultDataverse;
1515}
1516{
1517 nameComponents = QualifiedName()
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001518 (
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001519 (
1520 typeExpr = DatasetTypeSpecification()
1521 ifNotExists = IfNotExists()
1522 <IDENTIFIER> { expectToken(DEFAULT); } <NULL> { defaultNull = true; }
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001523 (
1524 LOOKAHEAD(2) <IDENTIFIER> { propertyName = token.image.toLowerCase(); } propertyValue = StringLiteral()
1525 {
1526 if (viewConfig == null) {
1527 viewConfig = new HashMap<String, String>();
1528 }
1529 viewConfig.put(propertyName, propertyValue);
1530 }
1531 )*
1532 (
1533 <PRIMARY> <KEY> <LEFTPAREN> primaryKeyFields = PrimaryKeyFields() <RIGHTPAREN>
1534 <NOT> <ENFORCED>
1535 )?
1536 (
1537 <IDENTIFIER> { expectToken(FOREIGN); } <KEY> <LEFTPAREN> foreignKeyFields = PrimaryKeyFields() <RIGHTPAREN>
1538 <IDENTIFIER> { expectToken(REFERENCES); } refNameComponents = QualifiedName()
1539 <NOT> <ENFORCED>
1540 {
1541 if (foreignKeyDecls == null) {
1542 foreignKeyDecls = new ArrayList<CreateViewStatement.ForeignKeyDecl>();
1543 }
1544 foreignKeyDecls.add(new CreateViewStatement.ForeignKeyDecl(foreignKeyFields.second,
1545 foreignKeyFields.first, refNameComponents.first, refNameComponents.second));
1546 }
1547 )*
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001548 )
1549 |
1550 ( ifNotExists = IfNotExists() )
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001551 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001552 {
1553 if (orReplace && ifNotExists) {
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001554 throw new SqlppParseException(getSourceLocation(startStmtToken), "Unexpected IF NOT EXISTS");
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001555 }
1556 }
1557 <AS>
1558 {
1559 beginPos = token;
1560 createNewScope();
1561 if (nameComponents.first != null) {
1562 defaultDataverse = nameComponents.first;
1563 }
1564 }
1565 viewBodyExpr = ViewBody()
1566 {
1567 endPos = token;
1568 String viewBody = extractFragment(beginPos.beginLine, beginPos.beginColumn + 1, endPos.endLine,
1569 endPos.endColumn + 1);
1570 removeCurrentScope();
1571 defaultDataverse = currentDataverse;
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -08001572 if (typeExpr != null && primaryKeyFields != null) {
1573 DatasetDeclParametersUtil.adjustInlineTypeDecl(typeExpr, primaryKeyFields.second, primaryKeyFields.first, false);
1574 }
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001575 CreateViewStatement.KeyDecl primaryKeyDecl = primaryKeyFields != null ?
1576 new CreateViewStatement.KeyDecl(primaryKeyFields.second, primaryKeyFields.first) : null;
1577 CreateViewStatement stmt = new CreateViewStatement(nameComponents.first, nameComponents.second.getValue(),
1578 typeExpr, viewBody, viewBodyExpr, defaultNull, viewConfig, primaryKeyDecl, foreignKeyDecls, orReplace,
1579 ifNotExists);
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001580 return addSourceLocation(stmt, startStmtToken);
1581 }
1582}
1583
1584Expression ViewBody() throws ParseException:
1585{
1586 Expression viewBodyExpr = null;
1587}
1588{
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001589 (
1590 ( viewBodyExpr = VariableRef() ( viewBodyExpr = FieldAccessor(viewBodyExpr) )* )
1591 | viewBodyExpr = SelectExpression(true)
1592 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001593 {
1594 return viewBodyExpr;
1595 }
1596}
1597
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001598CreateFunctionStatement CreateFunctionStatement(Token startStmtToken, boolean orReplace) throws ParseException:
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001599{
1600 CreateFunctionStatement stmt = null;
1601}
1602{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001603 <FUNCTION> stmt = FunctionSpecification(startStmtToken, orReplace)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001604 {
1605 return stmt;
1606 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001607}
1608
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001609CreateFunctionStatement FunctionSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001610{
Ian Maxon38fe9402020-01-29 19:27:40 -08001611 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001612 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001613 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001614 List<Pair<VarIdentifier,TypeExpression>> params = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001615 int arity = 0;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001616 TypeExpression returnType = null;
1617 Token beginPos = null, endPos = null;
1618 Expression functionBodyExpr = null;
1619 Pair<DataverseName,Identifier> libraryName = null;
1620 List<String> externalIdentifier = null;
1621 RecordConstructor withOptions = null;
1622 boolean ifNotExists = false;
1623 CreateFunctionStatement stmt = null;
1624 DataverseName currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001625}
1626{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001627 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001628 {
1629 defaultDataverse = fctName.dataverse;
1630 }
Dmitry Lychagin541652082020-11-09 14:04:53 -08001631 paramsWithArity = FunctionParameters()
1632 {
1633 arity = paramsWithArity.first;
1634 params = paramsWithArity.second;
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001635 signature = new FunctionSignature(fctName.dataverse, fctName.function, arity);
Dmitry Lychagin541652082020-11-09 14:04:53 -08001636 }
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001637 ifNotExists = IfNotExists()
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001638 {
1639 if (orReplace && ifNotExists) {
1640 throw new SqlppParseException(getSourceLocation(token), "Unexpected IF NOT EXISTS");
1641 }
1642 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001643 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001644 (
1645 (
1646 <LEFTBRACE>
1647 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001648 beginPos = token;
1649 createNewScope();
Ian Maxon38fe9402020-01-29 19:27:40 -08001650 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001651 functionBodyExpr = FunctionBody()
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001652 <RIGHTBRACE>
1653 {
1654 endPos = token;
1655 String functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine,
1656 endPos.beginColumn);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001657 getCurrentScope().addFunctionDescriptor(signature, false);
1658 removeCurrentScope();
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001659 ensureNoTypeDeclsInFunction(fctName.function, params, returnType, startStmtToken);
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001660 stmt = new CreateFunctionStatement(signature, params, functionBody, functionBodyExpr, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001661 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001662 )
1663 |
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001664 (
1665 <AS> externalIdentifier = FunctionExternalIdentifier()
1666 <AT> libraryName = QualifiedName()
1667 (<WITH> withOptions = RecordConstructor())?
1668 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001669 try {
1670 stmt = new CreateFunctionStatement(signature, params, returnType, libraryName.first,
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001671 libraryName.second.getValue(), externalIdentifier, withOptions, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001672 } catch (AlgebricksException e) {
1673 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1674 }
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001675 }
1676 )
Ian Maxon38fe9402020-01-29 19:27:40 -08001677 )
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001678 {
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001679 defaultDataverse = currentDataverse;
1680 return addSourceLocation(stmt, startStmtToken);
1681 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001682}
1683
Dmitry Lychagin541652082020-11-09 14:04:53 -08001684Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameters() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001685{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001686 Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> params = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001687}
1688{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001689 <LEFTPAREN> (params = FunctionParameterList())? <RIGHTPAREN>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001690 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001691 if (params == null) {
1692 params = new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(
1693 0, Collections.<Pair<VarIdentifier, TypeExpression>>emptyList()
1694 );
1695 }
1696 return params;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001697 }
1698}
1699
Dmitry Lychagin541652082020-11-09 14:04:53 -08001700Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameterList() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001701{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001702 List<Pair<VarIdentifier, TypeExpression>> paramList = null;
1703 Pair<VarIdentifier, TypeExpression> param = null;
1704 int arity = 0;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001705}
1706{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001707 (
1708 (
1709 <DOT> <DOT> <DOT>
1710 {
1711 param = new Pair<VarIdentifier, TypeExpression>(
1712 SqlppVariableUtil.toInternalVariableIdentifier(UDF_VARARGS_PARAM_NAME), null
1713 );
1714 paramList = Collections.<Pair<VarIdentifier, TypeExpression>>singletonList(param);
1715 arity = FunctionIdentifier.VARARGS;
1716 }
1717 )
1718 |
1719 (
1720 param = FunctionParameter()
1721 {
1722 paramList = new ArrayList<Pair<VarIdentifier, TypeExpression>>();
1723 paramList.add(param);
1724 }
1725 ( <COMMA> param = FunctionParameter() { paramList.add(param); } )*
1726 {
1727 arity = paramList.size();
1728 }
1729 )
1730 )
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001731 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001732 return new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(arity, paramList);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001733 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001734}
1735
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001736Pair<VarIdentifier,TypeExpression> FunctionParameter() throws ParseException:
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001737{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001738 String name = null;
1739 TypeExpression type = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001740}
1741{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001742 name = VariableIdentifier()
1743 ( LOOKAHEAD(3) (<COLON>)? type = TypeExpr(false) )?
1744 {
1745 return new Pair<VarIdentifier,TypeExpression>(new VarIdentifier(name), type);
1746 }
1747}
1748
1749TypeExpression FunctionReturnType() throws ParseException:
1750{
1751 TypeExpression returnType = null;
1752}
1753{
1754 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = TypeExpr(false) )?
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001755 {
1756 return returnType;
1757 }
1758}
1759
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001760Expression FunctionBody() throws ParseException:
1761{
1762 Expression functionBodyExpr = null;
1763}
1764{
1765 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1766 {
1767 return functionBodyExpr;
1768 }
1769}
1770
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001771List<String> FunctionExternalIdentifier() throws ParseException:
1772{
1773 String ident = null;
1774 List<String> identList = new ArrayList(2);
1775}
1776{
1777 ident = StringLiteral() { identList.add(ident.trim()); }
1778 ( <COMMA> ident = StringLiteral() { identList.add(ident.trim()); } )*
1779 {
1780 return identList;
1781 }
1782}
1783
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001784CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1785{
1786 CreateFeedStatement stmt = null;
1787}
1788{
1789 <FEED> stmt = FeedSpecification(startStmtToken)
1790 {
1791 return stmt;
1792 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001793}
1794
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001795CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001796{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001797 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001798 boolean ifNotExists = false;
1799 String adapterName = null;
1800 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001801 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001802 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001803 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001804}
1805{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001806 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001807 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001808 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001809 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001810 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001811 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001812 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001813 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001814 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001815 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001816}
1817
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001818CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1819{
1820 CreateFeedPolicyStatement stmt = null;
1821}
1822{
1823 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1824 {
1825 return stmt;
1826 }
1827}
1828
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001829CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001830{
Michael Blowd6cf6412016-06-30 02:44:35 -04001831 String policyName = null;
1832 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001833 String sourcePolicyFile = null;
1834 String definition = null;
1835 boolean ifNotExists = false;
1836 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001837 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001838}
1839{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001840 policyName = Identifier() ifNotExists = IfNotExists()
1841 <FROM>
1842 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001843 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001844 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001845 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001846 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1847 {
1848 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1849 }
1850 )
1851 {
1852 return addSourceLocation(stmt, startStmtToken);
1853 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001854}
1855
Rui Guoe6986dd2020-11-25 19:50:06 -08001856Statement CreateFullTextStatement(Token startStmtToken) throws ParseException:
1857{
1858 Statement stmt = null;
1859}
1860{
1861 (
1862 <FULLTEXT>
1863 (
1864 <FILTER> stmt = CreateFullTextFilterSpec(startStmtToken)
1865 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = CreateFullTextConfigSpec(startStmtToken))
1866 )
1867 )
1868 {
1869 return stmt;
1870 }
1871}
1872
1873CreateFullTextFilterStatement CreateFullTextFilterSpec(Token startStmtToken) throws ParseException:
1874{
1875 CreateFullTextFilterStatement stmt = null;
1876 Pair<DataverseName,Identifier> nameComponents = null;
1877 boolean ifNotExists = false;
1878 RecordConstructor expr = null;
1879}
1880{
1881 (
1882 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1883 <AS>
1884 expr = RecordConstructor()
1885 )
1886 {
1887 try {
1888 stmt = new CreateFullTextFilterStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
1889 return addSourceLocation(stmt, startStmtToken);
1890 } catch (CompilationException e) {
1891 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1892 }
1893 }
1894}
1895
1896CreateFullTextConfigStatement CreateFullTextConfigSpec(Token startStmtToken) throws ParseException:
1897{
1898 CreateFullTextConfigStatement stmt = null;
1899 Pair<DataverseName,Identifier> nameComponents = null;
1900 boolean ifNotExists = false;
1901 RecordConstructor expr = null;
1902}
1903{
1904 (
1905 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1906 <AS>
1907 expr = RecordConstructor()
1908 )
1909 {
1910 try {
1911 stmt = new CreateFullTextConfigStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
1912 return addSourceLocation(stmt, startStmtToken);
1913 } catch (CompilationException e) {
1914 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1915 }
1916 }
1917}
1918
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001919CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
1920{
1921 CreateSynonymStatement stmt = null;
1922}
1923{
1924 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
1925 {
1926 return stmt;
1927 }
1928}
1929
1930CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
1931{
1932 Pair<DataverseName,Identifier> nameComponents = null;
1933 Pair<DataverseName,Identifier> objectNameComponents = null;
1934 boolean ifNotExists = false;
1935}
1936{
1937 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1938 <FOR> objectNameComponents = QualifiedName()
1939 {
1940 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
1941 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
1942 return addSourceLocation(stmt, startStmtToken);
1943 }
1944}
1945
Yingyi Bu391f09e2015-10-29 13:49:39 -07001946boolean IfNotExists() throws ParseException:
1947{
1948}
1949{
Yingyi Budaa549c2016-06-28 22:30:52 -07001950 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001951 {
1952 return true;
1953 }
1954 )?
1955 {
1956 return false;
1957 }
1958}
1959
Xikui Wang9d63f622017-05-18 17:50:44 -07001960void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001961{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001962 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07001963 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001964}
1965{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001966 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001967 {
1968 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1969 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
1970 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07001971 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001972 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07001973 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001974 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1975 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07001976 }
1977 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07001978}
1979
1980String GetPolicy() throws ParseException:
1981{
1982 String policy = null;
1983}
1984{
1985 <USING> <POLICY> policy = Identifier()
1986 {
1987 return policy;
1988 }
1989
1990}
1991
1992FunctionSignature FunctionSignature() throws ParseException:
1993{
1994 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001995 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> params = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001996 int arity = 0;
1997}
1998{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001999 fctName = FunctionName()
2000 (
Dmitry Lychagin541652082020-11-09 14:04:53 -08002001 LOOKAHEAD(2) params = FunctionParameters() { arity = params.first; }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002002 | ( <LEFTPAREN> arity = FunctionArity() <RIGHTPAREN> )
2003 | ( <ATT> arity = FunctionArity() ) // back-compat
2004 )
2005 {
2006 return new FunctionSignature(fctName.dataverse, fctName.function, arity);
2007 }
2008}
Yingyi Bu391f09e2015-10-29 13:49:39 -07002009
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002010int FunctionArity() throws ParseException:
2011{
2012 int arity;
2013}
2014{
2015 <INTEGER_LITERAL>
2016 {
2017 try {
2018 arity = Integer.parseInt(token.image);
2019 } catch (NumberFormatException e) {
2020 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002021 }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002022 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
2023 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + arity);
2024 }
2025 return arity;
2026 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002027}
2028
Yingyi Buc9bfe252016-03-01 00:02:40 -08002029Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002030{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07002031 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
2032}
2033{
2034 <PRIMARY> <KEY> primaryKeyFields = PrimaryKeyFields()
2035 {
2036 return primaryKeyFields;
2037 }
2038}
2039
2040Pair<List<Integer>, List<List<String>>> PrimaryKeyFields() throws ParseException:
2041{
Yingyi Buc9bfe252016-03-01 00:02:40 -08002042 Pair<Integer, List<String>> tmp = null;
2043 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002044 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
2045}
2046{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07002047 tmp = NestedField()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002048 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002049 keyFieldSourceIndicators.add(tmp.first);
2050 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002051 }
2052 ( <COMMA> tmp = NestedField()
2053 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002054 keyFieldSourceIndicators.add(tmp.first);
2055 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002056 }
2057 )*
2058 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002059 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002060 }
2061}
2062
2063Statement DropStatement() throws ParseException:
2064{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002065 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002066 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002067}
2068{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002069 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002070 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002071 stmt = DropDatasetStatement(startToken)
2072 | stmt = DropIndexStatement(startToken)
2073 | stmt = DropNodeGroupStatement(startToken)
2074 | stmt = DropTypeStatement(startToken)
2075 | stmt = DropDataverseStatement(startToken)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002076 | stmt = DropAdapterStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002077 | stmt = DropFunctionStatement(startToken)
2078 | stmt = DropFeedStatement(startToken)
2079 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002080 | stmt = DropSynonymStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -08002081 | stmt = DropFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002082 | stmt = DropViewStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002083 )
2084 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002085 return stmt;
2086 }
2087}
2088
2089DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
2090{
2091 DropDatasetStatement stmt = null;
2092}
2093{
2094 Dataset() stmt = DropDatasetSpecification(startStmtToken)
2095 {
2096 return stmt;
2097 }
2098}
2099
2100DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
2101{
2102 Pair<DataverseName,Identifier> pairId = null;
2103 boolean ifExists = false;
2104}
2105{
2106 pairId = QualifiedName() ifExists = IfExists()
2107 {
2108 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
2109 return addSourceLocation(stmt, startStmtToken);
2110 }
2111}
2112
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002113ViewDropStatement DropViewStatement(Token startStmtToken) throws ParseException:
2114{
2115 ViewDropStatement stmt = null;
2116}
2117{
2118 <VIEW> stmt = DropViewSpecification(startStmtToken)
2119 {
2120 return stmt;
2121 }
2122}
2123
2124ViewDropStatement DropViewSpecification(Token startStmtToken) throws ParseException:
2125{
2126 Pair<DataverseName,Identifier> pairId = null;
2127 boolean ifExists = false;
2128}
2129{
2130 pairId = QualifiedName() ifExists = IfExists()
2131 {
2132 ViewDropStatement stmt = new ViewDropStatement(pairId.first, pairId.second, ifExists);
2133 return addSourceLocation(stmt, startStmtToken);
2134 }
2135}
2136
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002137IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
2138{
2139 IndexDropStatement stmt = null;
2140}
2141{
2142 <INDEX> stmt = DropIndexSpecification(startStmtToken)
2143 {
2144 return stmt;
2145 }
2146}
2147
2148IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
2149{
2150 Triple<DataverseName,Identifier,Identifier> tripleId = null;
2151 boolean ifExists = false;
2152}
2153{
2154 tripleId = DoubleQualifiedName() ifExists = IfExists()
2155 {
2156 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
2157 return addSourceLocation(stmt, startStmtToken);
2158 }
2159}
2160
Rui Guoe6986dd2020-11-25 19:50:06 -08002161Statement DropFullTextStatement(Token startStmtToken) throws ParseException:
2162{
2163 Statement stmt = null;
2164}
2165{
2166 <FULLTEXT>
2167 (
2168 <FILTER> stmt = DropFullTextFilterSpec(startStmtToken)
2169 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = DropFullTextConfigSpec(startStmtToken))
2170 )
2171 {
2172 return stmt;
2173 }
2174}
2175
2176FullTextFilterDropStatement DropFullTextFilterSpec(Token startStmtToken) throws ParseException:
2177{
2178 FullTextFilterDropStatement stmt = null;
2179 Pair<DataverseName,Identifier> nameComponents = null;
2180 boolean ifExists = false;
2181}
2182{
2183 nameComponents = QualifiedName() ifExists = IfExists()
2184 {
2185 stmt = new FullTextFilterDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2186 return addSourceLocation(stmt, startStmtToken);
2187 }
2188}
2189
2190FullTextConfigDropStatement DropFullTextConfigSpec(Token startStmtToken) throws ParseException:
2191{
2192 FullTextConfigDropStatement stmt = null;
2193 Pair<DataverseName,Identifier> nameComponents = null;
2194 boolean ifExists = false;
2195}
2196{
2197 nameComponents = QualifiedName() ifExists = IfExists()
2198 {
2199 stmt = new FullTextConfigDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2200 return addSourceLocation(stmt, startStmtToken);
2201 }
2202}
2203
2204
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002205NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
2206{
2207 NodeGroupDropStatement stmt = null;
2208}
2209{
2210 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
2211 {
2212 return stmt;
2213 }
2214}
2215
2216NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
2217{
2218 String id = null;
2219 boolean ifExists = false;
2220}
2221{
2222 id = Identifier() ifExists = IfExists()
2223 {
2224 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
2225 return addSourceLocation(stmt, startStmtToken);
2226 }
2227}
2228
2229TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
2230{
2231 TypeDropStatement stmt = null;
2232}
2233{
2234 <TYPE> stmt = DropTypeSpecification(startStmtToken)
2235 {
2236 return stmt;
2237 }
2238}
2239
2240TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
2241{
2242 Pair<DataverseName,Identifier> pairId = null;
2243 boolean ifExists = false;
2244}
2245{
2246 pairId = TypeName() ifExists = IfExists()
2247 {
2248 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
2249 return addSourceLocation(stmt, startStmtToken);
2250 }
2251}
2252
2253DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
2254{
2255 DataverseDropStatement stmt = null;
2256}
2257{
2258 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
2259 {
2260 return stmt;
2261 }
2262}
2263
2264DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
2265{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002266 DataverseName dvName = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002267 boolean ifExists = false;
2268}
2269{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002270 dvName = DataverseName() ifExists = IfExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002271 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002272 DataverseDropStatement stmt = new DataverseDropStatement(dvName, ifExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002273 return addSourceLocation(stmt, startStmtToken);
2274 }
2275}
2276
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002277AdapterDropStatement DropAdapterStatement(Token startStmtToken) throws ParseException:
2278{
2279 AdapterDropStatement stmt = null;
2280}
2281{
2282 <ADAPTER> stmt = DropAdapterSpecification(startStmtToken)
2283 {
2284 return stmt;
2285 }
2286}
2287
2288AdapterDropStatement DropAdapterSpecification(Token startStmtToken) throws ParseException:
2289{
2290 Pair<DataverseName,Identifier> adapterName = null;
2291 boolean ifExists = false;
2292}
2293{
2294 adapterName = QualifiedName() ifExists = IfExists()
2295 {
2296 AdapterDropStatement stmt = new AdapterDropStatement(adapterName.first, adapterName.second.getValue(), ifExists);
2297 return addSourceLocation(stmt, startStmtToken);
2298 }
2299}
2300
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002301FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
2302{
2303 FunctionDropStatement stmt = null;
2304}
2305{
2306 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
2307 {
2308 return stmt;
2309 }
2310}
2311
2312FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
2313{
2314 FunctionSignature funcSig = null;
2315 boolean ifExists = false;
2316}
2317{
2318 funcSig = FunctionSignature() ifExists = IfExists()
2319 {
2320 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
2321 return addSourceLocation(stmt, startStmtToken);
2322 }
2323}
2324
2325FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
2326{
2327 FeedDropStatement stmt = null;
2328}
2329{
2330 <FEED> stmt = DropFeedSpecification(startStmtToken)
2331 {
2332 return stmt;
2333 }
2334}
2335
2336FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
2337{
2338 Pair<DataverseName,Identifier> pairId = null;
2339 boolean ifExists = false;
2340}
2341{
2342 pairId = QualifiedName() ifExists = IfExists()
2343 {
2344 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
2345 return addSourceLocation(stmt, startStmtToken);
2346 }
2347}
2348
2349FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
2350{
2351 FeedPolicyDropStatement stmt = null;
2352}
2353{
2354 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
2355 {
2356 return stmt;
2357 }
2358}
2359
2360FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
2361{
2362 Pair<DataverseName,Identifier> pairId = null;
2363 boolean ifExists = false;
2364}
2365{
2366 pairId = QualifiedName() ifExists = IfExists()
2367 {
2368 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
2369 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002370 }
2371}
2372
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002373SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
2374{
2375 SynonymDropStatement stmt = null;
2376}
2377{
2378 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
2379 {
2380 return stmt;
2381 }
2382}
2383
2384SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
2385{
2386 Pair<DataverseName,Identifier> pairId = null;
2387 boolean ifExists = false;
2388}
2389{
2390 pairId = QualifiedName() ifExists = IfExists()
2391 {
2392 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
2393 return addSourceLocation(stmt, startStmtToken);
2394 }
2395}
2396
Yingyi Bu391f09e2015-10-29 13:49:39 -07002397boolean IfExists() throws ParseException :
2398{
2399}
2400{
2401 ( LOOKAHEAD(1) <IF> <EXISTS>
2402 {
2403 return true;
2404 }
2405 )?
2406 {
2407 return false;
2408 }
2409}
2410
2411InsertStatement InsertStatement() throws ParseException:
2412{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002413 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002414 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002415 VariableExpr var = null;
2416 Query query = null;
2417 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002418}
2419{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002420 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002421 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002422 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002423 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002424 if (var == null && returnExpression != null) {
2425 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2426 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002427 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08002428 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002429 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2430 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002431 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002432 }
2433}
2434
2435UpsertStatement UpsertStatement() throws ParseException:
2436{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002437 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002438 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002439 VariableExpr var = null;
2440 Query query = null;
2441 Expression returnExpression = null;
2442}
2443{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002444 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002445 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002446 ( <RETURNING> returnExpression = Expression())?
2447 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002448 if (var == null && returnExpression != null) {
2449 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2450 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002451 }
2452 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002453 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2454 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002455 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002456 }
2457}
2458
2459DeleteStatement DeleteStatement() throws ParseException:
2460{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002461 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002462 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002463 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002464 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002465}
2466{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002467 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002468 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002469 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07002470 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002471 if (var == null) {
2472 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2473 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002474 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002475 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07002476 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002477 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002478 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002479}
2480
2481UpdateStatement UpdateStatement() throws ParseException:
2482{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002483 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002484 VariableExpr vars;
2485 Expression target;
2486 Expression condition;
2487 UpdateClause uc;
2488 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
2489}
2490{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002491 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002492 <WHERE> condition = Expression()
2493 <LEFTPAREN> (uc = UpdateClause()
2494 {
2495 ucs.add(uc);
2496 }
2497 (<COMMA> uc = UpdateClause()
2498 {
2499 ucs.add(uc);
2500 }
2501 )*) <RIGHTPAREN>
2502 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002503 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002504 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002505 }
2506}
2507
2508UpdateClause UpdateClause() throws ParseException:
2509{
2510 Expression target = null;
2511 Expression value = null ;
2512 InsertStatement is = null;
2513 DeleteStatement ds = null;
2514 UpdateStatement us = null;
2515 Expression condition = null;
2516 UpdateClause ifbranch = null;
2517 UpdateClause elsebranch = null;
2518}
2519{
2520 (<SET> target = Expression() <EQ> value = Expression()
2521 | is = InsertStatement()
2522 | ds = DeleteStatement()
2523 | us = UpdateStatement()
2524 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
2525 <THEN> ifbranch = UpdateClause()
2526 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
2527 {
2528 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
2529 }
2530 )
2531}
2532
2533Statement SetStatement() throws ParseException:
2534{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002535 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002536 String pn = null;
2537 String pv = null;
2538}
2539{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002540 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002541 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002542 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002543 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002544 }
2545}
2546
2547Statement WriteStatement() throws ParseException:
2548{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002549 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002550 String nodeName = null;
2551 String fileName = null;
2552 Query query;
2553 String writerClass = null;
2554 Pair<Identifier,Identifier> nameComponents = null;
2555}
2556{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002557 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07002558 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002559 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002560 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002561 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002562 }
2563}
2564
2565LoadStatement LoadStatement() throws ParseException:
2566{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002567 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002568 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002569 Identifier datasetName = null;
2570 boolean alreadySorted = false;
2571 String adapterName;
2572 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002573 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002574}
2575{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002576 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002577 {
2578 dataverseName = nameComponents.first;
2579 datasetName = nameComponents.second;
2580 }
2581 <USING> adapterName = AdapterName() properties = Configuration()
2582 (<PRESORTED>
2583 {
2584 alreadySorted = true;
2585 }
2586 )?
2587 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002588 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
2589 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002590 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002591 }
2592}
2593
2594
2595String AdapterName() throws ParseException :
2596{
2597 String adapterName = null;
2598}
2599{
2600 adapterName = Identifier()
2601 {
2602 return adapterName;
2603 }
2604}
2605
2606Statement CompactStatement() throws ParseException:
2607{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002608 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002609 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002610}
2611{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002612 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002613 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002614 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002615 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002616 }
2617}
2618
Yingyi Buab817482016-08-19 21:29:31 -07002619Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002620{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002621 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002622 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002623}
2624{
2625 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002626 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
2627 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
2628 | <START> { startToken = token; } stmt = StartStatement(startToken)
2629 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07002630 )
2631 {
2632 return stmt;
2633 }
2634}
2635
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002636Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002637{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002638 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002639 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002640}
2641{
2642 <FEED> feedNameComponents = QualifiedName()
2643 {
2644 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002645 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002646 }
2647}
2648
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002649AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002650{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002651 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002652 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002653}
2654{
2655 <FEED> feedNameComponents = QualifiedName()
2656 {
2657 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002658 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002659 }
2660}
2661
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002662AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002663{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002664 Pair<DataverseName,Identifier> feedNameComponents = null;
2665 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002666
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002667 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002668}
2669{
2670 (
2671 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002672 {
2673 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
2674 }
2675 )
Yingyi Buab817482016-08-19 21:29:31 -07002676 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002677 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002678 }
2679}
2680
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002681AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002682{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002683 Pair<DataverseName,Identifier> feedNameComponents = null;
2684 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002685
2686 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002687 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002688 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002689 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002690 String whereClauseBody = null;
2691 WhereClause whereClause = null;
2692 Token beginPos = null;
2693 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002694}
2695{
2696 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002697 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002698 (ApplyFunction(appliedFunctions))?
2699 (policy = GetPolicy())?
2700 (
2701 <WHERE>
2702 {
2703 beginPos = token;
2704 whereClause = new WhereClause();
2705 Expression whereExpr;
2706 }
2707 whereExpr = Expression()
2708 {
2709 whereClause.setWhereExpr(whereExpr);
2710 }
2711 )?
2712 {
2713 if (whereClause != null) {
2714 endPos = token;
2715 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2716 }
2717 }
Yingyi Buab817482016-08-19 21:29:31 -07002718 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002719 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002720 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002721 }
2722 )
2723 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002724 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002725 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002726}
2727
2728Map<String,String> Configuration() throws ParseException :
2729{
2730 Map<String,String> configuration = new LinkedHashMap<String,String>();
2731 Pair<String, String> keyValuePair = null;
2732}
2733{
2734 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2735 {
2736 configuration.put(keyValuePair.first, keyValuePair.second);
2737 }
2738 ( <COMMA> keyValuePair = KeyValuePair()
2739 {
2740 configuration.put(keyValuePair.first, keyValuePair.second);
2741 }
2742 )* )? <RIGHTPAREN>
2743 {
2744 return configuration;
2745 }
2746}
2747
2748Pair<String, String> KeyValuePair() throws ParseException:
2749{
2750 String key;
2751 String value;
2752}
2753{
Ali Alsulimane2986012020-05-14 18:14:22 -07002754 <LEFTPAREN> key = ConstantString()
2755 <EQ> ( value = ConstantString() | (<TRUE> | <FALSE>) {value = token.image.toLowerCase();} )
2756 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002757 {
2758 return new Pair<String, String>(key, value);
2759 }
2760}
2761
2762Map<String,String> Properties() throws ParseException:
2763{
2764 Map<String,String> properties = new HashMap<String,String>();
2765 Pair<String, String> property;
2766}
2767{
2768 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
2769 {
2770 properties.put(property.first, property.second);
2771 }
2772 ( <COMMA> property = Property()
2773 {
2774 properties.put(property.first, property.second);
2775 }
2776 )* <RIGHTPAREN> )?
2777 {
2778 return properties;
2779 }
2780}
2781
2782Pair<String, String> Property() throws ParseException:
2783{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002784 String key = null;
2785 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002786}
2787{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002788 (key = Identifier() | key = StringLiteral())
2789 <EQ>
2790 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002791 {
2792 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002793 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002794 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002795 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002796 }
2797 }
2798 )
2799 {
2800 return new Pair<String, String>(key.toUpperCase(), value);
2801 }
2802}
2803
Glenn58329202021-04-09 12:03:46 -07002804IndexedTypeExpression IndexedTypeExpr(boolean allowQues) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002805{
2806 TypeExpression typeExpr = null;
Glenn58329202021-04-09 12:03:46 -07002807 boolean isUnknownable = !allowQues;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002808}
2809{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002810 typeExpr = TypeExpr(false)
Glenn58329202021-04-09 12:03:46 -07002811 ( <QUES>
2812 {
2813 if (!allowQues) {
2814 throw new SqlppParseException(getSourceLocation(token), "'?' quantifier is illegal for the type expression.");
2815 }
2816 isUnknownable = true;
2817 }
2818 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002819 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002820 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002821 }
2822}
2823
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002824TypeExpression TypeExpr(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002825{
2826 TypeExpression typeExpr = null;
2827}
2828{
2829 (
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002830 typeExpr = TypeReference()
2831 | typeExpr = OrderedListTypeDef(allowRecordTypeDef)
2832 | typeExpr = UnorderedListTypeDef(allowRecordTypeDef)
2833 | typeExpr = RecordTypeDef() {
2834 if (!allowRecordTypeDef) {
2835 throw new SqlppParseException(typeExpr.getSourceLocation(), "Unexpected record type declaration");
2836 }
2837 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002838 )
2839 {
2840 return typeExpr;
2841 }
2842}
2843
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002844RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
2845{
2846 RecordTypeDefinition.RecordKind recordKind = null;
2847}
2848{
2849 (
2850 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
2851 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
2852 )
2853 {
2854 return recordKind;
2855 }
2856}
2857
Yingyi Bu391f09e2015-10-29 13:49:39 -07002858RecordTypeDefinition RecordTypeDef() throws ParseException:
2859{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002860 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002861 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002862 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002863}
2864{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002865 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002866 <LEFTBRACE>
2867 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002868 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002869 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
2870 if (hintToken != null) {
2871 String hintParams = hintToken.hintParams;
2872 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
2873 if (splits == null || splits.length != 4) {
2874 throw new SqlppParseException(getSourceLocation(hintToken),
2875 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002876 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002877 if (!splits[0].equals("int")) {
2878 throw new SqlppParseException(getSourceLocation(hintToken),
2879 "The only supported type for gen-fields is int.");
2880 }
2881 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
2882 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
2883 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002884 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002885 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002886 (
2887 RecordField(recType)
2888 ( <COMMA> RecordField(recType) )*
2889 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002890 <RIGHTBRACE>
2891 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002892 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002893 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002894 }
2895}
2896
2897void RecordField(RecordTypeDefinition recType) throws ParseException:
2898{
2899 String fieldName;
2900 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002901 boolean nullable = false, missable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002902}
2903{
2904 fieldName = Identifier()
2905 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002906 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
2907 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
2908 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
2909 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002910 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002911 <COLON> type = TypeExpr(true) ( <QUES> { nullable = true; missable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002912 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002913 recType.addField(fieldName, type, nullable, missable, rfdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002914 }
2915}
2916
2917TypeReferenceExpression TypeReference() throws ParseException:
2918{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002919 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002920}
2921{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002922 id = QualifiedName()
2923 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07002924 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
2925 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002926 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002927
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002928 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002929 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002930 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002931}
2932
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002933OrderedListTypeDefinition OrderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002934{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002935 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002936 TypeExpression type = null;
2937}
2938{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002939 <LEFTBRACKET> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002940 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002941 <RIGHTBRACKET>
2942 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002943 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002944 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002945 }
2946}
2947
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002948UnorderedListTypeDefinition UnorderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002949{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002950 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002951 TypeExpression type = null;
2952}
2953{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002954 <LEFTDBLBRACE> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002955 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002956 <RIGHTDBLBRACE>
2957 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002958 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002959 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002960 }
2961}
2962
2963FunctionName FunctionName() throws ParseException:
2964{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002965 Triple<List<String>, Token, Token> prefix = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002966 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002967}
2968{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002969 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
2970 // that copy must be kept in sync with this code
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002971 prefix = MultipartIdentifierWithHints(
Tin Vu4f1090d2021-09-21 02:06:32 -07002972 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.RANGE_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT,
2973 SqlppHint.SPATIAL_JOIN_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002974 )
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002975 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002976 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002977 Token startToken = prefix.second;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002978 FunctionName result = new FunctionName();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002979 result.sourceLoc = getSourceLocation(startToken);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002980 result.hintToken = prefix.third;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002981 List<String> list = prefix.first;
2982 int ln = list.size();
2983 String last = list.get(ln - 1);
2984 if (suffix == null) {
2985 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
2986 // no library name
2987 result.function = last;
2988 } else {
2989 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
2990 // suffix = func_name
2991 result.library = last;
2992 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002993 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002994 if (ln > 1) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002995 result.dataverse = createDataverseName(list, 0, ln - 1, startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002996 } else {
2997 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002998 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002999
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003000 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
3001 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003002 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003003 return result;
3004 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003005}
3006
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003007Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003008{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003009 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003010}
3011{
3012 name = QualifiedName()
3013 {
3014 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003015 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003016 }
3017 return name;
3018 }
3019}
3020
3021String Identifier() throws ParseException:
3022{
3023 String lit = null;
3024}
3025{
3026 (<IDENTIFIER>
3027 {
3028 return token.image;
3029 }
3030 | lit = QuotedString()
3031 {
3032 return lit;
3033 }
3034 )
3035}
3036
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003037List<String> ParenthesizedIdentifierList() throws ParseException:
3038{
3039 List<String> list = new ArrayList<String>();
3040 String ident = null;
3041}
3042{
3043 <LEFTPAREN>
3044 ident = Identifier() { list.add(ident); }
3045 ( <COMMA> ident = Identifier() { list.add(ident); } )*
3046 <RIGHTPAREN>
3047 {
3048 return list;
3049 }
3050}
3051
Tin Vu4f1090d2021-09-21 02:06:32 -07003052List<Literal> ParenthesizedLiteralList() throws ParseException:
3053{
3054 List<Literal> list = new ArrayList<Literal>();
3055 boolean minus = false;
3056 Expression litExpr = null;
3057 Literal lit = null;
3058}
3059{
3060 <LEFTPAREN>
3061 (<MINUS> { minus = true; })? litExpr = Literal()
3062 {
3063 lit = ((LiteralExpr) litExpr).getValue();
3064 if (minus)
3065 {
3066 try {
3067 lit = ExpressionUtils.reverseSign(lit);
3068 } catch (TypeMismatchException e) {
3069 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
3070 }
3071 minus = false;
3072 }
3073 list.add(lit);
3074 }
3075 ( <COMMA> (<MINUS> { minus = true; })? litExpr = Literal()
3076 {
3077 lit = ((LiteralExpr) litExpr).getValue();
3078 if (minus)
3079 {
3080 try {
3081 lit = ExpressionUtils.reverseSign(lit);
3082 } catch (TypeMismatchException e) {
3083 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
3084 }
3085 minus = false;
3086 }
3087 list.add(lit);
3088 } )*
3089 <RIGHTPAREN>
3090 {
3091 return list;
3092 }
3093}
3094
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003095Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003096{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003097 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003098 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003099}
3100{
3101 fieldList = NestedField()
Glenn58329202021-04-09 12:03:46 -07003102 ( <COLON> fieldType = IndexedTypeExpr(true) )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003103 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003104 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
3105 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07003106 }
3107}
3108
Yingyi Buc9bfe252016-03-01 00:02:40 -08003109Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003110{
3111 List<String> exprList = new ArrayList<String>();
3112 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003113 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003114 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003115}
3116{
3117 lit = Identifier()
3118 {
Yingyi Bub9169b62016-02-26 21:21:49 -08003119 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003120 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08003121 }
3122 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08003123 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08003124 <LEFTPAREN><RIGHTPAREN>
3125 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003126 if(!lit.equalsIgnoreCase("meta")){
3127 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08003128 }
3129 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003130 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08003131 }
3132 )?
3133 {
3134 if(!meetParens){
3135 exprList.add(lit);
3136 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003137 }
3138 (<DOT>
3139 lit = Identifier()
3140 {
3141 exprList.add(lit);
3142 }
3143 )*
3144 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08003145 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003146 }
3147}
3148
Yingyi Bu6d57e492016-06-06 21:24:42 -07003149String ConstantString() throws ParseException:
3150{
3151 String value = null;
3152}
3153{
3154 (value = QuotedString() | value = StringLiteral())
3155 {
3156 return value;
3157 }
3158}
3159
Yingyi Bu391f09e2015-10-29 13:49:39 -07003160String QuotedString() throws ParseException:
3161{
3162}
3163{
3164 <QUOTED_STRING>
3165 {
3166 return removeQuotesAndEscapes(token.image);
3167 }
3168}
3169
Yingyi Bu391f09e2015-10-29 13:49:39 -07003170String StringLiteral() throws ParseException:
3171{
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003172 StringBuilder str = null;
3173 char quote = 0;
3174 boolean ext = false;
3175 Token litToken = null;
3176 String lit = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003177}
3178{
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003179 ( <STRING_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003180 {
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003181 String quoted = token.image;
3182 char q = quoted.charAt(0);
3183 boolean e = q == 'E';
3184 if (e) {
3185 q = quoted.charAt(1);
3186 quoted = quoted.substring(1);
3187 }
3188 if (lit == null) {
3189 quote = q;
3190 ext = e;
3191 } else {
3192 boolean isAdjacent = litToken.endLine == token.beginLine && litToken.endColumn + 1 == token.beginColumn;
3193 if (!isAdjacent || ext || e || (q != quote)) {
3194 throw new SqlppParseException(getSourceLocation(token), "Invalid string literal");
3195 }
3196 if (str == null) {
3197 str = new StringBuilder();
3198 }
3199 str.append(lit);
3200 str.append(quote);
3201 }
3202 lit = removeQuotesAndEscapes(quoted);
3203 litToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003204 }
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003205 )+
3206 {
3207 if (str == null) {
3208 return lit;
3209 } else {
3210 str.append(lit);
3211 return str.toString();
3212 }
3213 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003214}
3215
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003216Triple<List<String>, Token, Token> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003217{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003218 Triple<List<String>, Token, Token> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003219}
3220{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003221 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003222 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003223 return result;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003224 }
3225}
3226
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003227Triple<List<String>, Token, Token> MultipartIdentifierWithHints(SqlppHint... expectedHints)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003228 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003229{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003230 List<String> list = new ArrayList<String>();
3231 SourceLocation sourceLoc = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003232 Token startToken, hintToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003233 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003234}
3235{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003236 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003237 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003238 list.add(item);
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003239 startToken = token;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003240 if (expectedHints != null && expectedHints.length > 0) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003241 hintToken = fetchHint(token, expectedHints);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003242 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003243 }
3244 (<DOT> item = Identifier() { list.add(item); } )*
3245 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003246 return new Triple<List<String>, Token, Token>(list, startToken, hintToken);
3247 }
3248}
3249
3250DataverseName DataverseName() throws ParseException:
3251{
3252 Triple<List<String>, Token, Token> ident = null;
3253}
3254{
3255 ident = MultipartIdentifier()
3256 {
3257 List<String> list = ident.first;
3258 Token startToken = ident.second;
3259 return createDataverseName(list, 0, list.size(), startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003260 }
3261}
3262
3263Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
3264{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003265 Triple<List<String>, Token, Token> ident = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003266}
3267{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003268 ident = MultipartIdentifier()
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003269 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003270 List<String> list = ident.first;
3271 Token startToken = ident.second;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003272 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003273 DataverseName id1 = len > 1 ? createDataverseName(list, 0, len - 1, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003274 Identifier id2 = new Identifier(list.get(len - 1));
3275 return new Pair<DataverseName,Identifier>(id1, id2);
3276 }
3277}
3278
3279Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
3280{
3281 List<String> list = new ArrayList<String>();
3282 String item = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003283 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003284}
3285{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003286 item = Identifier() { list.add(item); startToken = token; }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003287 (<DOT> item = Identifier() { list.add(item); } )+
3288 {
3289 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003290 DataverseName id1 = len > 2 ? createDataverseName(list, 0, len - 2, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003291 Identifier id2 = new Identifier(list.get(len - 2));
3292 Identifier id3 = new Identifier(list.get(len - 1));
3293 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003294 }
3295}
3296
3297FunctionDecl FunctionDeclaration() throws ParseException:
3298{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003299 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003300 String functionName;
Dmitry Lychagin541652082020-11-09 14:04:53 -08003301 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003302 Expression funcBody;
3303 createNewScope();
3304}
3305{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003306 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07003307 functionName = Identifier()
Dmitry Lychagin541652082020-11-09 14:04:53 -08003308 paramsWithArity = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07003309 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003310 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07003311 <RIGHTBRACE>
3312 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08003313 int arity = paramsWithArity.first;
3314 List<Pair<VarIdentifier,TypeExpression>> paramList = paramsWithArity.second;
3315 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, arity);
Xikui Wang3de700a2018-03-15 16:32:55 -07003316 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003317 ensureNoTypeDeclsInFunction(functionName, paramList, null, startToken);
3318 List<VarIdentifier> params = new ArrayList<VarIdentifier>(paramList.size());
3319 for (Pair<VarIdentifier,TypeExpression> p: paramList) {
3320 params.add(p.getFirst());
Ian Maxon38fe9402020-01-29 19:27:40 -08003321 }
Dmitry Lychagin9ba74872021-04-05 14:38:20 -07003322 FunctionDecl stmt = new FunctionDecl(signature, params, funcBody, false);
Xikui Wang3de700a2018-03-15 16:32:55 -07003323 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003324 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07003325 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003326}
3327
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003328Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003329{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003330 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003331 Expression expr;
3332}
3333{
3334 (
3335 expr = Expression()
3336 |
3337 expr = SelectExpression(false)
3338 )
3339 {
3340 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003341 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003342 return query;
3343 }
3344}
3345
3346
Yingyi Bu391f09e2015-10-29 13:49:39 -07003347Expression Expression():
3348{
3349 Expression expr = null;
3350 Expression exprP = null;
3351}
3352{
3353(
3354 LOOKAHEAD(2)
3355 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003356 | expr = QuantifiedExpression()
3357)
3358 {
3359 return (exprP==null) ? expr : exprP;
3360 }
3361}
3362
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003363Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003364{
3365 OperatorExpr op = null;
3366 Expression operand = null;
3367}
3368{
3369 operand = AndExpr()
3370 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003371 <OR>
3372 {
3373 if (op == null) {
3374 op = new OperatorExpr();
3375 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07003376 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003377 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003378 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003379 try{
3380 op.addOperator(token.image.toLowerCase());
3381 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003382 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003383 }
Xikui Wang3de700a2018-03-15 16:32:55 -07003384 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003385
Xikui Wang3de700a2018-03-15 16:32:55 -07003386 operand = AndExpr()
3387 {
3388 op.addOperand(operand);
3389 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003390
3391 )*
3392
3393 {
3394 return op==null? operand: op;
3395 }
3396}
3397
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003398Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003399{
3400 OperatorExpr op = null;
3401 Expression operand = null;
3402}
3403{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003404 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003405 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003406 <AND>
3407 {
3408 if (op == null) {
3409 op = new OperatorExpr();
3410 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003411 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003412 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003413 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003414 try{
3415 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003416 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003417 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003418 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003419 }
3420
Yingyi Bu196db5d2016-07-15 19:07:20 -07003421 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003422 {
3423 op.addOperand(operand);
3424 }
3425
3426 )*
3427
3428 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003429 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003430 }
3431}
3432
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003433Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07003434{
3435 Expression inputExpr;
3436 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003437 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07003438}
3439{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003440 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07003441 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003442 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003443 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003444 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003445 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003446 } else {
3447 return inputExpr;
3448 }
3449 }
3450}
Yingyi Bu391f09e2015-10-29 13:49:39 -07003451
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003452Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003453{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003454 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003455 OperatorExpr op = null;
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003456 Token opToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003457 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003458 IExpressionAnnotation annotation = null;
3459}
3460{
Yingyi Bu6c638342016-09-02 17:54:34 -07003461 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003462
3463 (
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003464 LOOKAHEAD(3)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN> |
3465 <IS> (<NOT> { not = true; })? <DISTINCT> { opToken = token; } <FROM> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003466 {
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003467 if (opToken == null) {
3468 opToken = token;
3469 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003470 Token hintToken = fetchHint(token,
3471 SqlppHint.HASH_BROADCAST_JOIN_HINT, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
3472 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
3473 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003474 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003475 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003476 }
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003477 String operator = opToken.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07003478 if (operator.equals("<>")){
3479 operator = "!=";
3480 }
3481 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003482 operator = "not_" + operator;
3483 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003484 if (op == null) {
3485 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07003486 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003487 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003488 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003489 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003490 try{
3491 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003492 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003493 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003494 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003495 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003496
Yingyi Bu6c638342016-09-02 17:54:34 -07003497 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07003498 {
Shiva2ea73232019-10-09 18:04:05 -07003499 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07003500 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003501 )?
3502
3503 {
3504 if (annotation != null) {
3505 op.addHint(annotation);
3506 }
3507 return op==null? operand: op;
3508 }
3509}
3510
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003511Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07003512{
3513 boolean not = false;
3514 OperatorExpr op = null;
3515 Expression operand = null;
3516 IExpressionAnnotation annotation = null;
3517}
3518{
3519 operand = IsExpr()
3520 (
3521 LOOKAHEAD(2)
3522 (<NOT> { not = true; })? <BETWEEN>
3523 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003524 Token hintToken = fetchHint(token,
3525 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT,
3526 SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
3527 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003528 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003529 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu6c638342016-09-02 17:54:34 -07003530 }
3531 String operator = token.image.toLowerCase();
3532 if(not){
3533 operator = "not_" + operator;
3534 }
3535 if (op == null) {
3536 op = new OperatorExpr();
3537 op.addOperand(operand);
3538 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003539 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07003540 }
3541 try{
3542 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003543 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003544 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07003545 }
3546 }
3547
3548 operand = IsExpr()
3549 {
3550 op.addOperand(operand);
3551 }
3552
3553 <AND>
3554 operand = IsExpr()
3555 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08003556 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07003557 op.addOperand(operand);
3558 }
3559 )?
3560
3561 {
3562 if (annotation != null) {
3563 op.addHint(annotation);
3564 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003565 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07003566 }
3567}
3568
Yingyi Budaa549c2016-06-28 22:30:52 -07003569Expression IsExpr() throws ParseException:
3570{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003571 Token notToken = null;
3572 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003573 Expression operand = null;
3574 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003575 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003576}
3577{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003578 operand = LikeExpr()
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003579 (
3580 LOOKAHEAD(3)
3581 <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003582 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003583 (
3584 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
3585 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08003586 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07003587 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003588 )
Yingyi Budaa549c2016-06-28 22:30:52 -07003589 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003590 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07003591 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003592 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003593 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003594 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07003595 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003596 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07003597 }
3598 }
3599 )?
3600 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003601 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07003602 }
3603}
3604
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003605Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003606{
3607 boolean not = false;
3608 OperatorExpr op = null;
3609 Expression operand = null;
3610}
3611{
3612 operand = ConcatExpr()
3613 (
3614 LOOKAHEAD(2)
3615 (<NOT> { not = true; })? <LIKE>
3616 {
3617 op = new OperatorExpr();
3618 op.addOperand(operand);
3619 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003620 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003621
3622 String operator = token.image.toLowerCase();
3623 if (not) {
3624 operator = "not_" + operator;
3625 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003626 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003627 op.addOperator(operator);
3628 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003629 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003630 }
3631 }
3632
3633 operand = ConcatExpr()
3634 {
3635 op.addOperand(operand);
3636 }
3637 )?
3638
3639 {
3640 return op == null ? operand : op;
3641 }
3642}
3643
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003644Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003645{
3646 OperatorExpr op = null;
3647 Expression operand = null;
3648}
3649{
3650 operand = AddExpr()
3651 (
3652 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003653 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003654 {
3655 if (op == null) {
3656 op = new OperatorExpr();
3657 op.addOperand(operand);
3658 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003659 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003660 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003661 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003662 }
3663 operand = AddExpr()
3664 {
3665 op.addOperand(operand);
3666 }
3667 )*
3668
3669 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003670 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003671 }
3672}
Yingyi Budaa549c2016-06-28 22:30:52 -07003673
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003674Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003675{
3676 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003677 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003678 Expression operand = null;
3679}
3680{
3681 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07003682 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003683 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003684 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003685 {
3686 if (op == null) {
3687 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003688 op.addOperand(operand);
3689 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003690 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003691 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003692 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003693 }
3694
3695 operand = MultExpr()
3696 {
3697 op.addOperand(operand);
3698 }
3699 )*
3700
3701 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003702 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003703 }
3704}
3705
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003706Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003707{
3708 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003709 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003710 Expression operand = null;
3711}
3712{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003713 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003714
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003715 ( (
3716 <MUL> { opType = OperatorType.MUL; } |
3717 <DIVIDE> { opType = OperatorType.DIVIDE; } |
3718 <DIV> { opType = OperatorType.DIV; } |
3719 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
3720 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003721 {
3722 if (op == null) {
3723 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003724 op.addOperand(operand);
3725 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003726 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003727 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003728 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003729 }
3730 operand = ExponentExpr()
3731 {
3732 op.addOperand(operand);
3733 }
3734 )*
3735
3736 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003737 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003738 }
3739}
3740
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003741Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003742{
3743 OperatorExpr op = null;
3744 Expression operand = null;
3745}
3746{
3747 operand = UnaryExpr()
3748 (<CARET>
3749 {
3750 if (op == null) {
3751 op = new OperatorExpr();
3752 op.addOperand(operand);
3753 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003754 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003755 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003756 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003757 }
3758 operand = UnaryExpr()
3759 {
3760 op.addOperand(operand);
3761 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003762 )?
3763 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003764 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003765 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003766}
3767
3768Expression UnaryExpr() throws ParseException:
3769{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003770 boolean not = false;
3771 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003772 Expression expr = null;
3773}
Yingyi Budaa549c2016-06-28 22:30:52 -07003774{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003775 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003776 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003777 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003778 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003779 exprType = "not_" + exprType;
3780 }
3781 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003782 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003783 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003784 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003785 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003786 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07003787 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003788 }
3789 )?
3790
3791 expr = ValueExpr()
3792 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003793 if (uexpr == null) {
3794 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003795 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003796 uexpr.setExpr(expr);
3797 return uexpr;
3798 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003799 }
3800}
3801
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003802Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003803{
3804 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003805 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003806}
3807{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003808 expr = PrimaryExpr()
3809 (
3810 accessor = FieldAccessor(accessor != null ? accessor : expr)
3811 |
3812 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003813 )*
3814 {
3815 return accessor == null ? expr : accessor;
3816 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003817}
3818
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003819FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003820{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003821 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003822 String ident = null;
3823}
3824{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003825 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003826 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003827 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003828 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003829 }
3830}
3831
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003832AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003833{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003834 Token startToken = null;
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003835 boolean star = false, slice = false;
3836 Expression expr1 = null, expr2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003837}
3838{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003839 <LEFTBRACKET> { startToken = token; }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003840 (
3841 <MUL> { star = true; }
3842 |
3843 ( expr1 = Expression() ( <COLON> { slice = true; } ( expr2 = Expression() )? )? )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003844 )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003845 <RIGHTBRACKET>
3846 {
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003847 if (expr1 != null && expr1.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3848 ensureIntegerLiteral( (LiteralExpr) expr1, "Index");
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003849 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003850 if (expr2 != null && expr2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3851 ensureIntegerLiteral( (LiteralExpr) expr2, "Index");
3852 }
3853 AbstractAccessor resultAccessor;
3854 if (slice) {
3855 resultAccessor = new ListSliceExpression(inputExpr, expr1, expr2);
3856 } else if (star) {
3857 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.STAR, null);
3858 } else {
3859 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.ELEMENT, expr1);
3860 }
3861 return addSourceLocation(resultAccessor, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003862 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003863}
3864
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003865Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003866{
3867 Expression expr = null;
3868}
3869{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003870 (
3871 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07003872 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003873 | expr = Literal()
3874 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003875 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003876 | expr = ListConstructor()
3877 | expr = RecordConstructor()
3878 | expr = ParenthesizedExpression()
3879 )
3880 {
3881 return expr;
3882 }
3883}
3884
3885Expression Literal() throws ParseException:
3886{
3887 LiteralExpr lit = new LiteralExpr();
3888 String str = null;
3889}
3890{
3891 ( str = StringLiteral()
3892 {
3893 lit.setValue(new StringLiteral(str));
3894 }
3895 | <INTEGER_LITERAL>
3896 {
Till Westmann68c6a992016-09-30 00:19:12 -07003897 try {
3898 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
3899 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003900 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003901 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003902 }
3903 | <FLOAT_LITERAL>
3904 {
Till Westmann68c6a992016-09-30 00:19:12 -07003905 try {
3906 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
3907 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003908 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003909 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003910 }
3911 | <DOUBLE_LITERAL>
3912 {
Till Westmann68c6a992016-09-30 00:19:12 -07003913 try {
3914 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
3915 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003916 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003917 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003918 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07003919 | <MISSING>
3920 {
3921 lit.setValue(MissingLiteral.INSTANCE);
3922 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003923 | <NULL>
3924 {
3925 lit.setValue(NullLiteral.INSTANCE);
3926 }
3927 | <TRUE>
3928 {
3929 lit.setValue(TrueLiteral.INSTANCE);
3930 }
3931 | <FALSE>
3932 {
3933 lit.setValue(FalseLiteral.INSTANCE);
3934 }
3935 )
3936 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003937 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003938 }
3939}
3940
Yingyi Bu391f09e2015-10-29 13:49:39 -07003941VariableExpr VariableRef() throws ParseException:
3942{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003943 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003944}
3945{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003946 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003947 {
3948 Identifier ident = lookupSymbol(id);
3949 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003950 throw new SqlppParseException(getSourceLocation(token),
3951 "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 -07003952 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003953 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003954 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003955 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003956 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003957 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07003958 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003959 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003960 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003961 }
3962}
3963
Yingyi Bu391f09e2015-10-29 13:49:39 -07003964VariableExpr Variable() throws ParseException:
3965{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003966 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003967}
3968{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003969 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003970 {
3971 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003972 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
3973 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07003974 varExp.setIsNewVar(false);
3975 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003976 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003977 }
3978}
3979
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003980String VariableIdentifier() throws ParseException:
3981{
3982 String id = null;
3983}
3984{
3985 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3986 {
3987 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
3988 }
3989}
3990
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003991Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
3992{
3993 VariableExpr var = null;
3994 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
3995}
3996{
3997 var = Variable()
3998 ( LOOKAHEAD(1)
3999 {
4000 VariableExpr fieldVarExpr = null;
4001 String fieldIdentifierStr = null;
4002 }
4003 <LEFTPAREN>
4004 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
4005 {
4006 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
4007 }
4008 (<COMMA>
4009 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
4010 {
4011 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
4012 }
4013 )*
4014 <RIGHTPAREN>
4015 )?
4016 {
4017 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
4018 }
4019}
4020
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004021VariableExpr ExternalVariableRef() throws ParseException:
4022{
4023 String name = null;
4024}
4025{
4026 (
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004027 <DOLLAR_IDENTIFIER> { name = token.image.substring(1); }
4028 | <DOLLAR_INTEGER_LITERAL> { name = token.image.substring(1); }
4029 | <DOLLAR_QUOTED_STRING> { name = removeQuotesAndEscapes(token.image.substring(1)); }
4030 | <QUES> { name = String.valueOf(++externalVarCounter); }
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004031 )
4032 {
4033 String idName = SqlppVariableUtil.toExternalVariableName(name);
4034 VarIdentifier id = new VarIdentifier(idName);
4035 VariableExpr varExp = new VariableExpr(id);
4036 return addSourceLocation(varExp, token);
4037 }
4038}
4039
Yingyi Bu391f09e2015-10-29 13:49:39 -07004040Expression ListConstructor() throws ParseException:
4041{
4042 Expression expr = null;
4043}
4044{
4045 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004046 expr = OrderedListConstructor() |
4047 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004048 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004049 {
4050 return expr;
4051 }
4052}
4053
Yingyi Bu391f09e2015-10-29 13:49:39 -07004054ListConstructor OrderedListConstructor() throws ParseException:
4055{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004056 Token startToken = null;
4057 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004058}
4059{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004060 <LEFTBRACKET> { startToken = token; }
4061 exprList = ExpressionList()
4062 <RIGHTBRACKET>
4063 {
4064 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004065 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004066 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004067}
4068
4069ListConstructor UnorderedListConstructor() throws ParseException:
4070{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004071 Token startToken = null;
4072 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004073}
4074{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004075 <LEFTDBLBRACE> { startToken = token; }
4076 exprList = ExpressionList()
4077 <RIGHTDBLBRACE>
4078 {
4079 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004080 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004081 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004082}
4083
4084List<Expression> ExpressionList() throws ParseException:
4085{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004086 Expression expr = null;
4087 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004088}
4089{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004090 (
4091 expr = Expression()
4092 {
4093 exprList.add(expr);
4094 }
4095 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07004096 {
4097 exprList.add(expr);
4098 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004099 )*
4100 )?
4101 {
4102 return exprList;
4103 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004104}
4105
Yingyi Bu391f09e2015-10-29 13:49:39 -07004106RecordConstructor RecordConstructor() throws ParseException:
4107{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004108 Token startToken = null;
4109 FieldBinding fb = null;
4110 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004111}
4112{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004113 <LEFTBRACE> { startToken = token; }
4114 (
4115 fb = FieldBinding() { fbList.add(fb); }
4116 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
4117 )?
4118 <RIGHTBRACE>
4119 {
4120 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004121 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004122 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004123}
4124
4125FieldBinding FieldBinding() throws ParseException:
4126{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004127 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004128}
4129{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004130 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004131 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004132 if (right == null) {
4133 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
4134 if (generatedIdentifier == null) {
4135 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
4136 }
4137 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07004138 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
4139 generatedNameExpr.setSourceLocation(left.getSourceLocation());
4140 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004141 } else {
4142 return new FieldBinding(left, right);
4143 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004144 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004145}
4146
Yingyi Bu391f09e2015-10-29 13:49:39 -07004147Expression FunctionCallExpr() throws ParseException:
4148{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004149 List<Expression> argList = new ArrayList<Expression>();
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004150 Expression argExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004151 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07004152 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07004153 boolean distinct = false;
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004154 Expression filterExpr = null;
4155 WindowExpression windowExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004156}
4157{
4158 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07004159 <LEFTPAREN> (
4160 ( <DISTINCT> { distinct = true; } )?
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004161 ( argExpr = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004162 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004163 if (star) {
4164 if (funcName.function.equalsIgnoreCase(BuiltinFunctions.SCALAR_COUNT.getName())) {
4165 argExpr = new LiteralExpr(new LongIntegerLiteral(1L));
4166 } else {
4167 throw new SqlppParseException(getSourceLocation(token),
4168 "The parameter * can only be used in " + BuiltinFunctions.SCALAR_COUNT.getName() + "().");
Yingyi Buf4d09842016-08-26 00:03:52 -07004169 }
Yingyi Buf4d09842016-08-26 00:03:52 -07004170 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004171 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004172 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004173 (<COMMA> argExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004174 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004175 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004176 }
4177 )*)? <RIGHTPAREN>
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004178
4179 {
4180 String name = funcName.function;
4181 if (distinct) {
4182 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
4183 }
4184 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
4185 int arity = argList.size();
4186 FunctionSignature signature = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
4187 if (signature == null) {
4188 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
4189 }
4190 }
4191
4192 ( <FILTER> <LEFTPAREN> <WHERE> filterExpr = Expression() <RIGHTPAREN> )?
4193
4194 ( LOOKAHEAD(5) windowExpr = WindowExpr(signature, argList, filterExpr) )?
4195
4196 {
4197 if (windowExpr != null) {
4198 return windowExpr;
4199 } else {
4200 CallExpr callExpr = new CallExpr(signature, argList, filterExpr);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06004201 if (funcName.hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08004202 IExpressionAnnotation annotation = parseExpressionAnnotation(funcName.hintToken);
4203 if (annotation != null) {
4204 callExpr.addHint(annotation);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004205 }
4206 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004207 FunctionMapUtil.normalizedListInputFunctions(callExpr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004208 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004209 return callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004210 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004211 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004212}
4213
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004214WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr)
4215 throws ParseException:
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004216{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004217 WindowExpression windowExpr = null;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004218 Boolean fromLast = null, ignoreNulls = null;
4219}
4220{
4221 (
4222 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
4223 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
4224 <FROM> <IDENTIFIER>
4225 {
4226 if (isToken(FIRST)) {
4227 fromLast = false;
4228 } else if (isToken(LAST)) {
4229 fromLast = true;
4230 } else {
4231 throw createUnexpectedTokenError();
4232 }
4233 }
4234 )?
4235 (
4236 // ( RESPECT | IGNORE ) NULLS OVER
4237 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
4238 <IDENTIFIER>
4239 {
4240 if (isToken(RESPECT)) {
4241 ignoreNulls = false;
4242 } else if (isToken(IGNORE)) {
4243 ignoreNulls = true;
4244 } else {
4245 throw createUnexpectedTokenError();
4246 }
4247 }
4248 <IDENTIFIER>
4249 {
4250 if (!isToken(NULLS)) {
4251 throw createUnexpectedTokenError();
4252 }
4253 }
4254 )?
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004255 <OVER> windowExpr = OverClause(signature, argList, aggFilterExpr, token, fromLast, ignoreNulls)
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004256 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004257 return windowExpr;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004258 }
4259}
4260
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004261WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr,
4262 Token startToken, Boolean fromLast, Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004263{
4264 Expression partitionExpr = null;
4265 List<Expression> partitionExprs = new ArrayList<Expression>();
4266 OrderbyClause orderByClause = null;
4267 List<Expression> orderbyList = null;
4268 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004269 List<OrderbyClause.NullOrderModifier> orderbyNullModifierList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004270 WindowExpression.FrameMode frameMode = null;
4271 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
4272 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
4273 Expression frameStartExpr = null, frameEndExpr = null;
4274 WindowExpression.FrameExclusionKind frameExclusionKind = null;
4275 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
4276 VariableExpr windowVar = null;
4277 List<Pair<Expression, Identifier>> windowFieldList = null;
4278}
4279{
4280 (
4281 windowVarWithFieldList = VariableWithFieldMap() <AS>
4282 {
4283 windowVar = windowVarWithFieldList.first;
4284 windowFieldList = windowVarWithFieldList.second;
4285 }
4286 )?
4287 <LEFTPAREN>
4288 (
4289 <IDENTIFIER> { expectToken(PARTITION); } <BY>
4290 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
4291 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
4292 )?
4293 (
4294 orderByClause = OrderbyClause()
4295 {
4296 orderbyList = orderByClause.getOrderbyList();
4297 orderbyModifierList = orderByClause.getModifierList();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004298 orderbyNullModifierList = orderByClause.getNullModifierList();
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004299 }
4300 (
4301 frameMode = WindowFrameMode()
4302 (
4303 frameStart = WindowFrameBoundary() |
4304 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
4305 )
4306 ( frameExclusionKind = WindowFrameExclusion() )?
4307 {
4308 frameStartKind = frameStart.first;
4309 frameStartExpr = frameStart.second;
4310 if (frameEnd == null) {
4311 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4312 } else {
4313 frameEndKind = frameEnd.first;
4314 frameEndExpr = frameEnd.second;
4315 }
4316 if (frameExclusionKind == null) {
4317 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
4318 }
4319 }
4320 )?
4321 )?
4322 <RIGHTPAREN>
4323 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004324 WindowExpression winExpr = new WindowExpression(signature, argList, aggFilterExpr, partitionExprs, orderbyList,
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004325 orderbyModifierList, orderbyNullModifierList, frameMode, frameStartKind, frameStartExpr, frameEndKind,
4326 frameEndExpr, frameExclusionKind, windowVar, windowFieldList, ignoreNulls, fromLast);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004327 return addSourceLocation(winExpr, startToken);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004328 }
4329}
4330
4331WindowExpression.FrameMode WindowFrameMode() throws ParseException:
4332{
4333}
4334{
4335 <IDENTIFIER>
4336 {
4337 if (isToken(RANGE)) {
4338 return WindowExpression.FrameMode.RANGE;
4339 } else if (isToken(ROWS)) {
4340 return WindowExpression.FrameMode.ROWS;
4341 } else if (isToken(GROUPS)) {
4342 return WindowExpression.FrameMode.GROUPS;
4343 } else {
4344 throw createUnexpectedTokenError();
4345 }
4346 }
4347}
4348
4349Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
4350{
4351 boolean current = false;
4352 Expression expr = null;
4353}
4354{
4355 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004356 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004357 | expr = Expression()
4358 )
4359 <IDENTIFIER>
4360 {
4361 WindowExpression.FrameBoundaryKind kind;
4362 if (current && isToken(ROW)) {
4363 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4364 } else if (!current && isToken(PRECEDING)) {
4365 kind = expr == null
4366 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
4367 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
4368 } else if (!current && isToken(FOLLOWING)) {
4369 kind = expr == null
4370 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
4371 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
4372 } else {
4373 throw createUnexpectedTokenError();
4374 }
4375 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
4376 }
4377}
4378
4379WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
4380{
4381 boolean current = false, no = false;
4382}
4383{
4384 <IDENTIFIER>
4385 {
4386 expectToken(EXCLUDE);
4387 }
4388 (
4389 <GROUP>
4390 {
4391 return WindowExpression.FrameExclusionKind.GROUP;
4392 }
4393 |
4394 (
4395 <IDENTIFIER>
4396 {
4397 if (isToken(TIES)) {
4398 return WindowExpression.FrameExclusionKind.TIES;
4399 } else if (isToken(CURRENT)) {
4400 current = true;
4401 } else if (isToken(NO)) {
4402 no = true;
4403 } else {
4404 throw createUnexpectedTokenError();
4405 }
4406 }
4407 <IDENTIFIER>
4408 {
4409 if (current && isToken(ROW)) {
4410 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
4411 } else if (no && isToken(OTHERS)) {
4412 return WindowExpression.FrameExclusionKind.NO_OTHERS;
4413 } else {
4414 throw createUnexpectedTokenError();
4415 }
4416 }
4417 )
4418 )
4419}
4420
Yingyi Bu391f09e2015-10-29 13:49:39 -07004421Expression ParenthesizedExpression() throws ParseException:
4422{
Dmitry Lychagine9630592021-08-30 15:29:30 -07004423 Token startToken = null;
4424 Expression expr = null, expr2 = null;
4425 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004426}
4427{
4428 (
4429 LOOKAHEAD(2)
Dmitry Lychagine9630592021-08-30 15:29:30 -07004430 <LEFTPAREN> { startToken = token; } expr = Expression()
4431 (
4432 <COMMA> expr2 = Expression()
4433 {
4434 if (exprList == null) {
4435 exprList = new ArrayList<Expression>();
4436 exprList.add(expr);
4437 }
4438 exprList.add(expr2);
4439 }
4440 )*
4441 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004442 |
4443 expr = Subquery()
4444 )
4445 {
Dmitry Lychagine9630592021-08-30 15:29:30 -07004446 if (exprList != null) {
4447 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
4448 return addSourceLocation(listExpr, startToken);
4449 } else {
4450 return expr;
4451 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004452 }
4453}
4454
Yingyi Buc8c067c2016-07-25 23:37:19 -07004455Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004456{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004457 Token startToken = null;
4458 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004459 List<Expression> whenExprs = new ArrayList<Expression>();
4460 List<Expression> thenExprs = new ArrayList<Expression>();
4461 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004462 Expression whenExpr = null;
4463 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004464}
4465{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004466 <CASE> { startToken = token; }
4467 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07004468 (
4469 <WHEN> whenExpr = Expression()
4470 {
4471 whenExprs.add(whenExpr);
4472 }
4473 <THEN> thenExpr = Expression()
4474 {
4475 thenExprs.add(thenExpr);
4476 }
4477 )*
4478 (<ELSE> elseExpr = Expression() )?
4479 <END>
4480 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004481 if (conditionExpr == null) {
4482 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004483 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004484 }
4485 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004486 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07004487 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004488}
4489
Yingyi Buab817482016-08-19 21:29:31 -07004490SelectExpression SelectExpression(boolean subquery) throws ParseException:
4491{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004492 List<LetClause> letClauses = new ArrayList<LetClause>();
4493 SelectSetOperation selectSetOperation;
4494 OrderbyClause orderbyClause = null;
4495 LimitClause limitClause = null;
4496 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07004497}
4498{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004499 ( letClauses = LetClause() )?
4500 selectSetOperation = SelectSetOperation()
4501 (orderbyClause = OrderbyClause() {})?
4502 (limitClause = LimitClause() {})?
4503 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004504 SelectExpression selectExpr =
4505 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
4506 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
4507 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004508 }
4509}
4510
Yingyi Buab817482016-08-19 21:29:31 -07004511SelectSetOperation SelectSetOperation() throws ParseException:
4512{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004513 SetOperationInput setOperationInputLeft;
4514 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
4515}
4516{
4517 {
4518 SelectBlock selectBlockLeft = null;
4519 SelectExpression subqueryLeft = null;
4520 Expression expr = null;
4521 }
4522 selectBlockLeft = SelectBlock()
4523 {
4524 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
4525 }
4526 (
4527 {
4528 SetOpType opType = SetOpType.UNION;
4529 boolean setSemantics = true;
4530 SelectBlock selectBlockRight = null;
4531 SelectExpression subqueryRight = null;
4532 }
4533 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
4534 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
4535 {
4536 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
4537 }
4538 )*
4539 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004540 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
4541 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
4542 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004543 }
4544}
4545
Yingyi Buab817482016-08-19 21:29:31 -07004546SelectExpression Subquery() throws ParseException:
4547{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004548 SelectExpression selectExpr = null;
4549}
4550{
4551 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
4552 {
4553 return selectExpr;
4554 }
4555}
4556
Yingyi Buab817482016-08-19 21:29:31 -07004557SelectBlock SelectBlock() throws ParseException:
4558{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004559 SelectClause selectClause = null;
4560 FromClause fromClause = null;
4561 List<LetClause> fromLetClauses = null;
4562 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004563 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004564 GroupbyClause groupbyClause = null;
4565 List<LetClause> gbyLetClauses = null;
4566 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004567 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004568 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004569}
4570{
4571 (
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004572 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004573 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004574 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004575 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004576 fromClause = FromClause()
4577 (
4578 fromLetClauses = LetClause()
4579 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004580 (whereClause = WhereClause())?
4581 (
4582 groupbyClause = GroupbyClause()
4583 (
4584 gbyLetClauses = LetClause()
4585 )?
4586 (havingClause = HavingClause())?
4587 )?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004588 )
4589 |
4590 (
4591 fromLetClauses = LetClause()
4592 {
4593 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
4594 SourceLocation sourceLoc = getSourceLocation(token);
4595 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
4596 missingExpr.setSourceLocation(sourceLoc);
4597 List<Expression> list = new ArrayList<Expression>(1);
4598 list.add(missingExpr);
4599 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
4600 listExpr.setSourceLocation(sourceLoc);
4601 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
4602 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
4603 fromVar.setSourceLocation(sourceLoc);
4604 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
4605 fromClause = new FromClause(fromTerms);
4606 }
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004607 (whereClause = WhereClause())?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004608 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004609 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004610 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004611 |
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004612 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004613 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004614 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004615 fromLetClauses = LetClause()
4616 )?
4617 (whereClause = WhereClause())?
4618 (
4619 groupbyClause = GroupbyClause()
4620 (
4621 gbyLetClauses = LetClause()
4622 )?
4623 (havingClause = HavingClause())?
4624 )?
4625 selectClause = SelectClause()
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004626 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004627 )
4628 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004629 if (fromLetClauses != null) {
4630 fromLetWhereClauses.addAll(fromLetClauses);
4631 }
4632 if (whereClause != null) {
4633 fromLetWhereClauses.add(whereClause);
4634 }
4635 if (gbyLetClauses != null) {
4636 gbyLetHavingClauses.addAll(gbyLetClauses);
4637 }
4638 if (havingClause != null) {
4639 gbyLetHavingClauses.add(havingClause);
4640 }
4641 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
4642 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004643 selectBlock.setSourceLocation(startSrcLoc);
4644 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004645 }
4646}
4647
Yingyi Buab817482016-08-19 21:29:31 -07004648SelectClause SelectClause() throws ParseException:
4649{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004650 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004651 SelectRegular selectRegular = null;
4652 SelectElement selectElement = null;
4653 boolean distinct = false;
4654}
4655{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004656 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004657 (
4658 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04004659 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07004660 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07004661 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004662 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004663 SourceLocation sourceLoc = getSourceLocation(startToken);
4664 if (selectRegular == null && selectElement == null){
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004665 Projection projection = new Projection(Projection.Kind.STAR, null, null);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004666 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004667 List<Projection> projections = new ArrayList<Projection>();
4668 projections.add(projection);
4669 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004670 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004671 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004672 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
4673 selectClause.setSourceLocation(sourceLoc);
4674 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004675 }
4676}
4677
Yingyi Buab817482016-08-19 21:29:31 -07004678SelectRegular SelectRegular() throws ParseException:
4679{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004680 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004681 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004682 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004683}
4684{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004685 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004686 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004687 projections.add(projection);
4688 startSrcLoc = projection.getSourceLocation();
4689 }
4690 ( LOOKAHEAD(2) <COMMA> projection = Projection()
4691 {
4692 projections.add(projection);
4693 }
4694 )*
4695 {
4696 SelectRegular selectRegular = new SelectRegular(projections);
4697 selectRegular.setSourceLocation(startSrcLoc);
4698 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004699 }
4700}
4701
Yingyi Buab817482016-08-19 21:29:31 -07004702SelectElement SelectElement() throws ParseException:
4703{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004704 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004705 Expression expr = null;
4706 String name = null;
4707}
4708{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004709 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004710 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004711 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004712 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004713 }
4714}
4715
Yingyi Buab817482016-08-19 21:29:31 -07004716Projection Projection() throws ParseException :
4717{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004718 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004719 Expression expr = null;
4720 Identifier identifier = null;
4721 String name = null;
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004722 Projection.Kind kind = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004723 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004724 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004725}
4726{
4727 (
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004728 <MUL> { kind = Projection.Kind.STAR; startSrcLoc = getSourceLocation(token); }
4729 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { kind = Projection.Kind.VAR_STAR; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004730 | expr = Expression() ((<AS>)? name = Identifier())?
4731 {
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004732 kind = Projection.Kind.NAMED_EXPR;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004733 if (name == null) {
4734 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
4735 if (generatedColumnIdentifier != null) {
4736 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
4737 }
4738 }
4739 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004740 )
4741 {
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004742 Projection projection = new Projection(kind, expr, name);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004743 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
4744 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004745 }
4746}
4747
4748FromClause FromClause() throws ParseException :
4749{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004750 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004751 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
4752 extendCurrentScope();
4753}
4754{
4755 {
4756 FromTerm fromTerm = null;
4757 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004758 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004759 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
4760 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004761 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004762 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004763 }
4764}
4765
4766FromTerm FromTerm() throws ParseException :
4767{
4768 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004769 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004770 VariableExpr posVar = null;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004771 AbstractBinaryCorrelateClause correlateClause = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004772 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
4773}
4774{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004775 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004776 (
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004777 (
4778 correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER)
4779 | ( <INNER> correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER) )
4780 | ( <LEFT> ( <OUTER> )? correlateClause = JoinOrUnnestClause(JoinType.LEFTOUTER, UnnestType.LEFTOUTER) )
4781 | ( <RIGHT> ( <OUTER> )? correlateClause = JoinClause(JoinType.RIGHTOUTER) )
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004782 | ( <CROSS> correlateClause = CrossJoinClause() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004783 )
4784 {
4785 correlateClauses.add(correlateClause);
4786 }
4787 )*
4788 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004789 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07004790 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004791 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004792 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
4793 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
4794 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004795 }
4796}
4797
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004798AbstractBinaryCorrelateClause JoinOrUnnestClause(JoinType joinType, UnnestType unnestType) throws ParseException :
4799{
4800 AbstractBinaryCorrelateClause correlateClause = null;
4801}
4802{
4803 ( correlateClause = JoinClause(joinType) | correlateClause = UnnestClause(unnestType) )
4804 {
4805 return correlateClause;
4806 }
4807}
4808
Yingyi Bu391f09e2015-10-29 13:49:39 -07004809JoinClause JoinClause(JoinType joinType) throws ParseException :
4810{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004811 Token startToken = null;
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004812 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004813 Expression conditionExpr = null;
4814}
4815{
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004816 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput() <ON> conditionExpr = Expression()
4817 {
4818 JoinClause joinClause = new JoinClause(joinType, rightInput.first, rightInput.second, rightInput.third,
4819 conditionExpr);
4820 return addSourceLocation(joinClause, startToken);
4821 }
4822}
4823
4824JoinClause CrossJoinClause() throws ParseException :
4825{
4826 Token startToken = null;
4827 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
4828 Expression conditionExpr = null;
4829}
4830{
4831 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput()
4832 {
4833 JoinClause joinClause = new JoinClause(JoinType.INNER, rightInput.first, rightInput.second, rightInput.third,
4834 new LiteralExpr(TrueLiteral.INSTANCE));
4835 return addSourceLocation(joinClause, startToken);
4836 }
4837}
4838
4839Triple<Expression, VariableExpr, VariableExpr> JoinClauseRightInput() throws ParseException :
4840{
4841 Expression rightExpr = null;
4842 VariableExpr rightVar = null;
4843 VariableExpr posVar = null;
4844}
4845{
4846 rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004847 {
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004848 if (rightVar == null) {
4849 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004850 }
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004851 return new Triple<Expression, VariableExpr, VariableExpr>(rightExpr, rightVar, posVar);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004852 }
4853}
4854
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004855UnnestClause UnnestClause(UnnestType unnestType) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004856{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004857 Token startToken = null;
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08004858 Expression rightExpr = null;
4859 VariableExpr rightVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004860 VariableExpr posVar = null;
4861}
4862{
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08004863 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004864 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004865 if (rightVar == null) {
4866 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004867 }
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004868 UnnestClause unnestClause = new UnnestClause(unnestType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004869 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004870 }
4871}
4872
Yingyi Bu391f09e2015-10-29 13:49:39 -07004873List<LetClause> LetClause() throws ParseException:
4874{
4875 List<LetClause> letList = new ArrayList<LetClause>();
4876 LetClause letClause;
4877}
4878{
4879 (
4880 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
4881 |
4882 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
4883 )
4884 {
4885 return letList;
4886 }
4887}
4888
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004889WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004890{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004891 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004892 Expression whereExpr;
4893}
4894{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004895 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004896 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004897 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004898 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004899 }
4900}
4901
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004902OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004903{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004904 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004905 OrderbyClause oc = new OrderbyClause();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004906 Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> orderbyExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004907 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004908 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004909 List<OrderbyClause.NullOrderModifier> nullModifierList = new ArrayList<OrderbyClause.NullOrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004910}
4911{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004912 <ORDER>
4913 {
4914 startToken = token;
4915 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
4916 if (hintToken != null) {
4917 switch (hintToken.hint) {
4918 case INMEMORY_HINT:
4919 String[] splits = hintToken.hintParams.split("\\s+");
4920 int numFrames = Integer.parseInt(splits[0]);
4921 int numTuples = Integer.parseInt(splits[1]);
4922 oc.setNumFrames(numFrames);
4923 oc.setNumTuples(numTuples);
4924 break;
4925 case RANGE_HINT:
4926 try {
4927 Expression rangeExpr = parseExpression(hintToken.hintParams);
4928 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
4929 oc.setRangeMap(rangeMap);
4930 } catch (CompilationException e) {
4931 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
Ali Alsuliman80225e22018-10-15 14:17:07 -07004932 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004933 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004934 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004935 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004936 }
4937 <BY> orderbyExpr = OrderByExpression()
4938 {
4939 orderbyList.add(orderbyExpr.first);
4940 modifierList.add(orderbyExpr.second);
4941 nullModifierList.add(orderbyExpr.third);
4942 }
4943 (
4944 LOOKAHEAD(2) <COMMA> orderbyExpr = OrderByExpression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004945 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004946 orderbyList.add(orderbyExpr.first);
4947 modifierList.add(orderbyExpr.second);
4948 nullModifierList.add(orderbyExpr.third);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004949 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004950 )*
4951 {
4952 oc.setOrderbyList(orderbyList);
4953 oc.setModifierList(modifierList);
4954 oc.setNullModifierList(nullModifierList);
4955 return addSourceLocation(oc, startToken);
4956 }
4957}
Yingyi Bu391f09e2015-10-29 13:49:39 -07004958
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004959Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> OrderByExpression()
4960 throws ParseException:
4961{
4962 Expression orderbyExpr = null;
4963 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
4964 OrderbyClause.NullOrderModifier nullModif = null;
4965}
4966{
4967 orderbyExpr = Expression()
4968 (
4969 <ASC> { modif = OrderbyClause.OrderModifier.ASC; }
4970 |
4971 <DESC> { modif = OrderbyClause.OrderModifier.DESC; }
4972 )?
4973 (
4974 LOOKAHEAD({ laIdentifier(NULLS) }) <IDENTIFIER> { expectToken(NULLS); } <IDENTIFIER>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004975 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004976 if (isToken(FIRST)) {
4977 nullModif = OrderbyClause.NullOrderModifier.FIRST;
4978 } else if (isToken(LAST)) {
4979 nullModif = OrderbyClause.NullOrderModifier.LAST;
4980 } else {
4981 throw createUnexpectedTokenError();
4982 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004983 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004984 )?
4985 {
4986 return new Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier>(orderbyExpr, modif,
4987 nullModif);
4988 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004989}
4990
4991GroupbyClause GroupbyClause()throws ParseException :
4992{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004993 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004994 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004995 List<List<GbyVariableExpressionPair>> gbyList = null;
4996 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004997 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07004998 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004999 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005000}
5001{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005002 {
5003 Scope newScope = extendCurrentScopeNoPush(true);
5004 // extendCurrentScope(true);
5005 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005006 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005007 {
5008 startToken = token;
5009 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
5010 if (hintToken != null) {
5011 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005012 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005013 }
5014 <BY> groupingElementList = GroupingElementList()
5015 (
5016 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08005017 {
5018 groupVar = groupVarWithFieldList.first;
5019 groupFieldList = groupVarWithFieldList.second;
5020 }
Yingyi Buacc12a92016-03-26 17:25:05 -07005021 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005022 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005023 if (groupingSetsParser == null) {
5024 groupingSetsParser = new SqlppGroupingSetsParser();
5025 }
5026 SourceLocation sourceLoc = getSourceLocation(startToken);
5027 try {
5028 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
5029 } catch (CompilationException e) {
5030 throw new SqlppParseException(sourceLoc, e.getMessage());
5031 }
5032 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005033 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07005034 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07005035 gbc.setGroupVar(groupVar);
5036 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005037 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005038 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005039 }
5040}
5041
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005042List<GroupingElement> GroupingElementList() throws ParseException:
5043{
5044 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
5045 GroupingElement groupingElement = null;
5046}
5047{
5048 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
5049 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
5050 {
5051 return groupingElementList;
5052 }
5053}
5054
5055GroupingElement GroupingElement() throws ParseException:
5056{
5057 GroupingElement groupingElement = null;
5058 List<GroupingSet> groupingSets = null;
5059 List<GroupingElement> groupingElements = null;
5060}
5061{
5062 (
5063 LOOKAHEAD(2)
5064 <LEFTPAREN> <RIGHTPAREN>
5065 {
5066 groupingElement = GroupingSet.EMPTY;
5067 }
5068 |
5069 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
5070 <IDENTIFIER> { expectToken(ROLLUP); }
5071 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
5072 {
5073 groupingElement = new RollupCube(groupingSets, false);
5074 }
5075 |
5076 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
5077 <IDENTIFIER> { expectToken(CUBE); }
5078 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
5079 {
5080 groupingElement = new RollupCube(groupingSets, true);
5081 }
5082 |
5083 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
5084 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
5085 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
5086 {
5087 groupingElement = new GroupingSets(groupingElements);
5088 }
5089 |
5090 groupingElement = OrdinaryGroupingSet()
5091 )
5092 {
5093 return groupingElement;
5094 }
5095}
5096
5097GroupingSet OrdinaryGroupingSet() throws ParseException:
5098{
5099 GbyVariableExpressionPair gbyExprPair = null;
5100 List<GbyVariableExpressionPair> items = null;
5101}
5102{
5103 (
5104 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
5105 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
5106 )
5107 {
5108 return new GroupingSet(items);
5109 }
5110}
5111
5112List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
5113{
5114 GroupingSet groupingSet = null;
5115 List<GroupingSet> items = new ArrayList<GroupingSet>();
5116}
5117{
5118 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
5119 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
5120 {
5121 return items;
5122 }
5123}
5124
5125List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
5126{
5127 GbyVariableExpressionPair gbyExprPair = null;
5128 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
5129}
5130{
5131 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
5132 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
5133 {
5134 return items;
5135 }
5136}
5137
5138GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
5139{
5140 Expression expr = null;
5141 VariableExpr var = null;
5142}
5143{
5144 expr = Expression() ( (<AS>)? var = Variable() )?
5145 {
5146 return new GbyVariableExpressionPair(var, expr);
5147 }
5148}
5149
Yingyi Bu391f09e2015-10-29 13:49:39 -07005150HavingClause HavingClause() throws ParseException:
5151{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005152 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005153 Expression filterExpr = null;
5154}
5155{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005156 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005157 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005158 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005159 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005160 }
5161}
5162
5163LimitClause LimitClause() throws ParseException:
5164{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005165 Token startToken = null;
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005166 Expression limitExpr = null, offsetExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005167}
5168{
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005169 (
5170 (
5171 <LIMIT> { startToken = token; pushForbiddenScope(getCurrentScope()); } limitExpr = Expression()
5172 ( <OFFSET> offsetExpr = Expression() )?
5173 { popForbiddenScope(); }
5174 )
5175 |
5176 (
5177 <OFFSET> { startToken = token; pushForbiddenScope(getCurrentScope()); } offsetExpr = Expression()
5178 { popForbiddenScope(); }
5179 )
5180 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005181 {
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005182 LimitClause lc = new LimitClause(limitExpr, offsetExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005183 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005184 }
5185}
5186
5187QuantifiedExpression QuantifiedExpression()throws ParseException:
5188{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005189 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005190 QuantifiedExpression qc = new QuantifiedExpression();
5191 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
5192 Expression satisfiesExpr;
5193 VariableExpr var;
5194 Expression inExpr;
5195 QuantifiedPair pair;
5196}
5197{
5198 {
5199 createNewScope();
5200 }
5201
ggalvizo21e52822021-07-27 10:26:31 -10005202 ( LOOKAHEAD(2)
5203 (<ANY>|<SOME>)<AND><EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME_AND_EVERY); }
5204 | (<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); }
5205 | <EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005206 var = Variable() <IN> inExpr = Expression()
5207 {
5208 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005209 quantifiedList.add(pair);
5210 }
5211 (
5212 <COMMA> var = Variable() <IN> inExpr = Expression()
5213 {
5214 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005215 quantifiedList.add(pair);
5216 }
5217 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07005218 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005219 {
5220 qc.setSatisfiesExpr(satisfiesExpr);
5221 qc.setQuantifiedList(quantifiedList);
5222 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005223 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005224 }
5225}
5226
5227LetClause LetElement() throws ParseException:
5228{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005229 VariableExpr varExp;
5230 Expression beExp;
5231 extendCurrentScope();
5232}
5233{
5234 varExp = Variable() <EQ> beExp = Expression()
5235 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005236 LetClause lc = new LetClause(varExp, beExp);
5237 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005238 return lc;
5239 }
5240}
5241
5242LetClause WithElement() throws ParseException:
5243{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005244 VariableExpr varExp;
5245 Expression beExp;
5246 extendCurrentScope();
5247}
5248{
5249 varExp = Variable() <AS> beExp = Expression()
5250 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005251 LetClause lc = new LetClause(varExp, beExp);
5252 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005253 return lc;
5254 }
5255}
5256
5257TOKEN_MGR_DECLS:
5258{
5259 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07005260 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005261 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005262
5263 public void pushState() {
5264 lexerStateStack.push( curLexState );
5265 }
5266
5267 public void popState(String token) {
5268 if (lexerStateStack.size() > 0) {
5269 SwitchTo( lexerStateStack.pop() );
5270 } else {
5271 int errorLine = input_stream.getEndLine();
5272 int errorColumn = input_stream.getEndColumn();
5273 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
5274 + "\" but state stack is empty.";
5275 throw new TokenMgrError(msg, -1);
5276 }
5277 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005278
5279 void CommonTokenAction(Token token) {
5280 Token hintToken = token.specialToken;
5281 if (hintToken != null) {
5282 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
5283 String text = hintToken.image.substring(1).trim();
5284 boolean hintFound = hintToken.parseHint(text);
5285 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
5286 }
5287 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005288}
5289
5290<DEFAULT,IN_DBL_BRACE>
5291TOKEN [IGNORE_CASE]:
5292{
Ian Maxon38fe9402020-01-29 19:27:40 -08005293 <ADAPTER: "adapter">
5294 | <ALL : "all">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005295 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07005296 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005297 | <APPLY : "apply">
5298 | <AS : "as">
5299 | <ASC : "asc">
5300 | <AT : "at">
5301 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005302 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005303 | <BTREE : "btree">
5304 | <BY : "by">
5305 | <CASE : "case">
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07005306 | <CAST : "cast">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005307 | <CLOSED : "closed">
5308 | <CREATE : "create">
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005309 | <CROSS : "cross">
Dmitry Lychagin0ebc4202021-01-25 13:12:41 -08005310 | <COMPACTION : "compaction"> // no longer used
Yingyi Bu391f09e2015-10-29 13:49:39 -07005311 | <COMPACT : "compact">
5312 | <CONNECT : "connect">
5313 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07005314 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07005315 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07005316 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005317 | <DECLARE : "declare">
5318 | <DEFINITION : "definition">
5319 | <DELETE : "delete">
5320 | <DESC : "desc">
5321 | <DISCONNECT : "disconnect">
5322 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005323 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005324 | <DROP : "drop">
5325 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07005326 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005327 | <ELSE : "else">
5328 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07005329 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005330 | <EVERY : "every">
5331 | <EXCEPT : "except">
5332 | <EXISTS : "exists">
5333 | <EXTERNAL : "external">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005334 | <FALSE : "false">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005335 | <FEED : "feed">
5336 | <FILTER : "filter">
5337 | <FLATTEN : "flatten">
5338 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005339 | <FROM : "from">
5340 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08005341 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005342 | <FUNCTION : "function">
5343 | <GROUP : "group">
5344 | <HAVING : "having">
5345 | <HINTS : "hints">
5346 | <IF : "if">
5347 | <INTO : "into">
5348 | <IN : "in">
5349 | <INDEX : "index">
5350 | <INGESTION : "ingestion">
5351 | <INNER : "inner">
5352 | <INSERT : "insert">
5353 | <INTERNAL : "internal">
5354 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07005355 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005356 | <JOIN : "join">
5357 | <KEYWORD : "keyword">
5358 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07005359 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005360 | <LEFT : "left">
5361 | <LETTING : "letting">
5362 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005363 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005364 | <LIMIT : "limit">
5365 | <LOAD : "load">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005366 | <MISSING : "missing">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005367 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005368 | <NODEGROUP : "nodegroup">
5369 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07005370 | <NOT : "not">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005371 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005372 | <OFFSET : "offset">
5373 | <ON : "on">
5374 | <OPEN : "open">
5375 | <OR : "or">
5376 | <ORDER : "order">
5377 | <OUTER : "outer">
5378 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07005379 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005380 | <PATH : "path">
5381 | <POLICY : "policy">
5382 | <PRESORTED : "pre-sorted">
5383 | <PRIMARY : "primary">
5384 | <RAW : "raw">
5385 | <REFRESH : "refresh">
5386 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005387 | <RETURNING : "returning">
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005388 | <RIGHT : "right">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005389 | <RTREE : "rtree">
5390 | <RUN : "run">
5391 | <SATISFIES : "satisfies">
5392 | <SECONDARY : "secondary">
5393 | <SELECT : "select">
5394 | <SET : "set">
5395 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08005396 | <START : "start">
5397 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08005398 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03005399 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07005400 | <THEN : "then">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005401 | <TO : "to">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005402 | <TRUE : "true">
5403 | <TYPE : "type">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005404 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08005405 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005406 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07005407 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005408 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07005409 | <USE : "use">
5410 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005411 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08005412 | <VALUED : "valued">
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07005413 | <VIEW : "view">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005414 | <WHEN : "when">
5415 | <WHERE : "where">
5416 | <WITH : "with">
5417 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005418}
5419
5420<DEFAULT,IN_DBL_BRACE>
5421TOKEN :
5422{
5423 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07005424 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005425 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005426 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005427 | <MUL : "*">
5428 | <PLUS : "+">
5429
5430 | <LEFTPAREN : "(">
5431 | <RIGHTPAREN : ")">
5432 | <LEFTBRACKET : "[">
5433 | <RIGHTBRACKET : "]">
5434
5435 | <ATT : "@">
5436 | <COLON : ":">
5437 | <COMMA : ",">
5438 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005439 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005440 | <QUES : "?">
5441 | <SEMICOLON : ";">
5442 | <SHARP : "#">
5443
5444 | <LT : "<">
5445 | <GT : ">">
5446 | <LE : "<=">
5447 | <GE : ">=">
5448 | <EQ : "=">
5449 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07005450 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005451 | <SIMILAR : "~=">
5452}
5453
5454<DEFAULT,IN_DBL_BRACE>
5455TOKEN :
5456{
5457 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
5458}
5459
5460<DEFAULT>
5461TOKEN :
5462{
5463 <RIGHTBRACE : "}"> { popState("}"); }
5464}
5465
5466<DEFAULT,IN_DBL_BRACE>
5467TOKEN :
5468{
5469 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
5470}
5471
5472<IN_DBL_BRACE>
5473TOKEN :
5474{
5475 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
5476}
5477
5478<DEFAULT,IN_DBL_BRACE>
5479TOKEN :
5480{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005481 <#DIGIT : ["0" - "9"]>
5482}
5483
5484<DEFAULT,IN_DBL_BRACE>
5485TOKEN:
5486{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005487 <INTEGER_LITERAL : <DIGITS> >
5488 | <DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07005489 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
5490 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005491 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005492 | <FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
Yingyi Bue4d919e2016-10-30 10:47:03 -07005493 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
5494 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005495 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005496 | <#DIGITS : (<DIGIT>)+ >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005497}
5498
5499<DEFAULT,IN_DBL_BRACE>
5500TOKEN :
5501{
5502 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005503 | <#IDENTIFIER_START_SPECIALCHAR : ["_"]>
5504 | <#IDENTIFIER_REST_SPECIALCHAR : ["$"]>
5505 | <#IDENTIFIER_START : <LETTER> | <IDENTIFIER_START_SPECIALCHAR> >
5506 | <#IDENTIFIER_REST : <LETTER> | <DIGIT> | <IDENTIFIER_START_SPECIALCHAR> | <IDENTIFIER_REST_SPECIALCHAR> >
5507 | <IDENTIFIER : <IDENTIFIER_START> (<IDENTIFIER_REST>)* >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005508}
5509
5510<DEFAULT,IN_DBL_BRACE>
5511TOKEN :
5512{
5513 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07005514 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005515 <EscapeQuot>
5516 | <EscapeBslash>
5517 | <EscapeSlash>
5518 | <EscapeBspace>
5519 | <EscapeFormf>
5520 | <EscapeNl>
5521 | <EscapeCr>
5522 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005523 | ~["`","\\"])* "`">
Dmitry Lychagin984bc522021-10-13 16:22:25 -07005524 | <STRING_LITERAL : ( ("E")? "\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005525 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005526 | <EscapeBslash>
5527 | <EscapeSlash>
5528 | <EscapeBspace>
5529 | <EscapeFormf>
5530 | <EscapeNl>
5531 | <EscapeCr>
5532 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005533 | ~["\"","\\"])* "\"")
Dmitry Lychagin984bc522021-10-13 16:22:25 -07005534 | ( ("E")? "\'" (
Yingyi Bu6d57e492016-06-06 21:24:42 -07005535 <EscapeApos>
5536 | <EscapeBslash>
5537 | <EscapeSlash>
5538 | <EscapeBspace>
5539 | <EscapeFormf>
5540 | <EscapeNl>
5541 | <EscapeCr>
5542 | <EscapeTab>
5543 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005544 | < #EscapeQuot: "\\\"" >
5545 | < #EscapeApos: "\\\'" >
5546 | < #EscapeBslash: "\\\\" >
5547 | < #EscapeSlash: "\\/" >
5548 | < #EscapeBspace: "\\b" >
5549 | < #EscapeFormf: "\\f" >
5550 | < #EscapeNl: "\\n" >
5551 | < #EscapeCr: "\\r" >
5552 | < #EscapeTab: "\\t" >
5553}
5554
5555<DEFAULT,IN_DBL_BRACE>
5556TOKEN :
5557{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005558 <DOLLAR_INTEGER_LITERAL : "$" <INTEGER_LITERAL> >
5559 | <DOLLAR_IDENTIFIER : "$" <IDENTIFIER> >
5560 | <DOLLAR_QUOTED_STRING: "$" <QUOTED_STRING> >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005561}
5562
5563<DEFAULT,IN_DBL_BRACE>
5564SKIP:
5565{
5566 " "
5567 | "\t"
5568 | "\r"
5569 | "\n"
5570}
5571
5572<DEFAULT,IN_DBL_BRACE>
5573SKIP:
5574{
5575 <"//" (~["\n"])* "\n">
5576}
5577
5578<DEFAULT,IN_DBL_BRACE>
5579SKIP:
5580{
5581 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5582}
5583
5584<DEFAULT,IN_DBL_BRACE>
5585SKIP:
5586{
Yingyi Bu93846a72016-09-13 16:30:39 -07005587 <"--" (~["\n"])* "\n">
5588}
5589
5590
5591<DEFAULT,IN_DBL_BRACE>
5592SKIP:
5593{
5594 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5595}
5596
5597<DEFAULT,IN_DBL_BRACE>
5598SKIP:
5599{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005600 <"/*"> { pushState(); } : INSIDE_COMMENT
5601}
5602
5603<INSIDE_COMMENT>
5604SPECIAL_TOKEN:
5605{
5606 <"+"(" ")*(~["*"])*>
5607}
5608
5609<INSIDE_COMMENT>
5610SKIP:
5611{
5612 <"/*"> { pushState(); }
5613}
5614
5615<INSIDE_COMMENT>
5616SKIP:
5617{
5618 <"*/"> { popState("*/"); }
5619 | <~[]>
5620}