blob: 9e3dea43f1d905e203bcbb5fae3307f37471e1d1 [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;
Ali Alsulimanba91b1c2023-02-22 21:33:55 -080055import org.apache.asterix.common.annotations.ExternalSubpathAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070056import org.apache.asterix.common.annotations.FieldIntervalDataGen;
57import org.apache.asterix.common.annotations.FieldValFileDataGen;
58import org.apache.asterix.common.annotations.FieldValFileSameIndexDataGen;
59import org.apache.asterix.common.annotations.IRecordFieldDataGen;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080060import org.apache.asterix.common.annotations.IndexedNLJoinExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070061import org.apache.asterix.common.annotations.InsertRandIntDataGen;
62import org.apache.asterix.common.annotations.ListDataGen;
63import org.apache.asterix.common.annotations.ListValFileDataGen;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -060064import org.apache.asterix.common.annotations.RangeAnnotation;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080065import org.apache.asterix.common.annotations.SecondaryIndexSearchPreferenceAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070066import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
Tin Vu4f1090d2021-09-21 02:06:32 -070067import org.apache.asterix.common.annotations.SpatialJoinAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070068import org.apache.asterix.common.annotations.TypeDataGen;
69import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
70import org.apache.asterix.common.config.DatasetConfig.DatasetType;
71import org.apache.asterix.common.config.DatasetConfig.IndexType;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -070072import org.apache.asterix.common.exceptions.AsterixException;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080073import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070074import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070075import org.apache.asterix.common.exceptions.WarningCollector;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080076import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070077import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin54c06012019-11-13 15:54:20 -080078import org.apache.asterix.common.metadata.DataverseName;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -070079import org.apache.asterix.common.metadata.DatasetFullyQualifiedName;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -070080import org.apache.asterix.external.dataset.adapter.AdapterIdentifier;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080081import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070082import org.apache.asterix.lang.common.base.AbstractLangExpression;
83import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070084import org.apache.asterix.lang.common.base.Expression;
85import org.apache.asterix.lang.common.base.Literal;
86import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070087import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070088import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070089import org.apache.asterix.lang.common.base.Statement;
90import org.apache.asterix.lang.common.clause.GroupbyClause;
91import org.apache.asterix.lang.common.clause.LetClause;
92import org.apache.asterix.lang.common.clause.LimitClause;
93import org.apache.asterix.lang.common.clause.OrderbyClause;
94import org.apache.asterix.lang.common.clause.UpdateClause;
95import org.apache.asterix.lang.common.clause.WhereClause;
96import org.apache.asterix.lang.common.context.RootScopeFactory;
97import org.apache.asterix.lang.common.context.Scope;
98import org.apache.asterix.lang.common.expression.AbstractAccessor;
99import org.apache.asterix.lang.common.expression.CallExpr;
100import org.apache.asterix.lang.common.expression.FieldAccessor;
101import org.apache.asterix.lang.common.expression.FieldBinding;
102import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
103import org.apache.asterix.lang.common.expression.IfExpr;
104import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700105import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700106import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +0300107import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700108import org.apache.asterix.lang.common.expression.LiteralExpr;
109import org.apache.asterix.lang.common.expression.OperatorExpr;
110import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
111import org.apache.asterix.lang.common.expression.QuantifiedExpression;
112import org.apache.asterix.lang.common.expression.RecordConstructor;
113import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
114import org.apache.asterix.lang.common.expression.TypeExpression;
115import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
116import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700117import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
118import org.apache.asterix.lang.common.expression.VariableExpr;
119import org.apache.asterix.lang.common.literal.DoubleLiteral;
120import org.apache.asterix.lang.common.literal.FalseLiteral;
121import org.apache.asterix.lang.common.literal.FloatLiteral;
122import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700123import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700124import org.apache.asterix.lang.common.literal.NullLiteral;
125import org.apache.asterix.lang.common.literal.StringLiteral;
126import org.apache.asterix.lang.common.literal.TrueLiteral;
127import org.apache.asterix.lang.common.parser.ScopeChecker;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -0700128import org.apache.asterix.lang.common.statement.AdapterDropStatement;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -0700129import org.apache.asterix.lang.common.statement.AnalyzeStatement;
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -0700130import org.apache.asterix.lang.common.statement.AnalyzeDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700131import org.apache.asterix.lang.common.statement.CompactStatement;
132import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800133import org.apache.asterix.lang.common.statement.StartFeedStatement;
134import org.apache.asterix.lang.common.statement.StopFeedStatement;
Ian Maxon38fe9402020-01-29 19:27:40 -0800135import org.apache.asterix.lang.common.statement.CreateAdapterStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700136import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
137import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
138import org.apache.asterix.lang.common.statement.CreateFeedStatement;
139import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
140import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800141import org.apache.asterix.lang.common.statement.CreateSynonymStatement;
Rui Guoe6986dd2020-11-25 19:50:06 -0800142import org.apache.asterix.lang.common.statement.CreateFullTextFilterStatement;
143import org.apache.asterix.lang.common.statement.CreateFullTextConfigStatement;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700144import org.apache.asterix.lang.common.statement.CreateViewStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700145import org.apache.asterix.lang.common.statement.DatasetDecl;
146import org.apache.asterix.lang.common.statement.DataverseDecl;
147import org.apache.asterix.lang.common.statement.DataverseDropStatement;
148import org.apache.asterix.lang.common.statement.DeleteStatement;
149import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700150import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700151import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
152import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300153import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700154import org.apache.asterix.lang.common.statement.FunctionDecl;
155import org.apache.asterix.lang.common.statement.FunctionDropStatement;
156import org.apache.asterix.lang.common.statement.IndexDropStatement;
Rui Guoe6986dd2020-11-25 19:50:06 -0800157import org.apache.asterix.lang.common.statement.FullTextFilterDropStatement;
158import org.apache.asterix.lang.common.statement.FullTextConfigDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700159import org.apache.asterix.lang.common.statement.InsertStatement;
160import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
161import org.apache.asterix.lang.common.statement.LoadStatement;
Peeyush Gupta066fd562023-04-30 13:03:00 -0700162import org.apache.asterix.lang.common.statement.CopyStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700163import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
164import org.apache.asterix.lang.common.statement.NodegroupDecl;
165import org.apache.asterix.lang.common.statement.Query;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700166import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800167import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700168import org.apache.asterix.lang.common.statement.TypeDecl;
169import org.apache.asterix.lang.common.statement.TypeDropStatement;
170import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800171import org.apache.asterix.lang.common.statement.UpsertStatement;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700172import org.apache.asterix.lang.common.statement.ViewDecl;
173import org.apache.asterix.lang.common.statement.ViewDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700174import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800175import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700176import org.apache.asterix.lang.common.struct.QuantifiedPair;
177import org.apache.asterix.lang.common.struct.VarIdentifier;
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -0800178import org.apache.asterix.lang.common.util.DatasetDeclParametersUtil;
Tin Vu4f1090d2021-09-21 02:06:32 -0700179import org.apache.asterix.lang.common.util.ExpressionUtils;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700180import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700181import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
182import org.apache.asterix.lang.sqlpp.clause.FromClause;
183import org.apache.asterix.lang.sqlpp.clause.FromTerm;
184import org.apache.asterix.lang.sqlpp.clause.HavingClause;
185import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700186import org.apache.asterix.lang.sqlpp.clause.Projection;
187import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
188import org.apache.asterix.lang.sqlpp.clause.SelectClause;
189import org.apache.asterix.lang.sqlpp.clause.SelectElement;
190import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
191import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
192import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800193import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700194import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700195import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700196import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700197import org.apache.asterix.lang.sqlpp.optype.JoinType;
198import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -0700199import org.apache.asterix.lang.sqlpp.optype.UnnestType;
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700200import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser;
201import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingElement;
202import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSet;
203import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSets;
204import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.RollupCube;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700205import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700206import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
207import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700208import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700209import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700210import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Peeyush Gupta6ba5a262023-04-13 11:08:53 -0700211import org.apache.asterix.metadata.bootstrap.MetadataBuiltinEntities;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -0700212import org.apache.asterix.metadata.utils.MetadataConstants;
Tin Vu4f1090d2021-09-21 02:06:32 -0700213import org.apache.asterix.om.exceptions.TypeMismatchException;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800214import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700215import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700216import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700217import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700218import org.apache.hyracks.algebricks.common.utils.Pair;
219import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800220import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700221import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700222import org.apache.hyracks.algebricks.core.algebra.expressions.HashJoinExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700223import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700224import org.apache.hyracks.algebricks.core.algebra.expressions.JoinProductivityAnnotation;
225import org.apache.hyracks.algebricks.core.algebra.expressions.PredicateCardinalityAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700226import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800227import org.apache.hyracks.api.exceptions.IWarningCollector;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700228import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700229import org.apache.hyracks.api.exceptions.Warning;
Dmitry Lychagin70bf47a2020-10-22 13:25:57 -0700230import org.apache.hyracks.dataflow.common.data.partition.range.RangeMap;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800231import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700232import org.apache.hyracks.util.StringUtil;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700233import java.util.regex.Matcher;
234import java.util.regex.Pattern;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700235
Yingyi Bucaea8f02015-11-16 15:12:15 -0800236class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700237
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800238 // tokens parsed as identifiers
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700239 private static final String CUBE = "CUBE";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800240 private static final String CURRENT = "CURRENT";
Dmitry Lychagin33c77f92021-07-21 15:59:04 -0700241 private static final String DEFAULT = "DEFAULT";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800242 private static final String EXCLUDE = "EXCLUDE";
Ali Alsuliman931e7382021-08-08 21:55:59 +0300243 private static final String INCLUDE = "INCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700244 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800245 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychagin914983f2021-09-30 15:11:11 -0700246 private static final String FOREIGN = "FOREIGN";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700247 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800248 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700249 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700250 private static final String LAST = "LAST";
Glenn67fd1f32021-02-25 16:04:49 -0800251 private static final String META = "META";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800252 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700253 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800254 private static final String OTHERS = "OTHERS";
255 private static final String PARTITION = "PARTITION";
256 private static final String PRECEDING = "PRECEDING";
257 private static final String RANGE = "RANGE";
Dmitry Lychagin914983f2021-09-30 15:11:11 -0700258 private static final String REFERENCES = "REFERENCES";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700259 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700260 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800261 private static final String ROW = "ROW";
262 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700263 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800264 private static final String TIES = "TIES";
265 private static final String UNBOUNDED = "UNBOUNDED";
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700266 private static final String REPLACE = "REPLACE";
Ian Maxon38fe9402020-01-29 19:27:40 -0800267 private static final String RETURNS = "RETURNS";
Rui Guoe6986dd2020-11-25 19:50:06 -0800268 private static final String CONFIG = "CONFIG";
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -0700269 private static final String STATISTICS = "STATISTICS";
Rui Guoe6986dd2020-11-25 19:50:06 -0800270
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800271
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700272 private static final String INT_TYPE_NAME = "int";
Dmitry Lychagin541652082020-11-09 14:04:53 -0800273 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 -0700274
Till Westmann7199a562016-09-17 16:07:32 -0700275 // error configuration
276 protected static final boolean REPORT_EXPECTED_TOKENS = false;
277
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700278 private int externalVarCounter;
279
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800280 private DataverseName defaultDataverse;
281
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700282 private SqlppGroupingSetsParser groupingSetsParser;
283
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700284 private final WarningCollector warningCollector = new WarningCollector();
285
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700286 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
287
Yingyi Bu391f09e2015-10-29 13:49:39 -0700288 private static class IndexParams {
289 public IndexType type;
290 public int gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800291 public String fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700292
Rui Guoe6986dd2020-11-25 19:50:06 -0800293 public IndexParams(IndexType type, int gramLength, String fullTextConfig) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700294 this.type = type;
295 this.gramLength = gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800296 this.fullTextConfig = fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700297 }
Ali Alsuliman69ce7d82021-11-01 15:00:16 -0700298 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700299
300 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800301 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700302 public String library;
303 public String function;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -0600304 public Token hintToken;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700305 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700306 }
307
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700308 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
309 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
310 switch (hintToken.hint) {
311 case VAL_FILE_HINT:
312 File[] valFiles = new File[splits.length];
313 for (int k=0; k<splits.length; k++) {
314 valFiles[k] = new File(splits[k]);
315 }
316 return new FieldValFileDataGen(valFiles);
317 case VAL_FILE_SAME_INDEX_HINT:
318 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
319 case LIST_VAL_FILE_HINT:
320 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
321 case LIST_HINT:
322 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
323 case INTERVAL_HINT:
324 FieldIntervalDataGen.ValueType vt;
325 switch (splits[0]) {
326 case "int":
327 vt = FieldIntervalDataGen.ValueType.INT;
328 break;
329 case "long":
330 vt = FieldIntervalDataGen.ValueType.LONG;
331 break;
332 case "float":
333 vt = FieldIntervalDataGen.ValueType.FLOAT;
334 break;
335 case "double":
336 vt = FieldIntervalDataGen.ValueType.DOUBLE;
337 break;
338 default:
339 throw new SqlppParseException(getSourceLocation(hintToken),
340 "Unknown type for interval data gen: " + splits[0]);
341 }
342 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
343 case INSERT_RAND_INT_HINT:
344 return new InsertRandIntDataGen(splits[0], splits[1]);
345 case DATE_BETWEEN_YEARS_HINT:
346 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
347 case DATETIME_BETWEEN_YEARS_HINT:
348 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
349 case DATETIME_ADD_RAND_HOURS_HINT:
350 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
351 case AUTO_HINT:
352 return new AutoDataGen(splits[0]);
353 default:
354 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700355 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700356 }
357
Till Westmann7199a562016-09-17 16:07:32 -0700358 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700359 this(new StringReader(s));
360 super.setInput(s);
361 }
362
Ali Alsuliman69ce7d82021-11-01 15:00:16 -0700363 public static void main(String[] args) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700364 File file = new File(args[0]);
365 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
366 SQLPPParser parser = new SQLPPParser(fis);
367 List<Statement> st = parser.parse();
368 //st.accept(new SQLPPPrintVisitor(), 0);
369 }
370
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800371 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800372 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700373 return parseImpl(new ParseFunction<List<Statement>>() {
374 @Override
375 public List<Statement> parse() throws ParseException {
376 return SQLPPParser.this.Statement();
377 }
378 });
379 }
380
Dmitry Lychagin62f0beb2020-09-08 11:44:19 -0700381 @Override
382 public Expression parseExpression() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700383 return parseImpl(new ParseFunction<Expression>() {
384 @Override
385 public Expression parse() throws ParseException {
386 return SQLPPParser.this.Expression();
387 }
388 });
389 }
390
391 private static Expression parseExpression(String text) throws CompilationException {
392 return new SQLPPParser(text).parseExpression();
393 }
394
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800395 @Override
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700396 public List<String> parseMultipartIdentifier() throws CompilationException {
397 return parseImpl(new ParseFunction<List<String>>() {
398 @Override
399 public List<String> parse() throws ParseException {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700400 return SQLPPParser.this.MultipartIdentifier().first;
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700401 }
402 });
403 }
404
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800405 private List<String> parseParenthesizedIdentifierList() throws CompilationException {
406 return parseImpl(new ParseFunction<List<String>>() {
407 @Override
408 public List<String> parse() throws ParseException {
409 return SQLPPParser.this.ParenthesizedIdentifierList();
410 }
411 });
412 }
413
414 private static List<String> parseParenthesizedIdentifierList(String text) throws CompilationException {
415 return new SQLPPParser(text).parseParenthesizedIdentifierList();
416 }
417
Vijay Sarathy691e93b2022-11-20 14:31:47 -0800418 private Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> parseHashJoinParams() throws CompilationException {
419 return parseImpl(new ParseFunction<Pair<HashJoinExpressionAnnotation.BuildOrProbe, String>>() {
420 @Override
421 public Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> parse() throws ParseException {
422 return SQLPPParser.this.buildOrProbeParenthesizedIdentifier();
423 }
424 });
425 }
426
427 private static Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> parseHashJoinParams(String text) throws CompilationException {
428 return new SQLPPParser(text).parseHashJoinParams();
429 }
430
431 private String parseBroadcastJoinParams() throws CompilationException {
432 return parseImpl(new ParseFunction<String>() {
433 @Override
434 public String parse() throws ParseException {
435 return SQLPPParser.this.parenthesizedIdentifier();
436 }
437 });
438 }
439
440 private static String parseBroadcastJoinParams(String text) throws CompilationException {
441 return new SQLPPParser(text).parseBroadcastJoinParams();
442 }
443
Tin Vu4f1090d2021-09-21 02:06:32 -0700444 private List<Literal> parseParenthesizedLiteralList() throws CompilationException {
445 return parseImpl(new ParseFunction<List<Literal>>() {
446 @Override
447 public List<Literal> parse() throws ParseException {
448 return SQLPPParser.this.ParenthesizedLiteralList();
449 }
450 });
451 }
452
453 private static List<Literal> parseParenthesizedLiteralList(String text) throws CompilationException {
454 return new SQLPPParser(text).parseParenthesizedLiteralList();
455 }
456
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700457 @Override
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700458 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames, boolean isStored)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800459 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800460 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800461 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800462 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800463 DataverseName dataverse = defaultDataverse;
464 defaultDataverse = signature.getDataverseName();
465 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800466 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
467 for (String paramName : paramNames) {
468 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
469 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800470 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
471 removeCurrentScope();
472 defaultDataverse = dataverse;
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700473 return new FunctionDecl(signature, paramVars, functionBodyExpr, isStored);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800474 }
475 });
476 }
477
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700478 @Override
479 public ViewDecl parseViewBody(DatasetFullyQualifiedName viewName) throws CompilationException {
480 return parseImpl(new ParseFunction<ViewDecl>() {
481 @Override
482 public ViewDecl parse() throws ParseException {
483 DataverseName dataverse = defaultDataverse;
484 defaultDataverse = viewName.getDataverseName();
485 createNewScope();
486 Expression viewBodyExpr = SQLPPParser.this.ViewBody();
487 removeCurrentScope();
488 defaultDataverse = dataverse;
489 return new ViewDecl(viewName, viewBodyExpr);
490 }
491 });
492 }
493
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700494 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700495 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700496 hintCollector.clear();
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800497 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700498 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700499 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700500 } catch (SqlppParseException e) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700501 if (e.getCause() instanceof AlgebricksException) {
502 AlgebricksException cause = (AlgebricksException) e.getCause();
503 if (cause.getError().isPresent() && cause.getError().get() instanceof ErrorCode) {
504 throw new CompilationException((ErrorCode) cause.getError().get(), e.getSourceLocation(),
505 cause.getParams());
506 }
507 }
508 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(),
509 LogRedactionUtil.userData(getMessage(e)));
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700510 } catch (ParseException e) {
511 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700512 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700513 // 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 -0700514 // 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 -0700515 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700516 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700517 } finally {
518 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700519 }
520 }
Till Westmann7199a562016-09-17 16:07:32 -0700521
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700522 @FunctionalInterface
523 private interface ParseFunction<T> {
524 T parse() throws ParseException;
525 }
526
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700527 @Override
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800528 public void getWarnings(IWarningCollector outWarningCollector) {
529 warningCollector.getWarnings(outWarningCollector);
530 }
531
532 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700533 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
534 warningCollector.getWarnings(outWarnings, maxWarnings);
535 }
536
537 @Override
538 public long getTotalWarningsCount() {
539 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700540 }
541
Till Westmann7199a562016-09-17 16:07:32 -0700542 protected String getMessage(ParseException pe) {
543 Token currentToken = pe.currentToken;
544 if (currentToken == null) {
545 return pe.getMessage();
546 }
547 int[][] expectedTokenSequences = pe.expectedTokenSequences;
548 String[] tokenImage = pe.tokenImage;
549 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
550 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
551 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
552 Token tok = currentToken.next;
553 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700554 StringBuilder message = new StringBuilder(128);
555 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700556 for (int i = 0; i < maxSize; i++) {
557 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700558 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700559 }
560 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700561 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700562 break;
563 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700564 final String fixedTokenImage = tokenImage[tok.kind];
565 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700566 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700567 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700568 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700569 tok = tok.next;
570 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700571 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700572 if (REPORT_EXPECTED_TOKENS) {
573 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700574 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700575 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700576 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700577 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700578 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700579 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700580 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700581 }
582
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700583 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700584 return
585 token == null ? null :
586 token.sourceLocation != null ? token.sourceLocation :
587 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700588 }
589
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700590 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
591 expr.setSourceLocation(getSourceLocation(token));
592 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700593 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800594
595 private boolean isToken(String image) {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700596 return isToken(token, image);
597 }
598
599 private static boolean isToken(Token token, String image) {
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800600 return token.image.equalsIgnoreCase(image);
601 }
602
603 private void expectToken(String image) throws SqlppParseException {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700604 expectToken(token, image);
605 }
606
607 private static void expectToken(Token token, String image) throws SqlppParseException {
608 if (!isToken(token, image)) {
609 throw createUnexpectedTokenError(token);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800610 }
611 }
612
613 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700614 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700615 }
616
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700617 private static SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700618 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800619 }
620
621 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700622 return createUnexpectedTokenError(token, expected);
623 }
624
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700625 private static SqlppParseException createUnexpectedTokenError(Token t, String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700626 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800627 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700628 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800629 }
630
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700631 private boolean laToken(int idx, int kind) {
632 Token t = getToken(idx);
633 return t.kind == kind;
634 }
635
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800636 private boolean laToken(int idx, int kind, String image) {
637 Token t = getToken(idx);
638 return t.kind == kind && t.image.equalsIgnoreCase(image);
639 }
640
641 private boolean laIdentifier(int idx, String image) {
642 return laToken(idx, IDENTIFIER, image);
643 }
644
645 private boolean laIdentifier(String image) {
646 return laIdentifier(1, image);
647 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700648
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700649 private Token fetchHint(Token token, SqlppHint... expectedHints) {
650 Token hintToken = token.specialToken;
651 if (hintToken == null) {
652 return null;
653 }
654 SourceLocation sourceLoc = getSourceLocation(hintToken);
655 hintCollector.remove(sourceLoc);
656 if (hintToken.hint == null) {
657 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
658 return null;
659 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
660 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
661 return null;
662 } else {
663 return hintToken;
664 }
665 }
666
667 private void reportUnclaimedHints() {
668 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
669 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
670 }
671 }
672
673 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
674 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
675 }
676
677 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700678 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500679 warningCollector.warn(Warning.of(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
Ali Alsulimanaa118862019-09-10 20:55:45 -0700680 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700681 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700682
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800683 private IExpressionAnnotation parseExpressionAnnotation(Token hintToken) {
684 // placeholder for the annotation that should be returned if this hint's parameters cannot be parsed
685 IExpressionAnnotation onParseErrorReturn = null;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700686 double selectivity, cardinality, productivity;
687 Pattern number = Pattern.compile("\\d+\\.\\d+");
688 Pattern stringNumber = Pattern.compile("\\w+\\s+\\d+\\.\\d+");
689 Pattern lessThanOnePat = Pattern.compile("0\\.\\d+");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800690 try {
691 switch (hintToken.hint) {
Vijay Sarathy7ba271a2023-03-09 08:07:43 -0800692 case SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT:
693 if (hintToken.hintParams == null) {
694 throw new SqlppParseException(getSourceLocation(hintToken), "Expected selectivity value");
695 }
696 else {
697 selectivity = 1.0; // uninitialized
698 Matcher mat = lessThanOnePat.matcher(hintToken.hintParams);
699 if (mat.find()) {
700 selectivity = Double.parseDouble (mat.group());
701 }
702 else {
Vijay Sarathyc70bae02023-03-13 13:24:05 -0700703 throw new SqlppParseException(getSourceLocation(hintToken), "Selectivity has to be a decimal value greater than 0 and less than 1");
Vijay Sarathy7ba271a2023-03-09 08:07:43 -0800704 }
705 return new PredicateCardinalityAnnotation(selectivity);
706 }
707 case JOIN_PREDICATE_PRODUCTIVITY_HINT:
708 if (hintToken.hintParams == null) {
709 throw new SqlppParseException(getSourceLocation(hintToken), "Expected productivity collection name and value");
710 }
711 else {
712 productivity = 1.0; // uninitialized
713 String leftSideDataSet = null;
714 Matcher StringNum = stringNumber.matcher(hintToken.hintParams);
715 if (StringNum.find()) {
716 String matchedGroup = StringNum.group();
717 Pattern var = Pattern.compile("[a-zA-Z]\\w*"); // any word character [a-zA-Z_0-9]
718 Matcher matVar = var.matcher(matchedGroup);
719 if (matVar.find()) {
720 leftSideDataSet = matVar.group();
721 }
722 else {
723 throw new SqlppParseException(getSourceLocation(hintToken), "Expected productivity collection name");
724 }
725 Matcher numMat = number.matcher(matchedGroup);
726 if (numMat.find()) {
727 productivity = Double.parseDouble (numMat.group());
728 }
729 else {
Vijay Sarathyc70bae02023-03-13 13:24:05 -0700730 throw new SqlppParseException(getSourceLocation(hintToken), "Productivity has to be a decimal value greater than 0");
Vijay Sarathy7ba271a2023-03-09 08:07:43 -0800731 }
732 }
733 else {
734 throw new SqlppParseException(getSourceLocation(hintToken), "Invalid format for productivity values");
735 }
736 // attach hint to global scope
737 return new JoinProductivityAnnotation (productivity, leftSideDataSet);
738 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800739 case HASH_BROADCAST_JOIN_HINT:
Vijay Sarathy691e93b2022-11-20 14:31:47 -0800740 if (hintToken.hintParams == null) {
741 return new BroadcastExpressionAnnotation(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
742 }
743 else {
744 // if parameter parsing fails then ignore this hint.
745 String name = parseBroadcastJoinParams(hintToken.hintParams);
746 return new BroadcastExpressionAnnotation(name);
747 }
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700748 case HASH_JOIN_HINT:
Vijay Sarathy691e93b2022-11-20 14:31:47 -0800749 if (hintToken.hintParams == null) {
750 throw new SqlppParseException(getSourceLocation(hintToken), "Expected hash join build/probe collection name");
751 }
752 else {
753 // if parameter parsing fails then ignore this hint.
754 Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> pair = parseHashJoinParams(hintToken.hintParams);
755 return new HashJoinExpressionAnnotation(pair);
756 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800757 case INDEXED_NESTED_LOOP_JOIN_HINT:
758 if (hintToken.hintParams == null) {
759 return IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
760 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800761 // if parameter parsing fails then return hint annotation without parameters
762 onParseErrorReturn = IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800763 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
764 return IndexedNLJoinExpressionAnnotation.newInstance(indexNames);
765 }
766 case RANGE_HINT:
767 Expression rangeExpr = parseExpression(hintToken.hintParams);
768 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
769 return new RangeAnnotation(rangeMap);
770 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
771 if (hintToken.hintParams == null) {
772 return SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE_ANY_INDEX;
773 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800774 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800775 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
776 return SkipSecondaryIndexSearchExpressionAnnotation.newInstance(indexNames);
777 }
Tin Vu4f1090d2021-09-21 02:06:32 -0700778 case SPATIAL_JOIN_HINT:
779 List<Literal> hintValues = parseParenthesizedLiteralList(hintToken.hintParams);
780
781 // Handle exceptions
782 if (hintValues.size() != 6) {
783 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. 6 arguments are required.",
784 hintToken.hint.toString()));
785 }
786
787 for (int i = 0; i < 4; i++) {
788 Literal lit = hintValues.get(i);
789 if ((lit.getLiteralType() != Literal.Type.DOUBLE)
790 && (lit.getLiteralType() != Literal.Type.FLOAT)
791 && (lit.getLiteralType() != Literal.Type.LONG)
792 && (lit.getLiteralType() != Literal.Type.INTEGER)){
793 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. Numeric value is required for first 4 arguments.",
794 hintToken.hint.toString()));
795 }
796 }
797
798 for (int i = 4; i < 6; i++) {
799 Literal lit = hintValues.get(i);
800 if ((lit.getLiteralType() != Literal.Type.LONG)
801 && (lit.getLiteralType() != Literal.Type.INTEGER)) {
802 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. Long/int is required for last 2 arguments.",
803 hintToken.hint.toString()));
804 }
805 }
806
807 try {
808 double minX = ExpressionUtils.getDoubleValue(hintValues.get(0));
809 double minY = ExpressionUtils.getDoubleValue(hintValues.get(1));
810 double maxX = ExpressionUtils.getDoubleValue(hintValues.get(2));
811 double maxY = ExpressionUtils.getDoubleValue(hintValues.get(3));
812 int numRows = (int) ExpressionUtils.getLongValue(hintValues.get(4));
813 int numColumns = (int) ExpressionUtils.getLongValue(hintValues.get(5));
814 SpatialJoinAnnotation spatialJoinAnn = new SpatialJoinAnnotation(minX, minY, maxX, maxY, numRows, numColumns);
815 return spatialJoinAnn;
816 } catch (TypeMismatchException e) {
817 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
818 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800819 case USE_SECONDARY_INDEX_SEARCH_HINT:
820 if (hintToken.hintParams == null) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800821 throw new SqlppParseException(getSourceLocation(hintToken), "Expected index name(s)");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800822 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800823 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800824 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
825 return SecondaryIndexSearchPreferenceAnnotation.newInstance(indexNames);
826 }
827 default:
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800828 throw new SqlppParseException(getSourceLocation(hintToken), "Unexpected hint");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800829 }
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800830 } catch (SqlppParseException e) {
831 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500832 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800833 hintToken.hint.toString(), e.getMessage()));
834 }
835 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800836 } catch (CompilationException e) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800837 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500838 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800839 hintToken.hint.toString(), e.getMessage()));
840 }
841 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800842 }
843 }
844
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700845 private void ensureNoTypeDeclsInFunction(String fnName, List<Pair<VarIdentifier, TypeExpression>> paramList,
846 TypeExpression returnType, Token startToken) throws SqlppParseException {
847 for (Pair<VarIdentifier, TypeExpression> p : paramList) {
848 if (p.second != null) {
849 String paramName = SqlppVariableUtil.toUserDefinedName(p.first.getValue());
850 throw new SqlppParseException(getSourceLocation(startToken),
851 "Unexpected type declaration for parameter " + paramName + " in function " + fnName);
852 }
853 }
854 if (returnType != null) {
855 throw new SqlppParseException(getSourceLocation(startToken),
856 "Unexpected return type declaration for function " + fnName);
857 }
858 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -0700859
860 private void ensureIntegerLiteral(LiteralExpr expr, String errorPrefix) throws SqlppParseException {
861 Literal lit = expr.getValue();
862 if (lit.getLiteralType() != Literal.Type.INTEGER && lit.getLiteralType() != Literal.Type.LONG) {
863 throw new SqlppParseException(expr.getSourceLocation(), errorPrefix + " should be an INTEGER");
864 }
865 }
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700866
867 private DataverseName createDataverseName(List<String> parts, int fromIndex, int toIndex, Token startToken)
868 throws SqlppParseException {
869 try {
870 return DataverseName.create(parts, fromIndex, toIndex);
871 } catch (AsterixException e) {
872 SqlppParseException pe = new SqlppParseException(getSourceLocation(startToken), e.getMessage());
873 pe.initCause(e);
874 throw pe;
875 }
876 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700877}
878
879PARSER_END(SQLPPParser)
880
Yingyi Bu391f09e2015-10-29 13:49:39 -0700881List<Statement> Statement() throws ParseException:
882{
883 scopeStack.push(RootScopeFactory.createRootScope(this));
884 List<Statement> decls = new ArrayList<Statement>();
885 Statement stmt = null;
886}
887{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300888 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700889 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300890 {
891 decls.add(stmt);
892 }
893 )?
894 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700895 )*
896 <EOF>
897 {
898 return decls;
899 }
900}
901
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700902Statement ExplainStatement() throws ParseException:
903{
904 Statement stmt = null;
905 Token explainToken = null;
906}
907{
908 ( <EXPLAIN> { explainToken = token; } )?
909 stmt = SingleStatement()
910 {
911 if (explainToken != null) {
912 if (stmt.getKind() == Statement.Kind.QUERY) {
913 ((Query)stmt).setExplain(true);
914 } else {
915 throw new SqlppParseException(getSourceLocation(explainToken),
916 "EXPLAIN is not supported for this kind of statement");
917 }
918 }
919 return stmt;
920 }
921}
922
Yingyi Bu391f09e2015-10-29 13:49:39 -0700923Statement SingleStatement() throws ParseException:
924{
925 Statement stmt = null;
926}
927{
928 (
929 stmt = DataverseDeclaration()
930 | stmt = FunctionDeclaration()
931 | stmt = CreateStatement()
932 | stmt = LoadStatement()
Peeyush Gupta066fd562023-04-30 13:03:00 -0700933 | stmt = CopyStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700934 | stmt = DropStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700935 | stmt = SetStatement()
936 | stmt = InsertStatement()
937 | stmt = DeleteStatement()
938 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800939 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700940 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700941 | stmt = CompactStatement()
Dmitry Lychaginf3eec282022-06-10 12:44:25 -0700942 | stmt = AnalyzeStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700943 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700944 )
945 {
946 return stmt;
947 }
948}
949
950DataverseDecl DataverseDeclaration() throws ParseException:
951{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700952 Token startToken = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700953 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700954}
955{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700956 <USE> { startToken = token; } dvName = DataverseName()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700957 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700958 defaultDataverse = dvName;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800959 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700960 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700961 }
962}
963
964Statement CreateStatement() throws ParseException:
965{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700966 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700967 Statement stmt = null;
968}
969{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700970 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700971 (
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700972 stmt = CreateOrReplaceStatement(startToken)
973 | stmt = CreateTypeStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800974 | stmt = CreateNodegroupStatement(startToken)
975 | stmt = CreateDatasetStatement(startToken)
976 | stmt = CreateIndexStatement(startToken)
977 | stmt = CreateDataverseStatement(startToken)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700978 | stmt = CreateFunctionStatement(startToken, false)
Ian Maxon38fe9402020-01-29 19:27:40 -0800979 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800980 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800981 | stmt = CreateFeedStatement(startToken)
982 | stmt = CreateFeedPolicyStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -0800983 | stmt = CreateFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700984 | stmt = CreateViewStatement(startToken, false)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700985 )
986 {
987 return stmt;
988 }
989}
990
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700991Statement CreateOrReplaceStatement(Token startStmtToken) throws ParseException:
992{
993 Statement stmt = null;
994 Token replaceToken = null;
995}
996{
997 <OR> <IDENTIFIER> { replaceToken = token; }
998 (
999 stmt = CreateFunctionStatement(startStmtToken, true)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001000 | stmt = CreateViewStatement(startStmtToken, true)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001001 )
1002 {
1003 // check expected token here to make the grammar extension plugin happy
1004 expectToken(replaceToken, REPLACE);
1005 return stmt;
1006 }
1007}
1008
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001009TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
1010{
1011 TypeDecl stmt = null;
1012}
1013{
1014 <TYPE> stmt = TypeSpecification(startStmtToken)
1015 {
1016 return stmt;
1017 }
1018}
1019
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07001020TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001021{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001022 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001023 boolean ifNotExists = false;
1024 TypeExpression typeExpr = null;
1025}
1026{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001027 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -07001028 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001029 {
1030 boolean dgen = false;
1031 long numValues = -1;
1032 String filename = null;
1033 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
1034 if (hintToken != null) {
1035 String hintParams = hintToken.hintParams;
1036 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
1037 if (splits == null || splits.length != 2) {
1038 throw new SqlppParseException(getSourceLocation(hintToken),
1039 "Expecting /*+ dgen <filename> <numberOfItems> */");
1040 }
1041 dgen = true;
1042 filename = splits[0];
1043 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001044 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001045 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
1046 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
1047 return addSourceLocation(stmt, startStmtToken);
1048 }
1049}
1050
1051NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
1052{
1053 NodegroupDecl stmt = null;
1054}
1055{
1056 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
1057 {
1058 return stmt;
1059 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001060}
1061
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001062NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001063{
1064 String name = null;
1065 String tmp = null;
1066 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001067 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001068}
1069{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001070 name = Identifier() ifNotExists = IfNotExists()
1071 <ON> tmp = Identifier()
1072 {
1073 ncNames.add(new Identifier(tmp));
1074 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001075 ( <COMMA> tmp = Identifier()
1076 {
1077 ncNames.add(new Identifier(tmp));
1078 }
1079 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001080 {
1081 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
1082 return addSourceLocation(stmt, startStmtToken);
1083 }
1084}
1085
1086void Dataset() throws ParseException:
1087{
1088}
1089{
1090 (<DATASET>|<COLLECTION>)
1091}
1092
Ali Alsuliman7e5c5742022-11-18 12:21:13 -08001093void DatasetToken() throws ParseException:
1094{
1095}
1096{
1097 Dataset()
1098}
1099
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001100DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
1101{
1102 DatasetDecl stmt = null;
1103}
1104{
1105 (
1106 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
1107 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
1108 )
1109 {
1110 return stmt;
1111 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001112}
1113
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001114DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001115{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001116 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001117 boolean ifNotExists = false;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001118 TypeExpression typeExpr = null;
1119 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001120 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001121 Triple<List<Integer>, List<List<String>>, List<TypeExpression>> primaryKeyFieldsWithTypes = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001122 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001123 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001124 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001125 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -08001126 RecordConstructor withRecord = null;
Peeyush Gupta0814b0e2023-05-03 11:08:12 -07001127 SelectExpression selectExpr = null;
1128 Query query = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001129}
1130{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001131 nameComponents = QualifiedName()
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001132 (typeExpr = DatasetTypeSpecification())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001133 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001134 { String name; }
1135 <WITH>
1136 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001137 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001138 if (!name.equalsIgnoreCase("meta")){
1139 throw new SqlppParseException(getSourceLocation(startStmtToken),
1140 "We can only support one additional associated field called \"meta\".");
1141 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001142 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001143 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001144 )?
1145 ifNotExists = IfNotExists()
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001146 (LOOKAHEAD(3) primaryKeyFieldsWithTypes = PrimaryKeyWithType()
1147 | primaryKeyFields = PrimaryKey())
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001148 (<AUTOGENERATED> { autogenerated = true; } )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001149 ( <HINTS> hints = Properties() )?
1150 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
1151 ( <WITH> withRecord = RecordConstructor() )?
Peeyush Gupta0814b0e2023-05-03 11:08:12 -07001152 ( <AS> selectExpr = SelectExpression(false) )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001153 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001154 try {
Peeyush Gupta0814b0e2023-05-03 11:08:12 -07001155 if (selectExpr != null) {
1156 query = new Query();
1157 query.setBody(selectExpr);
1158 query.setSourceLocation(selectExpr.getSourceLocation());
1159 }
1160
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001161 if (typeExpr == null) {
1162 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFieldsWithTypes.second,
1163 primaryKeyFieldsWithTypes.first, autogenerated, filterField == null? null : filterField.first,
1164 filterField == null? null : filterField.second, primaryKeyFieldsWithTypes.third);
1165 final TypeReferenceExpression anyObjectReference = new TypeReferenceExpression(
1166 new Pair(MetadataBuiltinEntities.ANY_OBJECT_DATATYPE.getDataverseName(),
1167 new Identifier(MetadataBuiltinEntities.ANY_OBJECT_DATATYPE.getDatatypeName())));
1168 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, anyObjectReference, null, hints,
Peeyush Gupta0814b0e2023-05-03 11:08:12 -07001169 DatasetType.INTERNAL, idd, withRecord, ifNotExists, query);
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001170 return addSourceLocation(stmt, startStmtToken);
1171 } else {
1172 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
1173 filterField == null? null : filterField.first, filterField == null? null : filterField.second);
1174 DatasetDeclParametersUtil.adjustInlineTypeDecl(typeExpr, primaryKeyFields.second, primaryKeyFields.first, false);
1175 if (metaTypeExpr != null) {
1176 DatasetDeclParametersUtil.adjustInlineTypeDecl(metaTypeExpr, primaryKeyFields.second, primaryKeyFields.first,
1177 true);
1178 }
1179 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr, hints,
Peeyush Gupta0814b0e2023-05-03 11:08:12 -07001180 DatasetType.INTERNAL, idd, withRecord, ifNotExists, query);
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001181 return addSourceLocation(stmt, startStmtToken);
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -08001182 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001183 } catch (CompilationException e) {
1184 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1185 }
1186 }
1187}
1188
1189DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
1190{
1191 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001192 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001193 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001194 String adapterName = null;
1195 Map<String,String> properties = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001196 Map<String,String> hints = new HashMap<String,String>();
1197 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001198 RecordConstructor withRecord = null;
1199}
1200{
1201 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001202 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001203 ifNotExists = IfNotExists()
1204 <USING> adapterName = AdapterName() properties = Configuration()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001205 ( <HINTS> hints = Properties() )?
1206 ( <WITH> withRecord = RecordConstructor() )?
1207 {
1208 ExternalDetailsDecl edd = new ExternalDetailsDecl();
1209 edd.setAdapter(adapterName);
1210 edd.setProperties(properties);
1211 try {
Murtadha Hubail353e95f2020-08-24 22:18:04 +03001212 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null, hints, DatasetType.EXTERNAL,
1213 edd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001214 return addSourceLocation(stmt, startStmtToken);
1215 } catch (CompilationException e) {
1216 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1217 }
1218 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001219}
1220
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001221TypeExpression DatasetTypeSpecification() throws ParseException:
1222{
1223 TypeExpression typeExpr = null;
1224}
1225{
1226 (
1227 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
1228 | typeExpr = DatasetReferenceTypeSpecification()
1229 )
1230 {
1231 return typeExpr;
1232 }
1233}
1234
1235TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
1236{
1237 TypeExpression typeExpr = null;
1238}
1239{
1240 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
1241 {
1242 return typeExpr;
1243 }
1244}
1245
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001246RecordTypeDefinition DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001247{
1248 RecordTypeDefinition recordTypeDef = null;
1249 RecordTypeDefinition.RecordKind recordKind = null;
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001250 Token startToken = null, recordKindToken = null;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001251}
1252{
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001253 <LEFTPAREN> { startToken = token; } recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001254 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
1255 {
1256 if (recordKind == null) {
1257 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
1258 } else if (!allowRecordKindModifier) {
1259 throw createUnexpectedTokenError(recordKindToken);
1260 }
1261 recordTypeDef.setRecordKind(recordKind);
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001262 return addSourceLocation(recordTypeDef, startToken);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001263 }
1264}
1265
1266RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
1267{
1268 RecordTypeDefinition recType = new RecordTypeDefinition();
1269}
1270{
1271 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
1272 {
1273 return recType;
1274 }
1275}
1276
1277void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
1278{
1279 String fieldName;
1280 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001281 boolean nullable = true, missable = true;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001282}
1283{
1284 fieldName = Identifier()
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001285 type = TypeReference() ( <NOT> <UNKNOWN> { nullable = false; missable = false; } )?
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001286 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001287 recType.addField(fieldName, type, nullable, missable);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001288 }
1289}
1290
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001291CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
1292{
1293 CreateIndexStatement stmt = null;
1294}
1295{
1296 (
1297 <INDEX> stmt = IndexSpecification(startStmtToken)
1298 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
1299 )
1300 {
1301 return stmt;
1302 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001303}
1304
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001305CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001306{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001307 Pair<DataverseName,Identifier> nameComponents = null;
Glenn67fd1f32021-02-25 16:04:49 -08001308 String indexName = null;
1309 IndexParams indexParams = null;
1310 CreateIndexStatement.IndexedElement indexedElement = null;
1311 List<CreateIndexStatement.IndexedElement> indexedElementList = new ArrayList<CreateIndexStatement.IndexedElement>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001312 boolean enforced = false;
Glenn67fd1f32021-02-25 16:04:49 -08001313 boolean ifNotExists = false;
1314 boolean hasUnnest = false;
1315 String fullTextConfigName = null;
1316 Token startElementToken = null;
Ali Alsuliman931e7382021-08-08 21:55:59 +03001317 Boolean excludeUnknown = null;
Ali Alsuliman465a7282021-11-24 12:51:32 -08001318 Pair<Map<String, String>, Boolean> castConfigDefaultNull;
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001319 Boolean castDefaultNull = null;
Ali Alsuliman465a7282021-11-24 12:51:32 -08001320 Map<String, String> castConfig = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001321}
1322{
Ali Alsuliman8351d252017-09-24 00:43:15 -07001323 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001324 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -07001325 <ON> nameComponents = QualifiedName()
Glenn67fd1f32021-02-25 16:04:49 -08001326 <LEFTPAREN> { startElementToken = token; }
1327 indexedElement = IndexedElement(startElementToken) {
1328 indexedElementList.add(indexedElement);
1329 hasUnnest |= indexedElement.hasUnnest();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001330 }
Glenn67fd1f32021-02-25 16:04:49 -08001331 (<COMMA> { startElementToken = token; }
1332 indexedElement = IndexedElement(startElementToken) {
1333 indexedElementList.add(indexedElement);
1334 hasUnnest |= indexedElement.hasUnnest();
1335 }
1336 )*
1337 <RIGHTPAREN>
1338 ( <TYPE> indexParams = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001339 ( LOOKAHEAD({laIdentifier(EXCLUDE) || laIdentifier(INCLUDE)}) <IDENTIFIER>
1340 {
1341 if (isToken(EXCLUDE)) {
1342 excludeUnknown = true;
1343 } else if (isToken(INCLUDE)) {
1344 excludeUnknown = false;
1345 } else {
1346 throw createUnexpectedTokenError();
1347 }
1348 } <UNKNOWN> <KEY>
Ali Alsuliman931e7382021-08-08 21:55:59 +03001349 )?
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001350
Ali Alsuliman465a7282021-11-24 12:51:32 -08001351 ( <CAST><LEFTPAREN> castConfigDefaultNull = CastDefaultNull() <RIGHTPAREN>
1352 {
1353 castConfig = castConfigDefaultNull.first;
1354 castDefaultNull = castConfigDefaultNull.second;
1355 }
1356 )?
Ali Alsuliman8351d252017-09-24 00:43:15 -07001357 )
1358 {
Glenn67fd1f32021-02-25 16:04:49 -08001359 IndexType indexType;
1360 int gramLength;
1361 if (indexParams != null) {
1362 indexType = indexParams.type;
1363 gramLength = indexParams.gramLength;
1364 fullTextConfigName = indexParams.fullTextConfig;
1365 } else {
1366 indexType = hasUnnest ? IndexType.ARRAY : IndexType.BTREE;
1367 gramLength = -1;
1368 fullTextConfigName = null;
Ali Alsuliman8351d252017-09-24 00:43:15 -07001369 }
Glenn67fd1f32021-02-25 16:04:49 -08001370 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman931e7382021-08-08 21:55:59 +03001371 new Identifier(indexName), indexType, indexedElementList, enforced, gramLength, fullTextConfigName, ifNotExists,
Ali Alsuliman465a7282021-11-24 12:51:32 -08001372 excludeUnknown, castDefaultNull, castConfig);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001373 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001374 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001375}
1376
Glenn67fd1f32021-02-25 16:04:49 -08001377CreateIndexStatement.IndexedElement IndexedElement(Token startElementToken) throws ParseException:
1378{
1379 Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1380 Pair<List<String>, IndexedTypeExpression> elementSimple = null;
1381 int elementSimpleSource = 0;
1382}
1383{
1384 (
1385 element = IndexedElementUnnestSelect()
1386 | (
1387 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1388 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1389 <DOT> elementSimple = IndexedField()
1390 { elementSimpleSource = 1; }
1391 )
1392 | elementSimple = IndexedField()
1393 | <LEFTPAREN> ( element = IndexedElementUnnestSelect() | elementSimple = IndexedField() ) <RIGHTPAREN>
1394 )
1395 {
1396 int source;
1397 List<List<String>> unnestList;
1398 List<Pair<List<String>, IndexedTypeExpression>> projectList;
1399 if (elementSimple != null) {
1400 source = elementSimpleSource;
1401 unnestList = null;
1402 projectList = Collections.singletonList(elementSimple);
1403 } else {
1404 source = element.first;
1405 unnestList = element.second;
1406 projectList = element.third;
1407 }
1408 CreateIndexStatement.IndexedElement ie = new CreateIndexStatement.IndexedElement(source, unnestList, projectList);
1409 ie.setSourceLocation(getSourceLocation(startElementToken));
1410 return ie;
1411 }
1412}
1413
1414Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelect()
1415 throws ParseException:
1416{
1417 int source = 0;
1418 Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1419}
1420{
1421 <UNNEST>
1422 (
1423 (
1424 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1425 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1426 <DOT> element = IndexedElementUnnestSelectBody() { source = 1; }
1427 ) | element = IndexedElementUnnestSelectBody()
1428 )
1429 {
1430 return new Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(
1431 source, element.first, element.second
1432 );
1433 }
1434}
1435
1436Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelectBody()
1437 throws ParseException:
1438{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001439 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001440 IndexedTypeExpression type = null;
1441 List<List<String>> unnestList = new ArrayList();
Glenn67fd1f32021-02-25 16:04:49 -08001442 List<Pair<List<String>, IndexedTypeExpression>> projectList = new ArrayList();
1443}
1444{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001445 path = MultipartIdentifier() { unnestList.add(path.first); }
1446 ( <UNNEST> path = MultipartIdentifier() { unnestList.add(path.first); })*
Glenn67fd1f32021-02-25 16:04:49 -08001447 (
Glenn58329202021-04-09 12:03:46 -07001448 ( <COLON> type = IndexedTypeExpr(false)
Glenn67fd1f32021-02-25 16:04:49 -08001449 {
1450 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, type));
1451 }
1452 ) |
1453 (
Glenn58329202021-04-09 12:03:46 -07001454 <SELECT> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1455 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001456 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001457 }
1458 ( <COMMA> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1459 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001460 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001461 }
1462 )*
Glenn67fd1f32021-02-25 16:04:49 -08001463 )
1464 )?
1465 {
1466 if (projectList.isEmpty()) {
1467 // To support the case (<UNNEST> IDENTIFIER)* IDENTIFIER w/o any type specification.
1468 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, null));
1469 }
1470
1471 return new Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(unnestList, projectList);
1472 }
1473}
1474
1475Pair<List<String>, IndexedTypeExpression> IndexedField() throws ParseException:
1476{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001477 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001478 IndexedTypeExpression type = null;
1479}
1480{
Glenn58329202021-04-09 12:03:46 -07001481 path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(true) )?
Glenn67fd1f32021-02-25 16:04:49 -08001482 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001483 return new Pair<List<String>, IndexedTypeExpression>(path.first, type);
Glenn67fd1f32021-02-25 16:04:49 -08001484 }
1485}
1486
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001487CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
1488{
Glenn67fd1f32021-02-25 16:04:49 -08001489 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001490 String indexName = null;
1491 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001492}
1493{
1494 (indexName = Identifier())? ifNotExists = IfNotExists()
1495 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
1496 {
1497 if (indexName == null) {
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07001498 indexName = MetadataConstants.PRIMARY_INDEX_PREFIX + nameComponents.second;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001499 }
Glenn67fd1f32021-02-25 16:04:49 -08001500 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman465a7282021-11-24 12:51:32 -08001501 new Identifier(indexName), IndexType.BTREE, Collections.emptyList(), false, -1, null, ifNotExists, null, null,
1502 Collections.emptyMap());
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001503 return addSourceLocation(stmt, startStmtToken);
1504 }
1505}
1506
Yingyi Bu391f09e2015-10-29 13:49:39 -07001507String FilterField() throws ParseException :
1508{
1509 String filterField = null;
1510}
1511{
1512 filterField = Identifier()
1513 {
1514 return filterField;
1515 }
1516}
1517
1518IndexParams IndexType() throws ParseException:
1519{
1520 IndexType type = null;
1521 int gramLength = 0;
Rui Guoe6986dd2020-11-25 19:50:06 -08001522 String fullTextConfig = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001523}
1524{
1525 (<BTREE>
1526 {
1527 type = IndexType.BTREE;
1528 }
1529 | <RTREE>
1530 {
1531 type = IndexType.RTREE;
1532 }
1533 | <KEYWORD>
1534 {
1535 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1536 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001537 | <FULLTEXT>
Taewoo Kimc49405a2017-01-04 00:30:43 -08001538 {
1539 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1540 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001541 // For now we don't allow inverted index creation using a full-text config in another data verse.
1542 // We may want to support corss-dataverse full-text config access later
1543 // If so, replace the Identifier() with QualifiedName() to get the dataverse name
1544 ( <USING> Identifier()
1545 {
1546 fullTextConfig = token.image;
1547 }
1548 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001549 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1550 {
1551 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1552 gramLength = Integer.valueOf(token.image);
1553 }
1554 <RIGHTPAREN>)
1555 {
Rui Guoe6986dd2020-11-25 19:50:06 -08001556 return new IndexParams(type, gramLength, fullTextConfig);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001557 }
1558}
1559
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001560CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1561{
1562 CreateDataverseStatement stmt = null;
1563}
1564{
1565 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1566 {
1567 return stmt;
1568 }
1569}
1570
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001571CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001572{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001573 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001574 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001575}
1576{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001577 dvName = DataverseName() ifNotExists = IfNotExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001578 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001579 CreateDataverseStatement stmt = new CreateDataverseStatement(dvName, null, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001580 return addSourceLocation(stmt, startStmtToken);
1581 }
1582}
1583
Ian Maxon38fe9402020-01-29 19:27:40 -08001584CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1585{
1586 CreateAdapterStatement stmt = null;
1587}
1588{
1589 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1590 {
1591 return stmt;
1592 }
1593}
1594
1595CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1596{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001597 Pair<DataverseName,Identifier> adapterName = null;
1598 Pair<DataverseName,Identifier> libraryName = null;
1599 List<String> externalIdentifier = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001600 boolean ifNotExists = false;
1601}
1602{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001603 adapterName = QualifiedName()
1604 ifNotExists = IfNotExists()
1605 <AS> externalIdentifier = FunctionExternalIdentifier()
1606 <AT> libraryName = QualifiedName()
1607 {
1608 CreateAdapterStatement stmt = new CreateAdapterStatement(adapterName.first, adapterName.second.getValue(),
1609 libraryName.first, libraryName.second.getValue(), externalIdentifier, ifNotExists);
1610 return addSourceLocation(stmt, startStmtToken);
1611 }
1612}
Ian Maxon38fe9402020-01-29 19:27:40 -08001613
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001614CreateViewStatement CreateViewStatement(Token startStmtToken, boolean orReplace) throws ParseException:
1615{
1616 CreateViewStatement stmt = null;
1617}
1618{
1619 <VIEW> stmt = ViewSpecification(startStmtToken, orReplace)
1620 {
1621 return stmt;
1622 }
1623}
1624
1625CreateViewStatement ViewSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
1626{
1627 Pair<DataverseName, Identifier> nameComponents = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001628 TypeExpression typeExpr = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001629 boolean ifNotExists = false;
1630 Token beginPos = null, endPos = null;
1631 Expression viewBodyExpr = null;
Ali Alsuliman465a7282021-11-24 12:51:32 -08001632 Pair<Map<String, String>, Boolean> viewConfigDefaultNull;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001633 Boolean defaultNull = null;
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001634 Map<String, String> viewConfig = null;
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001635 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001636 Pair<List<Integer>, List<List<String>>> foreignKeyFields = null;
1637 Pair<DataverseName, Identifier> refNameComponents = null;
1638 List<CreateViewStatement.ForeignKeyDecl> foreignKeyDecls = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001639 DataverseName currentDataverse = defaultDataverse;
1640}
1641{
1642 nameComponents = QualifiedName()
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001643 (
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001644 (
1645 typeExpr = DatasetTypeSpecification()
1646 ifNotExists = IfNotExists()
Ali Alsuliman465a7282021-11-24 12:51:32 -08001647 viewConfigDefaultNull = CastDefaultNull()
1648 {
1649 viewConfig = viewConfigDefaultNull.first;
1650 defaultNull = viewConfigDefaultNull.second;
1651 }
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001652 (
1653 <PRIMARY> <KEY> <LEFTPAREN> primaryKeyFields = PrimaryKeyFields() <RIGHTPAREN>
1654 <NOT> <ENFORCED>
1655 )?
1656 (
1657 <IDENTIFIER> { expectToken(FOREIGN); } <KEY> <LEFTPAREN> foreignKeyFields = PrimaryKeyFields() <RIGHTPAREN>
1658 <IDENTIFIER> { expectToken(REFERENCES); } refNameComponents = QualifiedName()
1659 <NOT> <ENFORCED>
1660 {
1661 if (foreignKeyDecls == null) {
1662 foreignKeyDecls = new ArrayList<CreateViewStatement.ForeignKeyDecl>();
1663 }
1664 foreignKeyDecls.add(new CreateViewStatement.ForeignKeyDecl(foreignKeyFields.second,
1665 foreignKeyFields.first, refNameComponents.first, refNameComponents.second));
1666 }
1667 )*
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001668 )
1669 |
1670 ( ifNotExists = IfNotExists() )
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001671 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001672 {
1673 if (orReplace && ifNotExists) {
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001674 throw new SqlppParseException(getSourceLocation(startStmtToken), "Unexpected IF NOT EXISTS");
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001675 }
1676 }
1677 <AS>
1678 {
1679 beginPos = token;
1680 createNewScope();
1681 if (nameComponents.first != null) {
1682 defaultDataverse = nameComponents.first;
1683 }
1684 }
1685 viewBodyExpr = ViewBody()
1686 {
1687 endPos = token;
1688 String viewBody = extractFragment(beginPos.beginLine, beginPos.beginColumn + 1, endPos.endLine,
1689 endPos.endColumn + 1);
1690 removeCurrentScope();
1691 defaultDataverse = currentDataverse;
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -08001692 if (typeExpr != null && primaryKeyFields != null) {
1693 DatasetDeclParametersUtil.adjustInlineTypeDecl(typeExpr, primaryKeyFields.second, primaryKeyFields.first, false);
1694 }
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001695 CreateViewStatement.KeyDecl primaryKeyDecl = primaryKeyFields != null ?
1696 new CreateViewStatement.KeyDecl(primaryKeyFields.second, primaryKeyFields.first) : null;
1697 CreateViewStatement stmt = new CreateViewStatement(nameComponents.first, nameComponents.second.getValue(),
1698 typeExpr, viewBody, viewBodyExpr, defaultNull, viewConfig, primaryKeyDecl, foreignKeyDecls, orReplace,
1699 ifNotExists);
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001700 return addSourceLocation(stmt, startStmtToken);
1701 }
1702}
1703
1704Expression ViewBody() throws ParseException:
1705{
1706 Expression viewBodyExpr = null;
1707}
1708{
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001709 (
1710 ( viewBodyExpr = VariableRef() ( viewBodyExpr = FieldAccessor(viewBodyExpr) )* )
1711 | viewBodyExpr = SelectExpression(true)
1712 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001713 {
1714 return viewBodyExpr;
1715 }
1716}
1717
Ali Alsuliman465a7282021-11-24 12:51:32 -08001718Pair<Map<String, String>, Boolean> CastDefaultNull() throws ParseException:
1719{
1720 Map<String, String> castConfig = null;
1721 Boolean defaultNull = null;
1722 String propertyName = null, propertyValue = null;
1723}
1724{
1725 <IDENTIFIER> { expectToken(DEFAULT); } <NULL> { defaultNull = true; }
1726 (
1727 LOOKAHEAD(2) <IDENTIFIER> { propertyName = token.image.toLowerCase(); } propertyValue = StringLiteral()
1728 {
1729 if (castConfig == null) {
1730 castConfig = new HashMap<String, String>();
1731 }
1732 castConfig.put(propertyName, propertyValue);
1733 }
1734 )*
1735 {
1736 return new Pair<Map<String, String>, Boolean>(castConfig, defaultNull);
1737 }
1738}
1739
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001740CreateFunctionStatement CreateFunctionStatement(Token startStmtToken, boolean orReplace) throws ParseException:
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001741{
1742 CreateFunctionStatement stmt = null;
1743}
1744{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001745 <FUNCTION> stmt = FunctionSpecification(startStmtToken, orReplace)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001746 {
1747 return stmt;
1748 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001749}
1750
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001751CreateFunctionStatement FunctionSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001752{
Ian Maxon38fe9402020-01-29 19:27:40 -08001753 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001754 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001755 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001756 List<Pair<VarIdentifier,TypeExpression>> params = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001757 int arity = 0;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001758 TypeExpression returnType = null;
1759 Token beginPos = null, endPos = null;
1760 Expression functionBodyExpr = null;
1761 Pair<DataverseName,Identifier> libraryName = null;
1762 List<String> externalIdentifier = null;
1763 RecordConstructor withOptions = null;
1764 boolean ifNotExists = false;
1765 CreateFunctionStatement stmt = null;
1766 DataverseName currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001767}
1768{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001769 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001770 {
1771 defaultDataverse = fctName.dataverse;
1772 }
Dmitry Lychagin541652082020-11-09 14:04:53 -08001773 paramsWithArity = FunctionParameters()
1774 {
1775 arity = paramsWithArity.first;
1776 params = paramsWithArity.second;
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001777 signature = new FunctionSignature(fctName.dataverse, fctName.function, arity);
Dmitry Lychagin541652082020-11-09 14:04:53 -08001778 }
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001779 ifNotExists = IfNotExists()
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001780 {
1781 if (orReplace && ifNotExists) {
1782 throw new SqlppParseException(getSourceLocation(token), "Unexpected IF NOT EXISTS");
1783 }
1784 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001785 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001786 (
1787 (
1788 <LEFTBRACE>
1789 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001790 beginPos = token;
1791 createNewScope();
Ian Maxon38fe9402020-01-29 19:27:40 -08001792 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001793 functionBodyExpr = FunctionBody()
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001794 <RIGHTBRACE>
1795 {
1796 endPos = token;
1797 String functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine,
1798 endPos.beginColumn);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001799 getCurrentScope().addFunctionDescriptor(signature, false);
1800 removeCurrentScope();
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001801 ensureNoTypeDeclsInFunction(fctName.function, params, returnType, startStmtToken);
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001802 stmt = new CreateFunctionStatement(signature, params, functionBody, functionBodyExpr, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001803 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001804 )
1805 |
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001806 (
1807 <AS> externalIdentifier = FunctionExternalIdentifier()
1808 <AT> libraryName = QualifiedName()
1809 (<WITH> withOptions = RecordConstructor())?
1810 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001811 try {
1812 stmt = new CreateFunctionStatement(signature, params, returnType, libraryName.first,
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001813 libraryName.second.getValue(), externalIdentifier, withOptions, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001814 } catch (AlgebricksException e) {
1815 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1816 }
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001817 }
1818 )
Ian Maxon38fe9402020-01-29 19:27:40 -08001819 )
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001820 {
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001821 defaultDataverse = currentDataverse;
1822 return addSourceLocation(stmt, startStmtToken);
1823 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001824}
1825
Dmitry Lychagin541652082020-11-09 14:04:53 -08001826Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameters() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001827{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001828 Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> params = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001829}
1830{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001831 <LEFTPAREN> (params = FunctionParameterList())? <RIGHTPAREN>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001832 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001833 if (params == null) {
1834 params = new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(
1835 0, Collections.<Pair<VarIdentifier, TypeExpression>>emptyList()
1836 );
1837 }
1838 return params;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001839 }
1840}
1841
Dmitry Lychagin541652082020-11-09 14:04:53 -08001842Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameterList() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001843{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001844 List<Pair<VarIdentifier, TypeExpression>> paramList = null;
1845 Pair<VarIdentifier, TypeExpression> param = null;
1846 int arity = 0;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001847}
1848{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001849 (
1850 (
1851 <DOT> <DOT> <DOT>
1852 {
1853 param = new Pair<VarIdentifier, TypeExpression>(
1854 SqlppVariableUtil.toInternalVariableIdentifier(UDF_VARARGS_PARAM_NAME), null
1855 );
1856 paramList = Collections.<Pair<VarIdentifier, TypeExpression>>singletonList(param);
1857 arity = FunctionIdentifier.VARARGS;
1858 }
1859 )
1860 |
1861 (
1862 param = FunctionParameter()
1863 {
1864 paramList = new ArrayList<Pair<VarIdentifier, TypeExpression>>();
1865 paramList.add(param);
1866 }
1867 ( <COMMA> param = FunctionParameter() { paramList.add(param); } )*
1868 {
1869 arity = paramList.size();
1870 }
1871 )
1872 )
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001873 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001874 return new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(arity, paramList);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001875 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001876}
1877
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001878Pair<VarIdentifier,TypeExpression> FunctionParameter() throws ParseException:
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001879{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001880 String name = null;
1881 TypeExpression type = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001882}
1883{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001884 name = VariableIdentifier()
1885 ( LOOKAHEAD(3) (<COLON>)? type = TypeExpr(false) )?
1886 {
1887 return new Pair<VarIdentifier,TypeExpression>(new VarIdentifier(name), type);
1888 }
1889}
1890
1891TypeExpression FunctionReturnType() throws ParseException:
1892{
1893 TypeExpression returnType = null;
1894}
1895{
1896 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = TypeExpr(false) )?
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001897 {
1898 return returnType;
1899 }
1900}
1901
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001902Expression FunctionBody() throws ParseException:
1903{
1904 Expression functionBodyExpr = null;
1905}
1906{
1907 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1908 {
1909 return functionBodyExpr;
1910 }
1911}
1912
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001913List<String> FunctionExternalIdentifier() throws ParseException:
1914{
1915 String ident = null;
1916 List<String> identList = new ArrayList(2);
1917}
1918{
1919 ident = StringLiteral() { identList.add(ident.trim()); }
1920 ( <COMMA> ident = StringLiteral() { identList.add(ident.trim()); } )*
1921 {
1922 return identList;
1923 }
1924}
1925
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001926CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1927{
1928 CreateFeedStatement stmt = null;
1929}
1930{
1931 <FEED> stmt = FeedSpecification(startStmtToken)
1932 {
1933 return stmt;
1934 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001935}
1936
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001937CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001938{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001939 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001940 boolean ifNotExists = false;
1941 String adapterName = null;
1942 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001943 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001944 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001945 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001946}
1947{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001948 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001949 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001950 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001951 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001952 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001953 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001954 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001955 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001956 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001957 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001958}
1959
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001960CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1961{
1962 CreateFeedPolicyStatement stmt = null;
1963}
1964{
1965 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1966 {
1967 return stmt;
1968 }
1969}
1970
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001971CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001972{
Michael Blowd6cf6412016-06-30 02:44:35 -04001973 String policyName = null;
1974 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001975 String sourcePolicyFile = null;
1976 String definition = null;
1977 boolean ifNotExists = false;
1978 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001979 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001980}
1981{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001982 policyName = Identifier() ifNotExists = IfNotExists()
1983 <FROM>
1984 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001985 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001986 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001987 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001988 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1989 {
1990 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1991 }
1992 )
1993 {
1994 return addSourceLocation(stmt, startStmtToken);
1995 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001996}
1997
Rui Guoe6986dd2020-11-25 19:50:06 -08001998Statement CreateFullTextStatement(Token startStmtToken) throws ParseException:
1999{
2000 Statement stmt = null;
2001}
2002{
2003 (
2004 <FULLTEXT>
2005 (
2006 <FILTER> stmt = CreateFullTextFilterSpec(startStmtToken)
2007 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = CreateFullTextConfigSpec(startStmtToken))
2008 )
2009 )
2010 {
2011 return stmt;
2012 }
2013}
2014
2015CreateFullTextFilterStatement CreateFullTextFilterSpec(Token startStmtToken) throws ParseException:
2016{
2017 CreateFullTextFilterStatement stmt = null;
2018 Pair<DataverseName,Identifier> nameComponents = null;
2019 boolean ifNotExists = false;
2020 RecordConstructor expr = null;
2021}
2022{
2023 (
2024 nameComponents = QualifiedName() ifNotExists = IfNotExists()
2025 <AS>
2026 expr = RecordConstructor()
2027 )
2028 {
2029 try {
2030 stmt = new CreateFullTextFilterStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
2031 return addSourceLocation(stmt, startStmtToken);
2032 } catch (CompilationException e) {
2033 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
2034 }
2035 }
2036}
2037
2038CreateFullTextConfigStatement CreateFullTextConfigSpec(Token startStmtToken) throws ParseException:
2039{
2040 CreateFullTextConfigStatement stmt = null;
2041 Pair<DataverseName,Identifier> nameComponents = null;
2042 boolean ifNotExists = false;
2043 RecordConstructor expr = null;
2044}
2045{
2046 (
2047 nameComponents = QualifiedName() ifNotExists = IfNotExists()
2048 <AS>
2049 expr = RecordConstructor()
2050 )
2051 {
2052 try {
2053 stmt = new CreateFullTextConfigStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
2054 return addSourceLocation(stmt, startStmtToken);
2055 } catch (CompilationException e) {
2056 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
2057 }
2058 }
2059}
2060
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002061CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
2062{
2063 CreateSynonymStatement stmt = null;
2064}
2065{
2066 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
2067 {
2068 return stmt;
2069 }
2070}
2071
2072CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
2073{
2074 Pair<DataverseName,Identifier> nameComponents = null;
2075 Pair<DataverseName,Identifier> objectNameComponents = null;
2076 boolean ifNotExists = false;
2077}
2078{
2079 nameComponents = QualifiedName() ifNotExists = IfNotExists()
2080 <FOR> objectNameComponents = QualifiedName()
2081 {
2082 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
2083 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
2084 return addSourceLocation(stmt, startStmtToken);
2085 }
2086}
2087
Yingyi Bu391f09e2015-10-29 13:49:39 -07002088boolean IfNotExists() throws ParseException:
2089{
2090}
2091{
Yingyi Budaa549c2016-06-28 22:30:52 -07002092 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002093 {
2094 return true;
2095 }
2096 )?
2097 {
2098 return false;
2099 }
2100}
2101
Xikui Wang9d63f622017-05-18 17:50:44 -07002102void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002103{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002104 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07002105 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002106}
2107{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002108 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002109 {
2110 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
2111 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
2112 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07002113 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002114 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07002115 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002116 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
2117 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07002118 }
2119 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07002120}
2121
2122String GetPolicy() throws ParseException:
2123{
2124 String policy = null;
2125}
2126{
2127 <USING> <POLICY> policy = Identifier()
2128 {
2129 return policy;
2130 }
2131
2132}
2133
2134FunctionSignature FunctionSignature() throws ParseException:
2135{
2136 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08002137 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> params = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002138 int arity = 0;
2139}
2140{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002141 fctName = FunctionName()
2142 (
Dmitry Lychagin541652082020-11-09 14:04:53 -08002143 LOOKAHEAD(2) params = FunctionParameters() { arity = params.first; }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002144 | ( <LEFTPAREN> arity = FunctionArity() <RIGHTPAREN> )
2145 | ( <ATT> arity = FunctionArity() ) // back-compat
2146 )
2147 {
2148 return new FunctionSignature(fctName.dataverse, fctName.function, arity);
2149 }
2150}
Yingyi Bu391f09e2015-10-29 13:49:39 -07002151
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002152int FunctionArity() throws ParseException:
2153{
2154 int arity;
2155}
2156{
2157 <INTEGER_LITERAL>
2158 {
2159 try {
2160 arity = Integer.parseInt(token.image);
2161 } catch (NumberFormatException e) {
2162 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002163 }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002164 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
2165 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + arity);
2166 }
2167 return arity;
2168 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002169}
2170
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07002171Triple<List<Integer>, List<List<String>>, List<TypeExpression>> PrimaryKeyWithType() throws ParseException:
2172{
2173 Triple<List<Integer>, List<List<String>>, List<TypeExpression>> primaryKeyFieldsWithTypes = null;
2174}
2175{
2176 <PRIMARY> <KEY> <LEFTPAREN> primaryKeyFieldsWithTypes = PrimaryKeyFieldsWithType() <RIGHTPAREN>
2177 {
2178 return primaryKeyFieldsWithTypes;
2179 }
2180}
2181
2182Triple<List<Integer>, List<List<String>>, List<TypeExpression>> PrimaryKeyFieldsWithType() throws ParseException:
2183{
2184 Pair<Integer, List<String>> tmp = null;
2185 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
2186 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
2187 List<TypeExpression> primaryKeyFieldTypes = new ArrayList<TypeExpression>();
2188 TypeExpression type = null;
2189}
2190{
2191 tmp = NestedField() <COLON> type = TypeReference()
2192 {
2193 keyFieldSourceIndicators.add(tmp.first);
2194 primaryKeyFields.add(tmp.second);
2195 primaryKeyFieldTypes.add(type);
2196 }
2197 ( <COMMA> tmp = NestedField() <COLON> type = TypeReference()
2198 {
2199 keyFieldSourceIndicators.add(tmp.first);
2200 primaryKeyFields.add(tmp.second);
2201 primaryKeyFieldTypes.add(type);
2202 }
2203 )*
2204 {
2205 return new Triple<List<Integer>, List<List<String>>, List<TypeExpression>> (keyFieldSourceIndicators,
2206 primaryKeyFields, primaryKeyFieldTypes);
2207 }
2208}
2209
Yingyi Buc9bfe252016-03-01 00:02:40 -08002210Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002211{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07002212 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
2213}
2214{
2215 <PRIMARY> <KEY> primaryKeyFields = PrimaryKeyFields()
2216 {
2217 return primaryKeyFields;
2218 }
2219}
2220
2221Pair<List<Integer>, List<List<String>>> PrimaryKeyFields() throws ParseException:
2222{
Yingyi Buc9bfe252016-03-01 00:02:40 -08002223 Pair<Integer, List<String>> tmp = null;
2224 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002225 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
2226}
2227{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07002228 tmp = NestedField()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002229 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002230 keyFieldSourceIndicators.add(tmp.first);
2231 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002232 }
2233 ( <COMMA> tmp = NestedField()
2234 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002235 keyFieldSourceIndicators.add(tmp.first);
2236 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002237 }
2238 )*
2239 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002240 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002241 }
2242}
2243
2244Statement DropStatement() throws ParseException:
2245{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002246 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002247 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002248}
2249{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002250 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002251 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002252 stmt = DropDatasetStatement(startToken)
2253 | stmt = DropIndexStatement(startToken)
2254 | stmt = DropNodeGroupStatement(startToken)
2255 | stmt = DropTypeStatement(startToken)
2256 | stmt = DropDataverseStatement(startToken)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002257 | stmt = DropAdapterStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002258 | stmt = DropFunctionStatement(startToken)
2259 | stmt = DropFeedStatement(startToken)
2260 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002261 | stmt = DropSynonymStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -08002262 | stmt = DropFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002263 | stmt = DropViewStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002264 )
2265 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002266 return stmt;
2267 }
2268}
2269
2270DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
2271{
2272 DropDatasetStatement stmt = null;
2273}
2274{
2275 Dataset() stmt = DropDatasetSpecification(startStmtToken)
2276 {
2277 return stmt;
2278 }
2279}
2280
2281DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
2282{
2283 Pair<DataverseName,Identifier> pairId = null;
2284 boolean ifExists = false;
2285}
2286{
2287 pairId = QualifiedName() ifExists = IfExists()
2288 {
2289 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
2290 return addSourceLocation(stmt, startStmtToken);
2291 }
2292}
2293
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002294ViewDropStatement DropViewStatement(Token startStmtToken) throws ParseException:
2295{
2296 ViewDropStatement stmt = null;
2297}
2298{
2299 <VIEW> stmt = DropViewSpecification(startStmtToken)
2300 {
2301 return stmt;
2302 }
2303}
2304
2305ViewDropStatement DropViewSpecification(Token startStmtToken) throws ParseException:
2306{
2307 Pair<DataverseName,Identifier> pairId = null;
2308 boolean ifExists = false;
2309}
2310{
2311 pairId = QualifiedName() ifExists = IfExists()
2312 {
2313 ViewDropStatement stmt = new ViewDropStatement(pairId.first, pairId.second, ifExists);
2314 return addSourceLocation(stmt, startStmtToken);
2315 }
2316}
2317
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002318IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
2319{
2320 IndexDropStatement stmt = null;
2321}
2322{
2323 <INDEX> stmt = DropIndexSpecification(startStmtToken)
2324 {
2325 return stmt;
2326 }
2327}
2328
2329IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
2330{
2331 Triple<DataverseName,Identifier,Identifier> tripleId = null;
2332 boolean ifExists = false;
2333}
2334{
2335 tripleId = DoubleQualifiedName() ifExists = IfExists()
2336 {
2337 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
2338 return addSourceLocation(stmt, startStmtToken);
2339 }
2340}
2341
Rui Guoe6986dd2020-11-25 19:50:06 -08002342Statement DropFullTextStatement(Token startStmtToken) throws ParseException:
2343{
2344 Statement stmt = null;
2345}
2346{
2347 <FULLTEXT>
2348 (
2349 <FILTER> stmt = DropFullTextFilterSpec(startStmtToken)
2350 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = DropFullTextConfigSpec(startStmtToken))
2351 )
2352 {
2353 return stmt;
2354 }
2355}
2356
2357FullTextFilterDropStatement DropFullTextFilterSpec(Token startStmtToken) throws ParseException:
2358{
2359 FullTextFilterDropStatement stmt = null;
2360 Pair<DataverseName,Identifier> nameComponents = null;
2361 boolean ifExists = false;
2362}
2363{
2364 nameComponents = QualifiedName() ifExists = IfExists()
2365 {
2366 stmt = new FullTextFilterDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2367 return addSourceLocation(stmt, startStmtToken);
2368 }
2369}
2370
2371FullTextConfigDropStatement DropFullTextConfigSpec(Token startStmtToken) throws ParseException:
2372{
2373 FullTextConfigDropStatement stmt = null;
2374 Pair<DataverseName,Identifier> nameComponents = null;
2375 boolean ifExists = false;
2376}
2377{
2378 nameComponents = QualifiedName() ifExists = IfExists()
2379 {
2380 stmt = new FullTextConfigDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2381 return addSourceLocation(stmt, startStmtToken);
2382 }
2383}
2384
2385
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002386NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
2387{
2388 NodeGroupDropStatement stmt = null;
2389}
2390{
2391 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
2392 {
2393 return stmt;
2394 }
2395}
2396
2397NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
2398{
2399 String id = null;
2400 boolean ifExists = false;
2401}
2402{
2403 id = Identifier() ifExists = IfExists()
2404 {
2405 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
2406 return addSourceLocation(stmt, startStmtToken);
2407 }
2408}
2409
2410TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
2411{
2412 TypeDropStatement stmt = null;
2413}
2414{
2415 <TYPE> stmt = DropTypeSpecification(startStmtToken)
2416 {
2417 return stmt;
2418 }
2419}
2420
2421TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
2422{
2423 Pair<DataverseName,Identifier> pairId = null;
2424 boolean ifExists = false;
2425}
2426{
2427 pairId = TypeName() ifExists = IfExists()
2428 {
2429 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
2430 return addSourceLocation(stmt, startStmtToken);
2431 }
2432}
2433
2434DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
2435{
2436 DataverseDropStatement stmt = null;
2437}
2438{
2439 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
2440 {
2441 return stmt;
2442 }
2443}
2444
2445DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
2446{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002447 DataverseName dvName = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002448 boolean ifExists = false;
2449}
2450{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002451 dvName = DataverseName() ifExists = IfExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002452 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002453 DataverseDropStatement stmt = new DataverseDropStatement(dvName, ifExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002454 return addSourceLocation(stmt, startStmtToken);
2455 }
2456}
2457
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002458AdapterDropStatement DropAdapterStatement(Token startStmtToken) throws ParseException:
2459{
2460 AdapterDropStatement stmt = null;
2461}
2462{
2463 <ADAPTER> stmt = DropAdapterSpecification(startStmtToken)
2464 {
2465 return stmt;
2466 }
2467}
2468
2469AdapterDropStatement DropAdapterSpecification(Token startStmtToken) throws ParseException:
2470{
2471 Pair<DataverseName,Identifier> adapterName = null;
2472 boolean ifExists = false;
2473}
2474{
2475 adapterName = QualifiedName() ifExists = IfExists()
2476 {
2477 AdapterDropStatement stmt = new AdapterDropStatement(adapterName.first, adapterName.second.getValue(), ifExists);
2478 return addSourceLocation(stmt, startStmtToken);
2479 }
2480}
2481
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002482FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
2483{
2484 FunctionDropStatement stmt = null;
2485}
2486{
2487 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
2488 {
2489 return stmt;
2490 }
2491}
2492
2493FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
2494{
2495 FunctionSignature funcSig = null;
2496 boolean ifExists = false;
2497}
2498{
2499 funcSig = FunctionSignature() ifExists = IfExists()
2500 {
2501 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
2502 return addSourceLocation(stmt, startStmtToken);
2503 }
2504}
2505
2506FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
2507{
2508 FeedDropStatement stmt = null;
2509}
2510{
2511 <FEED> stmt = DropFeedSpecification(startStmtToken)
2512 {
2513 return stmt;
2514 }
2515}
2516
2517FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
2518{
2519 Pair<DataverseName,Identifier> pairId = null;
2520 boolean ifExists = false;
2521}
2522{
2523 pairId = QualifiedName() ifExists = IfExists()
2524 {
2525 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
2526 return addSourceLocation(stmt, startStmtToken);
2527 }
2528}
2529
2530FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
2531{
2532 FeedPolicyDropStatement stmt = null;
2533}
2534{
2535 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
2536 {
2537 return stmt;
2538 }
2539}
2540
2541FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
2542{
2543 Pair<DataverseName,Identifier> pairId = null;
2544 boolean ifExists = false;
2545}
2546{
2547 pairId = QualifiedName() ifExists = IfExists()
2548 {
2549 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
2550 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002551 }
2552}
2553
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002554SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
2555{
2556 SynonymDropStatement stmt = null;
2557}
2558{
2559 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
2560 {
2561 return stmt;
2562 }
2563}
2564
2565SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
2566{
2567 Pair<DataverseName,Identifier> pairId = null;
2568 boolean ifExists = false;
2569}
2570{
2571 pairId = QualifiedName() ifExists = IfExists()
2572 {
2573 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
2574 return addSourceLocation(stmt, startStmtToken);
2575 }
2576}
2577
Yingyi Bu391f09e2015-10-29 13:49:39 -07002578boolean IfExists() throws ParseException :
2579{
2580}
2581{
2582 ( LOOKAHEAD(1) <IF> <EXISTS>
2583 {
2584 return true;
2585 }
2586 )?
2587 {
2588 return false;
2589 }
2590}
2591
2592InsertStatement InsertStatement() throws ParseException:
2593{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002594 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002595 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002596 VariableExpr var = null;
2597 Query query = null;
2598 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002599}
2600{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002601 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002602 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002603 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002604 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002605 if (var == null && returnExpression != null) {
2606 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2607 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002608 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08002609 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002610 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2611 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002612 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002613 }
2614}
2615
2616UpsertStatement UpsertStatement() throws ParseException:
2617{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002618 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002619 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002620 VariableExpr var = null;
2621 Query query = null;
2622 Expression returnExpression = null;
2623}
2624{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002625 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002626 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002627 ( <RETURNING> returnExpression = Expression())?
2628 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002629 if (var == null && returnExpression != null) {
2630 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2631 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002632 }
2633 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002634 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2635 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002636 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002637 }
2638}
2639
2640DeleteStatement DeleteStatement() throws ParseException:
2641{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002642 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002643 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002644 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002645 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002646}
2647{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002648 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002649 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002650 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07002651 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002652 if (var == null) {
2653 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2654 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002655 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002656 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07002657 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002658 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002659 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002660}
2661
2662UpdateStatement UpdateStatement() throws ParseException:
2663{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002664 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002665 VariableExpr vars;
2666 Expression target;
2667 Expression condition;
2668 UpdateClause uc;
2669 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
2670}
2671{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002672 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002673 <WHERE> condition = Expression()
2674 <LEFTPAREN> (uc = UpdateClause()
2675 {
2676 ucs.add(uc);
2677 }
2678 (<COMMA> uc = UpdateClause()
2679 {
2680 ucs.add(uc);
2681 }
2682 )*) <RIGHTPAREN>
2683 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002684 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002685 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002686 }
2687}
2688
2689UpdateClause UpdateClause() throws ParseException:
2690{
2691 Expression target = null;
2692 Expression value = null ;
2693 InsertStatement is = null;
2694 DeleteStatement ds = null;
2695 UpdateStatement us = null;
2696 Expression condition = null;
2697 UpdateClause ifbranch = null;
2698 UpdateClause elsebranch = null;
2699}
2700{
2701 (<SET> target = Expression() <EQ> value = Expression()
2702 | is = InsertStatement()
2703 | ds = DeleteStatement()
2704 | us = UpdateStatement()
2705 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
2706 <THEN> ifbranch = UpdateClause()
2707 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
2708 {
2709 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
2710 }
2711 )
2712}
2713
2714Statement SetStatement() throws ParseException:
2715{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002716 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002717 String pn = null;
2718 String pv = null;
2719}
2720{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002721 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002722 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002723 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002724 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002725 }
2726}
2727
Peeyush Gupta066fd562023-04-30 13:03:00 -07002728CopyStatement CopyStatement() throws ParseException:
2729{
2730 Token startToken = null;
2731 DataverseName dataverseName = null;
2732 Identifier datasetName = null;
2733 boolean alreadySorted = false;
2734 String adapterName;
2735 Map<String,String> properties;
2736 Pair<DataverseName,Identifier> nameComponents = null;
2737}
2738{
2739 <COPY> (<INTO>)? { startToken = token; } nameComponents = QualifiedName()
2740 {
2741 dataverseName = nameComponents.first;
2742 datasetName = nameComponents.second;
2743 }
2744 <USING> adapterName = AdapterName() properties = Configuration()
2745 {
2746 ExternalDetailsDecl edd = new ExternalDetailsDecl();
2747 edd.setAdapter(adapterName);
2748 edd.setProperties(properties);
2749 try {
2750 CopyStatement stmt = new CopyStatement(dataverseName, datasetName.getValue(), edd, null);
2751 return addSourceLocation(stmt, startToken);
2752 } catch (CompilationException e){
2753 throw new SqlppParseException(getSourceLocation(startToken), e.getMessage());
2754 }
2755 }
2756}
2757
Yingyi Bu391f09e2015-10-29 13:49:39 -07002758LoadStatement LoadStatement() throws ParseException:
2759{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002760 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002761 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002762 Identifier datasetName = null;
2763 boolean alreadySorted = false;
2764 String adapterName;
2765 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002766 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002767}
2768{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002769 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002770 {
2771 dataverseName = nameComponents.first;
2772 datasetName = nameComponents.second;
2773 }
2774 <USING> adapterName = AdapterName() properties = Configuration()
2775 (<PRESORTED>
2776 {
2777 alreadySorted = true;
2778 }
2779 )?
2780 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002781 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
2782 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002783 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002784 }
2785}
2786
Yingyi Bu391f09e2015-10-29 13:49:39 -07002787String AdapterName() throws ParseException :
2788{
2789 String adapterName = null;
2790}
2791{
2792 adapterName = Identifier()
2793 {
2794 return adapterName;
2795 }
2796}
2797
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002798Statement AnalyzeStatement() throws ParseException:
2799{
2800 Token startToken = null;
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002801 Statement stmt = null;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002802 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002803}
2804{
Ali Alsuliman7e5c5742022-11-18 12:21:13 -08002805 <ANALYZE> { startToken = token; } DatasetToken() nameComponents = QualifiedName()
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002806 (
2807 stmt = AnalyzeDatasetDropStatement(startToken, nameComponents.first, nameComponents.second)
2808 | stmt = AnalyzeDatasetStatement(startToken, nameComponents.first, nameComponents.second)
2809 )
2810 {
2811 return stmt;
2812 }
2813}
2814
2815Statement AnalyzeDatasetStatement(Token startToken, DataverseName dvName, Identifier identifier) throws ParseException:
2816{
2817 RecordConstructor withRecord = null;
2818}
2819{
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002820 ( <WITH> withRecord = RecordConstructor() )?
2821 {
2822 try {
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002823 AnalyzeStatement stmt = new AnalyzeStatement(dvName, identifier.getValue(), withRecord);
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002824 return addSourceLocation(stmt, startToken);
2825 } catch (CompilationException e) {
2826 throw new SqlppParseException(getSourceLocation(startToken), e.getMessage());
2827 }
2828 }
2829}
2830
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002831Statement AnalyzeDatasetDropStatement(Token startToken, DataverseName dvName, Identifier identifier) throws ParseException:
2832{
2833}
2834{
2835 <DROP> <IDENTIFIER> { expectToken(STATISTICS); }
2836 {
2837 AnalyzeDropStatement stmt = new AnalyzeDropStatement(dvName, identifier.getValue());
2838 return addSourceLocation(stmt, startToken);
2839 }
2840}
2841
Yingyi Bu391f09e2015-10-29 13:49:39 -07002842Statement CompactStatement() throws ParseException:
2843{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002844 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002845 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002846}
2847{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002848 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002849 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002850 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002851 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002852 }
2853}
2854
Yingyi Buab817482016-08-19 21:29:31 -07002855Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002856{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002857 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002858 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002859}
2860{
2861 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002862 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
2863 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
2864 | <START> { startToken = token; } stmt = StartStatement(startToken)
2865 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07002866 )
2867 {
2868 return stmt;
2869 }
2870}
2871
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002872Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002873{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002874 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002875 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002876}
2877{
2878 <FEED> feedNameComponents = QualifiedName()
2879 {
2880 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002881 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002882 }
2883}
2884
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002885AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002886{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002887 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002888 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002889}
2890{
2891 <FEED> feedNameComponents = QualifiedName()
2892 {
2893 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002894 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002895 }
2896}
2897
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002898AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002899{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002900 Pair<DataverseName,Identifier> feedNameComponents = null;
2901 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002902
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002903 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002904}
2905{
2906 (
2907 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002908 {
2909 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
2910 }
2911 )
Yingyi Buab817482016-08-19 21:29:31 -07002912 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002913 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002914 }
2915}
2916
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002917AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002918{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002919 Pair<DataverseName,Identifier> feedNameComponents = null;
2920 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002921
2922 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002923 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002924 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002925 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002926 String whereClauseBody = null;
2927 WhereClause whereClause = null;
2928 Token beginPos = null;
2929 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002930}
2931{
2932 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002933 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002934 (ApplyFunction(appliedFunctions))?
2935 (policy = GetPolicy())?
2936 (
2937 <WHERE>
2938 {
2939 beginPos = token;
2940 whereClause = new WhereClause();
2941 Expression whereExpr;
2942 }
2943 whereExpr = Expression()
2944 {
2945 whereClause.setWhereExpr(whereExpr);
2946 }
2947 )?
2948 {
2949 if (whereClause != null) {
2950 endPos = token;
2951 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2952 }
2953 }
Yingyi Buab817482016-08-19 21:29:31 -07002954 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002955 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002956 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002957 }
2958 )
2959 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002960 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002961 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002962}
2963
2964Map<String,String> Configuration() throws ParseException :
2965{
2966 Map<String,String> configuration = new LinkedHashMap<String,String>();
2967 Pair<String, String> keyValuePair = null;
2968}
2969{
2970 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2971 {
2972 configuration.put(keyValuePair.first, keyValuePair.second);
2973 }
2974 ( <COMMA> keyValuePair = KeyValuePair()
2975 {
2976 configuration.put(keyValuePair.first, keyValuePair.second);
2977 }
2978 )* )? <RIGHTPAREN>
2979 {
2980 return configuration;
2981 }
2982}
2983
2984Pair<String, String> KeyValuePair() throws ParseException:
2985{
2986 String key;
2987 String value;
2988}
2989{
Ali Alsulimane2986012020-05-14 18:14:22 -07002990 <LEFTPAREN> key = ConstantString()
2991 <EQ> ( value = ConstantString() | (<TRUE> | <FALSE>) {value = token.image.toLowerCase();} )
2992 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002993 {
2994 return new Pair<String, String>(key, value);
2995 }
2996}
2997
2998Map<String,String> Properties() throws ParseException:
2999{
3000 Map<String,String> properties = new HashMap<String,String>();
3001 Pair<String, String> property;
3002}
3003{
3004 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
3005 {
3006 properties.put(property.first, property.second);
3007 }
3008 ( <COMMA> property = Property()
3009 {
3010 properties.put(property.first, property.second);
3011 }
3012 )* <RIGHTPAREN> )?
3013 {
3014 return properties;
3015 }
3016}
3017
3018Pair<String, String> Property() throws ParseException:
3019{
Yingyi Bu6d57e492016-06-06 21:24:42 -07003020 String key = null;
3021 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003022}
3023{
Yingyi Bu6d57e492016-06-06 21:24:42 -07003024 (key = Identifier() | key = StringLiteral())
3025 <EQ>
3026 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003027 {
3028 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003029 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07003030 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003031 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003032 }
3033 }
3034 )
3035 {
3036 return new Pair<String, String>(key.toUpperCase(), value);
3037 }
3038}
3039
Glenn58329202021-04-09 12:03:46 -07003040IndexedTypeExpression IndexedTypeExpr(boolean allowQues) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003041{
3042 TypeExpression typeExpr = null;
Glenn58329202021-04-09 12:03:46 -07003043 boolean isUnknownable = !allowQues;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003044}
3045{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003046 typeExpr = TypeExpr(false)
Glenn58329202021-04-09 12:03:46 -07003047 ( <QUES>
3048 {
3049 if (!allowQues) {
3050 throw new SqlppParseException(getSourceLocation(token), "'?' quantifier is illegal for the type expression.");
3051 }
3052 isUnknownable = true;
3053 }
3054 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003055 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003056 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003057 }
3058}
3059
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003060TypeExpression TypeExpr(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003061{
3062 TypeExpression typeExpr = null;
3063}
3064{
3065 (
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003066 typeExpr = TypeReference()
3067 | typeExpr = OrderedListTypeDef(allowRecordTypeDef)
3068 | typeExpr = UnorderedListTypeDef(allowRecordTypeDef)
3069 | typeExpr = RecordTypeDef() {
3070 if (!allowRecordTypeDef) {
3071 throw new SqlppParseException(typeExpr.getSourceLocation(), "Unexpected record type declaration");
3072 }
3073 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003074 )
3075 {
3076 return typeExpr;
3077 }
3078}
3079
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07003080RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
3081{
3082 RecordTypeDefinition.RecordKind recordKind = null;
3083}
3084{
3085 (
3086 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
3087 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
3088 )
3089 {
3090 return recordKind;
3091 }
3092}
3093
Yingyi Bu391f09e2015-10-29 13:49:39 -07003094RecordTypeDefinition RecordTypeDef() throws ParseException:
3095{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003096 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003097 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07003098 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003099}
3100{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07003101 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003102 <LEFTBRACE>
3103 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003104 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003105 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
3106 if (hintToken != null) {
3107 String hintParams = hintToken.hintParams;
3108 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
3109 if (splits == null || splits.length != 4) {
3110 throw new SqlppParseException(getSourceLocation(hintToken),
3111 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07003112 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003113 if (!splits[0].equals("int")) {
3114 throw new SqlppParseException(getSourceLocation(hintToken),
3115 "The only supported type for gen-fields is int.");
3116 }
3117 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
3118 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
3119 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003120 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003121 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003122 (
3123 RecordField(recType)
3124 ( <COMMA> RecordField(recType) )*
3125 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003126 <RIGHTBRACE>
3127 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07003128 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003129 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003130 }
3131}
3132
3133void RecordField(RecordTypeDefinition recType) throws ParseException:
3134{
3135 String fieldName;
3136 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07003137 boolean nullable = false, missable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003138}
3139{
3140 fieldName = Identifier()
3141 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003142 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
3143 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
3144 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
3145 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003146 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003147 <COLON> type = TypeExpr(true) ( <QUES> { nullable = true; missable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003148 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07003149 recType.addField(fieldName, type, nullable, missable, rfdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003150 }
3151}
3152
3153TypeReferenceExpression TypeReference() throws ParseException:
3154{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003155 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003156}
3157{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003158 id = QualifiedName()
3159 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003160 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
3161 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003162 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003163
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003164 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003165 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003166 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003167}
3168
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003169OrderedListTypeDefinition OrderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003170{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003171 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003172 TypeExpression type = null;
3173}
3174{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003175 <LEFTBRACKET> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003176 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003177 <RIGHTBRACKET>
3178 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003179 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003180 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003181 }
3182}
3183
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003184UnorderedListTypeDefinition UnorderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003185{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003186 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003187 TypeExpression type = null;
3188}
3189{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003190 <LEFTDBLBRACE> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003191 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003192 <RIGHTDBLBRACE>
3193 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003194 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003195 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003196 }
3197}
3198
3199FunctionName FunctionName() throws ParseException:
3200{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003201 Triple<List<String>, Token, Token> prefix = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003202 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003203}
3204{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003205 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
3206 // that copy must be kept in sync with this code
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003207 prefix = MultipartIdentifierWithHints(
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003208 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.RANGE_HINT, SqlppHint.HASH_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT,
Tin Vu4f1090d2021-09-21 02:06:32 -07003209 SqlppHint.SPATIAL_JOIN_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003210 )
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003211 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003212 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003213 Token startToken = prefix.second;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003214 FunctionName result = new FunctionName();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003215 result.sourceLoc = getSourceLocation(startToken);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06003216 result.hintToken = prefix.third;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003217 List<String> list = prefix.first;
3218 int ln = list.size();
3219 String last = list.get(ln - 1);
3220 if (suffix == null) {
3221 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
3222 // no library name
3223 result.function = last;
3224 } else {
3225 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
3226 // suffix = func_name
3227 result.library = last;
3228 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003229 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003230 if (ln > 1) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003231 result.dataverse = createDataverseName(list, 0, ln - 1, startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003232 } else {
3233 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003234 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003235
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003236 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
3237 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003238 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003239 return result;
3240 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003241}
3242
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003243Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003244{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003245 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003246}
3247{
3248 name = QualifiedName()
3249 {
3250 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003251 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003252 }
3253 return name;
3254 }
3255}
3256
3257String Identifier() throws ParseException:
3258{
3259 String lit = null;
3260}
3261{
3262 (<IDENTIFIER>
3263 {
3264 return token.image;
3265 }
3266 | lit = QuotedString()
3267 {
3268 return lit;
3269 }
3270 )
3271}
3272
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003273List<String> ParenthesizedIdentifierList() throws ParseException:
3274{
3275 List<String> list = new ArrayList<String>();
3276 String ident = null;
3277}
3278{
3279 <LEFTPAREN>
3280 ident = Identifier() { list.add(ident); }
3281 ( <COMMA> ident = Identifier() { list.add(ident); } )*
3282 <RIGHTPAREN>
3283 {
3284 return list;
3285 }
3286}
3287
Vijay Sarathy691e93b2022-11-20 14:31:47 -08003288Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> buildOrProbeParenthesizedIdentifier() throws ParseException:
3289{
3290 String ident1 = null;
3291 String ident2 = null;
3292}
3293{
3294 ident1 = Identifier() <LEFTPAREN> ident2 = Identifier() <RIGHTPAREN>
3295 {
3296 // check
3297 if (ident1.equals("build")) {
3298 return new Pair<HashJoinExpressionAnnotation.BuildOrProbe, String>(HashJoinExpressionAnnotation.BuildOrProbe.BUILD, ident2);
3299 }
3300 else if (ident1.equals("probe")) {
3301 return new Pair<HashJoinExpressionAnnotation.BuildOrProbe, String>(HashJoinExpressionAnnotation.BuildOrProbe.PROBE, ident2);
3302 }
3303 else {
3304 throw new SqlppParseException(getSourceLocation(token), "The string after hashjoin has to be \"build\" or \"probe\".");
3305 }
3306 return null;
3307 }
3308}
3309
3310String parenthesizedIdentifier() throws ParseException:
3311{
3312 String ident = null;
3313}
3314{
3315 <LEFTPAREN> ident = Identifier() <RIGHTPAREN>
3316 {
3317 return ident;
3318 }
3319}
3320
Tin Vu4f1090d2021-09-21 02:06:32 -07003321List<Literal> ParenthesizedLiteralList() throws ParseException:
3322{
3323 List<Literal> list = new ArrayList<Literal>();
3324 boolean minus = false;
3325 Expression litExpr = null;
3326 Literal lit = null;
3327}
3328{
3329 <LEFTPAREN>
3330 (<MINUS> { minus = true; })? litExpr = Literal()
3331 {
3332 lit = ((LiteralExpr) litExpr).getValue();
3333 if (minus)
3334 {
3335 try {
3336 lit = ExpressionUtils.reverseSign(lit);
3337 } catch (TypeMismatchException e) {
3338 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
3339 }
3340 minus = false;
3341 }
3342 list.add(lit);
3343 }
3344 ( <COMMA> (<MINUS> { minus = true; })? litExpr = Literal()
3345 {
3346 lit = ((LiteralExpr) litExpr).getValue();
3347 if (minus)
3348 {
3349 try {
3350 lit = ExpressionUtils.reverseSign(lit);
3351 } catch (TypeMismatchException e) {
3352 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
3353 }
3354 minus = false;
3355 }
3356 list.add(lit);
3357 } )*
3358 <RIGHTPAREN>
3359 {
3360 return list;
3361 }
3362}
3363
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003364Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003365{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003366 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003367 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003368}
3369{
3370 fieldList = NestedField()
Glenn58329202021-04-09 12:03:46 -07003371 ( <COLON> fieldType = IndexedTypeExpr(true) )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003372 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003373 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
3374 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07003375 }
3376}
3377
Yingyi Buc9bfe252016-03-01 00:02:40 -08003378Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003379{
3380 List<String> exprList = new ArrayList<String>();
3381 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003382 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003383 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003384}
3385{
3386 lit = Identifier()
3387 {
Yingyi Bub9169b62016-02-26 21:21:49 -08003388 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003389 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08003390 }
3391 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08003392 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08003393 <LEFTPAREN><RIGHTPAREN>
3394 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003395 if(!lit.equalsIgnoreCase("meta")){
3396 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08003397 }
3398 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003399 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08003400 }
3401 )?
3402 {
3403 if(!meetParens){
3404 exprList.add(lit);
3405 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003406 }
3407 (<DOT>
3408 lit = Identifier()
3409 {
3410 exprList.add(lit);
3411 }
3412 )*
3413 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08003414 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003415 }
3416}
3417
Yingyi Bu6d57e492016-06-06 21:24:42 -07003418String ConstantString() throws ParseException:
3419{
3420 String value = null;
3421}
3422{
3423 (value = QuotedString() | value = StringLiteral())
3424 {
3425 return value;
3426 }
3427}
3428
Yingyi Bu391f09e2015-10-29 13:49:39 -07003429String QuotedString() throws ParseException:
3430{
3431}
3432{
3433 <QUOTED_STRING>
3434 {
3435 return removeQuotesAndEscapes(token.image);
3436 }
3437}
3438
Yingyi Bu391f09e2015-10-29 13:49:39 -07003439String StringLiteral() throws ParseException:
3440{
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003441 StringBuilder str = null;
3442 char quote = 0;
3443 boolean ext = false;
3444 Token litToken = null;
3445 String lit = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003446}
3447{
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003448 ( <STRING_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003449 {
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003450 String quoted = token.image;
3451 char q = quoted.charAt(0);
3452 boolean e = q == 'E';
3453 if (e) {
3454 q = quoted.charAt(1);
3455 quoted = quoted.substring(1);
3456 }
3457 if (lit == null) {
3458 quote = q;
3459 ext = e;
3460 } else {
3461 boolean isAdjacent = litToken.endLine == token.beginLine && litToken.endColumn + 1 == token.beginColumn;
3462 if (!isAdjacent || ext || e || (q != quote)) {
3463 throw new SqlppParseException(getSourceLocation(token), "Invalid string literal");
3464 }
3465 if (str == null) {
3466 str = new StringBuilder();
3467 }
3468 str.append(lit);
3469 str.append(quote);
3470 }
3471 lit = removeQuotesAndEscapes(quoted);
3472 litToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003473 }
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003474 )+
3475 {
3476 if (str == null) {
3477 return lit;
3478 } else {
3479 str.append(lit);
3480 return str.toString();
3481 }
3482 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003483}
3484
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003485Triple<List<String>, Token, Token> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003486{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003487 Triple<List<String>, Token, Token> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003488}
3489{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003490 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003491 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003492 return result;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003493 }
3494}
3495
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003496Triple<List<String>, Token, Token> MultipartIdentifierWithHints(SqlppHint... expectedHints)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003497 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003498{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003499 List<String> list = new ArrayList<String>();
3500 SourceLocation sourceLoc = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003501 Token startToken, hintToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003502 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003503}
3504{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003505 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003506 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003507 list.add(item);
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003508 startToken = token;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003509 if (expectedHints != null && expectedHints.length > 0) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003510 hintToken = fetchHint(token, expectedHints);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003511 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003512 }
3513 (<DOT> item = Identifier() { list.add(item); } )*
3514 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003515 return new Triple<List<String>, Token, Token>(list, startToken, hintToken);
3516 }
3517}
3518
3519DataverseName DataverseName() throws ParseException:
3520{
3521 Triple<List<String>, Token, Token> ident = null;
3522}
3523{
3524 ident = MultipartIdentifier()
3525 {
3526 List<String> list = ident.first;
3527 Token startToken = ident.second;
3528 return createDataverseName(list, 0, list.size(), startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003529 }
3530}
3531
3532Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
3533{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003534 Triple<List<String>, Token, Token> ident = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003535}
3536{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003537 ident = MultipartIdentifier()
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003538 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003539 List<String> list = ident.first;
3540 Token startToken = ident.second;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003541 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003542 DataverseName id1 = len > 1 ? createDataverseName(list, 0, len - 1, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003543 Identifier id2 = new Identifier(list.get(len - 1));
3544 return new Pair<DataverseName,Identifier>(id1, id2);
3545 }
3546}
3547
3548Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
3549{
3550 List<String> list = new ArrayList<String>();
3551 String item = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003552 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003553}
3554{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003555 item = Identifier() { list.add(item); startToken = token; }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003556 (<DOT> item = Identifier() { list.add(item); } )+
3557 {
3558 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003559 DataverseName id1 = len > 2 ? createDataverseName(list, 0, len - 2, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003560 Identifier id2 = new Identifier(list.get(len - 2));
3561 Identifier id3 = new Identifier(list.get(len - 1));
3562 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003563 }
3564}
3565
3566FunctionDecl FunctionDeclaration() throws ParseException:
3567{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003568 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003569 String functionName;
Dmitry Lychagin541652082020-11-09 14:04:53 -08003570 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003571 Expression funcBody;
3572 createNewScope();
3573}
3574{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003575 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07003576 functionName = Identifier()
Dmitry Lychagin541652082020-11-09 14:04:53 -08003577 paramsWithArity = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07003578 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003579 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07003580 <RIGHTBRACE>
3581 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08003582 int arity = paramsWithArity.first;
3583 List<Pair<VarIdentifier,TypeExpression>> paramList = paramsWithArity.second;
3584 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, arity);
Xikui Wang3de700a2018-03-15 16:32:55 -07003585 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003586 ensureNoTypeDeclsInFunction(functionName, paramList, null, startToken);
3587 List<VarIdentifier> params = new ArrayList<VarIdentifier>(paramList.size());
3588 for (Pair<VarIdentifier,TypeExpression> p: paramList) {
3589 params.add(p.getFirst());
Ian Maxon38fe9402020-01-29 19:27:40 -08003590 }
Dmitry Lychagin9ba74872021-04-05 14:38:20 -07003591 FunctionDecl stmt = new FunctionDecl(signature, params, funcBody, false);
Xikui Wang3de700a2018-03-15 16:32:55 -07003592 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003593 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07003594 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003595}
3596
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003597Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003598{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003599 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003600 Expression expr;
3601}
3602{
3603 (
3604 expr = Expression()
3605 |
3606 expr = SelectExpression(false)
3607 )
3608 {
3609 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003610 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003611 return query;
3612 }
3613}
3614
3615
Yingyi Bu391f09e2015-10-29 13:49:39 -07003616Expression Expression():
3617{
3618 Expression expr = null;
3619 Expression exprP = null;
3620}
3621{
3622(
3623 LOOKAHEAD(2)
3624 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003625 | expr = QuantifiedExpression()
3626)
3627 {
3628 return (exprP==null) ? expr : exprP;
3629 }
3630}
3631
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003632Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003633{
3634 OperatorExpr op = null;
3635 Expression operand = null;
3636}
3637{
3638 operand = AndExpr()
3639 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003640 <OR>
3641 {
3642 if (op == null) {
3643 op = new OperatorExpr();
3644 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07003645 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003646 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003647 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003648 try{
3649 op.addOperator(token.image.toLowerCase());
3650 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003651 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003652 }
Xikui Wang3de700a2018-03-15 16:32:55 -07003653 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003654
Xikui Wang3de700a2018-03-15 16:32:55 -07003655 operand = AndExpr()
3656 {
3657 op.addOperand(operand);
3658 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003659
3660 )*
3661
3662 {
3663 return op==null? operand: op;
3664 }
3665}
3666
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003667Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003668{
3669 OperatorExpr op = null;
3670 Expression operand = null;
3671}
3672{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003673 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003674 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003675 <AND>
3676 {
3677 if (op == null) {
3678 op = new OperatorExpr();
3679 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003680 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003681 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003682 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003683 try{
3684 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003685 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003686 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003687 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003688 }
3689
Yingyi Bu196db5d2016-07-15 19:07:20 -07003690 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003691 {
3692 op.addOperand(operand);
3693 }
3694
3695 )*
3696
3697 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003698 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003699 }
3700}
3701
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003702Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07003703{
3704 Expression inputExpr;
3705 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003706 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07003707}
3708{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003709 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07003710 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003711 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003712 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003713 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003714 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003715 } else {
3716 return inputExpr;
3717 }
3718 }
3719}
Yingyi Bu391f09e2015-10-29 13:49:39 -07003720
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003721Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003722{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003723 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003724 OperatorExpr op = null;
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003725 Token opToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003726 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003727 IExpressionAnnotation annotation = null;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003728 List<IExpressionAnnotation> annotationList = new ArrayList<IExpressionAnnotation>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003729}
3730{
Yingyi Bu6c638342016-09-02 17:54:34 -07003731 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003732
3733 (
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003734 LOOKAHEAD(3)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN> |
3735 <IS> (<NOT> { not = true; })? <DISTINCT> { opToken = token; } <FROM> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003736 {
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003737 if (opToken == null) {
3738 opToken = token;
3739 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003740 Token hintToken = fetchHint(token,
3741 SqlppHint.HASH_BROADCAST_JOIN_HINT, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003742 SqlppHint.HASH_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT,
3743 SqlppHint.SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT, SqlppHint.JOIN_PREDICATE_PRODUCTIVITY_HINT
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003744 );
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003745 while (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003746 annotation = parseExpressionAnnotation(hintToken);
Vijay Sarathyebf08b82023-03-20 18:31:35 -07003747 if (annotation != null) {
3748 // annotation may be null if hints are malformed
3749 annotationList.add(annotation);
3750 }
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003751 hintToken = hintToken.specialToken;
Vijay Sarathy8f4fccd2022-12-05 14:29:02 -08003752 if (hintToken != null) {
3753 SourceLocation sourceLoc = getSourceLocation(hintToken);
3754 hintCollector.remove(sourceLoc);
3755 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003756 }
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003757 String operator = opToken.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07003758 if (operator.equals("<>")){
3759 operator = "!=";
3760 }
3761 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003762 operator = "not_" + operator;
3763 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003764 if (op == null) {
3765 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07003766 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003767 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003768 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003769 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003770 try{
3771 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003772 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003773 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003774 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003775 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003776
Yingyi Bu6c638342016-09-02 17:54:34 -07003777 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07003778 {
Shiva2ea73232019-10-09 18:04:05 -07003779 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07003780 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003781 )?
3782
3783 {
3784 if (annotation != null) {
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003785 op.addHints(annotationList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003786 }
3787 return op==null? operand: op;
3788 }
3789}
3790
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003791Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07003792{
3793 boolean not = false;
3794 OperatorExpr op = null;
3795 Expression operand = null;
3796 IExpressionAnnotation annotation = null;
3797}
3798{
3799 operand = IsExpr()
3800 (
3801 LOOKAHEAD(2)
3802 (<NOT> { not = true; })? <BETWEEN>
3803 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003804 Token hintToken = fetchHint(token,
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003805 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.HASH_JOIN_HINT,
3806 SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003807 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003808 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003809 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu6c638342016-09-02 17:54:34 -07003810 }
3811 String operator = token.image.toLowerCase();
3812 if(not){
3813 operator = "not_" + operator;
3814 }
3815 if (op == null) {
3816 op = new OperatorExpr();
3817 op.addOperand(operand);
3818 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003819 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07003820 }
3821 try{
3822 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003823 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003824 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07003825 }
3826 }
3827
3828 operand = IsExpr()
3829 {
3830 op.addOperand(operand);
3831 }
3832
3833 <AND>
3834 operand = IsExpr()
3835 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08003836 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07003837 op.addOperand(operand);
3838 }
3839 )?
3840
3841 {
3842 if (annotation != null) {
3843 op.addHint(annotation);
3844 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003845 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07003846 }
3847}
3848
Yingyi Budaa549c2016-06-28 22:30:52 -07003849Expression IsExpr() throws ParseException:
3850{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003851 Token notToken = null;
3852 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003853 Expression operand = null;
3854 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003855 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003856}
3857{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003858 operand = LikeExpr()
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003859 (
3860 LOOKAHEAD(3)
3861 <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003862 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003863 (
3864 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
3865 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08003866 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07003867 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003868 )
Yingyi Budaa549c2016-06-28 22:30:52 -07003869 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003870 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07003871 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003872 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003873 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003874 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07003875 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003876 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07003877 }
3878 }
3879 )?
3880 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003881 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07003882 }
3883}
3884
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003885Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003886{
3887 boolean not = false;
3888 OperatorExpr op = null;
3889 Expression operand = null;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003890 IExpressionAnnotation annotation = null;
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003891}
3892{
3893 operand = ConcatExpr()
3894 (
3895 LOOKAHEAD(2)
3896 (<NOT> { not = true; })? <LIKE>
3897 {
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003898 Token hintToken = fetchHint(token, SqlppHint.SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT);
3899 if (hintToken != null) {
3900 annotation = parseExpressionAnnotation(hintToken);
3901 }
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003902 op = new OperatorExpr();
3903 op.addOperand(operand);
3904 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003905 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003906
3907 String operator = token.image.toLowerCase();
3908 if (not) {
3909 operator = "not_" + operator;
3910 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003911 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003912 op.addOperator(operator);
3913 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003914 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003915 }
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003916 if (annotation != null) {
3917 op.addHint(annotation);
3918 }
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003919 }
3920
3921 operand = ConcatExpr()
3922 {
3923 op.addOperand(operand);
3924 }
3925 )?
3926
3927 {
3928 return op == null ? operand : op;
3929 }
3930}
3931
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003932Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003933{
3934 OperatorExpr op = null;
3935 Expression operand = null;
3936}
3937{
3938 operand = AddExpr()
3939 (
3940 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003941 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003942 {
3943 if (op == null) {
3944 op = new OperatorExpr();
3945 op.addOperand(operand);
3946 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003947 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003948 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003949 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003950 }
3951 operand = AddExpr()
3952 {
3953 op.addOperand(operand);
3954 }
3955 )*
3956
3957 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003958 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003959 }
3960}
Yingyi Budaa549c2016-06-28 22:30:52 -07003961
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003962Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003963{
3964 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003965 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003966 Expression operand = null;
3967}
3968{
3969 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07003970 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003971 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003972 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003973 {
3974 if (op == null) {
3975 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003976 op.addOperand(operand);
3977 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003978 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003979 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003980 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003981 }
3982
3983 operand = MultExpr()
3984 {
3985 op.addOperand(operand);
3986 }
3987 )*
3988
3989 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003990 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003991 }
3992}
3993
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003994Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003995{
3996 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003997 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003998 Expression operand = null;
3999}
4000{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004001 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004002
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004003 ( (
4004 <MUL> { opType = OperatorType.MUL; } |
4005 <DIVIDE> { opType = OperatorType.DIVIDE; } |
4006 <DIV> { opType = OperatorType.DIV; } |
4007 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
4008 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004009 {
4010 if (op == null) {
4011 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004012 op.addOperand(operand);
4013 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004014 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004015 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07004016 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004017 }
4018 operand = ExponentExpr()
4019 {
4020 op.addOperand(operand);
4021 }
4022 )*
4023
4024 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004025 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004026 }
4027}
4028
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004029Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004030{
4031 OperatorExpr op = null;
4032 Expression operand = null;
4033}
4034{
4035 operand = UnaryExpr()
4036 (<CARET>
4037 {
4038 if (op == null) {
4039 op = new OperatorExpr();
4040 op.addOperand(operand);
4041 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004042 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004043 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004044 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004045 }
4046 operand = UnaryExpr()
4047 {
4048 op.addOperand(operand);
4049 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004050 )?
4051 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004052 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004053 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004054}
4055
4056Expression UnaryExpr() throws ParseException:
4057{
Yingyi Bu196db5d2016-07-15 19:07:20 -07004058 boolean not = false;
4059 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004060 Expression expr = null;
4061}
Yingyi Budaa549c2016-06-28 22:30:52 -07004062{
Yingyi Bu196db5d2016-07-15 19:07:20 -07004063 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004064 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07004065 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004066 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07004067 exprType = "not_" + exprType;
4068 }
4069 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004070 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004071 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07004072 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08004073 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004074 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07004075 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004076 }
4077 )?
4078
4079 expr = ValueExpr()
4080 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004081 if (uexpr == null) {
4082 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004083 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004084 uexpr.setExpr(expr);
4085 return uexpr;
4086 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004087 }
4088}
4089
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004090Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004091{
4092 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004093 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004094}
4095{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03004096 expr = PrimaryExpr()
4097 (
4098 accessor = FieldAccessor(accessor != null ? accessor : expr)
4099 |
4100 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004101 )*
4102 {
4103 return accessor == null ? expr : accessor;
4104 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004105}
4106
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004107FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004108{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004109 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004110 String ident = null;
4111}
4112{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004113 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004114 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004115 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004116 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004117 }
4118}
4119
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03004120AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004121{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004122 Token startToken = null;
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004123 boolean star = false, slice = false;
4124 Expression expr1 = null, expr2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004125}
4126{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004127 <LEFTBRACKET> { startToken = token; }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004128 (
4129 <MUL> { star = true; }
4130 |
4131 ( expr1 = Expression() ( <COLON> { slice = true; } ( expr2 = Expression() )? )? )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004132 )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004133 <RIGHTBRACKET>
4134 {
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004135 if (expr1 != null && expr1.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
4136 ensureIntegerLiteral( (LiteralExpr) expr1, "Index");
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03004137 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004138 if (expr2 != null && expr2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
4139 ensureIntegerLiteral( (LiteralExpr) expr2, "Index");
4140 }
4141 AbstractAccessor resultAccessor;
4142 if (slice) {
4143 resultAccessor = new ListSliceExpression(inputExpr, expr1, expr2);
4144 } else if (star) {
4145 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.STAR, null);
4146 } else {
4147 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.ELEMENT, expr1);
4148 }
4149 return addSourceLocation(resultAccessor, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004150 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004151}
4152
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004153Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004154{
4155 Expression expr = null;
4156}
4157{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08004158 (
4159 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07004160 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004161 | expr = Literal()
4162 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004163 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004164 | expr = ListConstructor()
4165 | expr = RecordConstructor()
4166 | expr = ParenthesizedExpression()
4167 )
4168 {
4169 return expr;
4170 }
4171}
4172
4173Expression Literal() throws ParseException:
4174{
4175 LiteralExpr lit = new LiteralExpr();
4176 String str = null;
4177}
4178{
4179 ( str = StringLiteral()
4180 {
4181 lit.setValue(new StringLiteral(str));
4182 }
4183 | <INTEGER_LITERAL>
4184 {
Till Westmann68c6a992016-09-30 00:19:12 -07004185 try {
4186 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
4187 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08004188 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07004189 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004190 }
4191 | <FLOAT_LITERAL>
4192 {
Till Westmann68c6a992016-09-30 00:19:12 -07004193 try {
4194 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
4195 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08004196 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07004197 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004198 }
4199 | <DOUBLE_LITERAL>
4200 {
Till Westmann68c6a992016-09-30 00:19:12 -07004201 try {
4202 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
4203 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08004204 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07004205 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004206 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07004207 | <MISSING>
4208 {
4209 lit.setValue(MissingLiteral.INSTANCE);
4210 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004211 | <NULL>
4212 {
4213 lit.setValue(NullLiteral.INSTANCE);
4214 }
4215 | <TRUE>
4216 {
4217 lit.setValue(TrueLiteral.INSTANCE);
4218 }
4219 | <FALSE>
4220 {
4221 lit.setValue(FalseLiteral.INSTANCE);
4222 }
4223 )
4224 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004225 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004226 }
4227}
4228
Yingyi Bu391f09e2015-10-29 13:49:39 -07004229VariableExpr VariableRef() throws ParseException:
4230{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004231 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004232}
4233{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08004234 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004235 {
4236 Identifier ident = lookupSymbol(id);
4237 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004238 throw new SqlppParseException(getSourceLocation(token),
4239 "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 -07004240 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004241 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004242 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004243 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004244 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004245 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07004246 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004247 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004248 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004249 }
4250}
4251
Yingyi Bu391f09e2015-10-29 13:49:39 -07004252VariableExpr Variable() throws ParseException:
4253{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004254 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004255}
4256{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08004257 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004258 {
4259 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004260 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
4261 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07004262 varExp.setIsNewVar(false);
4263 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004264 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004265 }
4266}
4267
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08004268String VariableIdentifier() throws ParseException:
4269{
4270 String id = null;
4271}
4272{
4273 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
4274 {
4275 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
4276 }
4277}
4278
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004279Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
4280{
4281 VariableExpr var = null;
4282 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
4283}
4284{
4285 var = Variable()
4286 ( LOOKAHEAD(1)
4287 {
4288 VariableExpr fieldVarExpr = null;
4289 String fieldIdentifierStr = null;
4290 }
4291 <LEFTPAREN>
4292 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
4293 {
4294 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
4295 }
4296 (<COMMA>
4297 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
4298 {
4299 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
4300 }
4301 )*
4302 <RIGHTPAREN>
4303 )?
4304 {
4305 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
4306 }
4307}
4308
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004309VariableExpr ExternalVariableRef() throws ParseException:
4310{
4311 String name = null;
4312}
4313{
4314 (
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004315 <DOLLAR_IDENTIFIER> { name = token.image.substring(1); }
4316 | <DOLLAR_INTEGER_LITERAL> { name = token.image.substring(1); }
4317 | <DOLLAR_QUOTED_STRING> { name = removeQuotesAndEscapes(token.image.substring(1)); }
4318 | <QUES> { name = String.valueOf(++externalVarCounter); }
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004319 )
4320 {
4321 String idName = SqlppVariableUtil.toExternalVariableName(name);
4322 VarIdentifier id = new VarIdentifier(idName);
4323 VariableExpr varExp = new VariableExpr(id);
4324 return addSourceLocation(varExp, token);
4325 }
4326}
4327
Yingyi Bu391f09e2015-10-29 13:49:39 -07004328Expression ListConstructor() throws ParseException:
4329{
4330 Expression expr = null;
4331}
4332{
4333 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004334 expr = OrderedListConstructor() |
4335 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004336 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004337 {
4338 return expr;
4339 }
4340}
4341
Yingyi Bu391f09e2015-10-29 13:49:39 -07004342ListConstructor OrderedListConstructor() throws ParseException:
4343{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004344 Token startToken = null;
4345 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004346}
4347{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004348 <LEFTBRACKET> { startToken = token; }
4349 exprList = ExpressionList()
4350 <RIGHTBRACKET>
4351 {
4352 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004353 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004354 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004355}
4356
4357ListConstructor UnorderedListConstructor() throws ParseException:
4358{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004359 Token startToken = null;
4360 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004361}
4362{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004363 <LEFTDBLBRACE> { startToken = token; }
4364 exprList = ExpressionList()
4365 <RIGHTDBLBRACE>
4366 {
4367 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004368 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004369 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004370}
4371
4372List<Expression> ExpressionList() throws ParseException:
4373{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004374 Expression expr = null;
4375 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004376}
4377{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004378 (
4379 expr = Expression()
4380 {
4381 exprList.add(expr);
4382 }
4383 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07004384 {
4385 exprList.add(expr);
4386 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004387 )*
4388 )?
4389 {
4390 return exprList;
4391 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004392}
4393
Yingyi Bu391f09e2015-10-29 13:49:39 -07004394RecordConstructor RecordConstructor() throws ParseException:
4395{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004396 Token startToken = null;
4397 FieldBinding fb = null;
4398 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004399}
4400{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004401 <LEFTBRACE> { startToken = token; }
4402 (
4403 fb = FieldBinding() { fbList.add(fb); }
4404 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
4405 )?
4406 <RIGHTBRACE>
4407 {
4408 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004409 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004410 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004411}
4412
4413FieldBinding FieldBinding() throws ParseException:
4414{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004415 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004416}
4417{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004418 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004419 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004420 if (right == null) {
4421 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
4422 if (generatedIdentifier == null) {
4423 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
4424 }
4425 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07004426 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
4427 generatedNameExpr.setSourceLocation(left.getSourceLocation());
4428 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004429 } else {
4430 return new FieldBinding(left, right);
4431 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004432 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004433}
4434
Yingyi Bu391f09e2015-10-29 13:49:39 -07004435Expression FunctionCallExpr() throws ParseException:
4436{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004437 List<Expression> argList = new ArrayList<Expression>();
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004438 Expression argExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004439 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07004440 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07004441 boolean distinct = false;
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004442 Expression filterExpr = null;
4443 WindowExpression windowExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004444}
4445{
4446 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07004447 <LEFTPAREN> (
4448 ( <DISTINCT> { distinct = true; } )?
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004449 ( argExpr = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004450 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004451 if (star) {
4452 if (funcName.function.equalsIgnoreCase(BuiltinFunctions.SCALAR_COUNT.getName())) {
4453 argExpr = new LiteralExpr(new LongIntegerLiteral(1L));
4454 } else {
4455 throw new SqlppParseException(getSourceLocation(token),
4456 "The parameter * can only be used in " + BuiltinFunctions.SCALAR_COUNT.getName() + "().");
Yingyi Buf4d09842016-08-26 00:03:52 -07004457 }
Yingyi Buf4d09842016-08-26 00:03:52 -07004458 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004459 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004460 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004461 (<COMMA> argExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004462 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004463 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004464 }
4465 )*)? <RIGHTPAREN>
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004466
4467 {
4468 String name = funcName.function;
4469 if (distinct) {
4470 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
4471 }
4472 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
4473 int arity = argList.size();
4474 FunctionSignature signature = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
4475 if (signature == null) {
4476 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
4477 }
4478 }
4479
4480 ( <FILTER> <LEFTPAREN> <WHERE> filterExpr = Expression() <RIGHTPAREN> )?
4481
4482 ( LOOKAHEAD(5) windowExpr = WindowExpr(signature, argList, filterExpr) )?
4483
4484 {
4485 if (windowExpr != null) {
4486 return windowExpr;
4487 } else {
4488 CallExpr callExpr = new CallExpr(signature, argList, filterExpr);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06004489 if (funcName.hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08004490 IExpressionAnnotation annotation = parseExpressionAnnotation(funcName.hintToken);
4491 if (annotation != null) {
4492 callExpr.addHint(annotation);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004493 }
4494 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004495 FunctionMapUtil.normalizedListInputFunctions(callExpr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004496 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004497 return callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004498 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004499 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004500}
4501
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004502WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr)
4503 throws ParseException:
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004504{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004505 WindowExpression windowExpr = null;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004506 Boolean fromLast = null, ignoreNulls = null;
4507}
4508{
4509 (
4510 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
4511 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
4512 <FROM> <IDENTIFIER>
4513 {
4514 if (isToken(FIRST)) {
4515 fromLast = false;
4516 } else if (isToken(LAST)) {
4517 fromLast = true;
4518 } else {
4519 throw createUnexpectedTokenError();
4520 }
4521 }
4522 )?
4523 (
4524 // ( RESPECT | IGNORE ) NULLS OVER
4525 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
4526 <IDENTIFIER>
4527 {
4528 if (isToken(RESPECT)) {
4529 ignoreNulls = false;
4530 } else if (isToken(IGNORE)) {
4531 ignoreNulls = true;
4532 } else {
4533 throw createUnexpectedTokenError();
4534 }
4535 }
4536 <IDENTIFIER>
4537 {
4538 if (!isToken(NULLS)) {
4539 throw createUnexpectedTokenError();
4540 }
4541 }
4542 )?
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004543 <OVER> windowExpr = OverClause(signature, argList, aggFilterExpr, token, fromLast, ignoreNulls)
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004544 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004545 return windowExpr;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004546 }
4547}
4548
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004549WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr,
4550 Token startToken, Boolean fromLast, Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004551{
4552 Expression partitionExpr = null;
4553 List<Expression> partitionExprs = new ArrayList<Expression>();
4554 OrderbyClause orderByClause = null;
4555 List<Expression> orderbyList = null;
4556 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004557 List<OrderbyClause.NullOrderModifier> orderbyNullModifierList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004558 WindowExpression.FrameMode frameMode = null;
4559 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
4560 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
4561 Expression frameStartExpr = null, frameEndExpr = null;
4562 WindowExpression.FrameExclusionKind frameExclusionKind = null;
4563 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
4564 VariableExpr windowVar = null;
4565 List<Pair<Expression, Identifier>> windowFieldList = null;
4566}
4567{
4568 (
4569 windowVarWithFieldList = VariableWithFieldMap() <AS>
4570 {
4571 windowVar = windowVarWithFieldList.first;
4572 windowFieldList = windowVarWithFieldList.second;
4573 }
4574 )?
4575 <LEFTPAREN>
4576 (
4577 <IDENTIFIER> { expectToken(PARTITION); } <BY>
4578 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
4579 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
4580 )?
4581 (
4582 orderByClause = OrderbyClause()
4583 {
4584 orderbyList = orderByClause.getOrderbyList();
4585 orderbyModifierList = orderByClause.getModifierList();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004586 orderbyNullModifierList = orderByClause.getNullModifierList();
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004587 }
4588 (
4589 frameMode = WindowFrameMode()
4590 (
4591 frameStart = WindowFrameBoundary() |
4592 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
4593 )
4594 ( frameExclusionKind = WindowFrameExclusion() )?
4595 {
4596 frameStartKind = frameStart.first;
4597 frameStartExpr = frameStart.second;
4598 if (frameEnd == null) {
4599 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4600 } else {
4601 frameEndKind = frameEnd.first;
4602 frameEndExpr = frameEnd.second;
4603 }
4604 if (frameExclusionKind == null) {
4605 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
4606 }
4607 }
4608 )?
4609 )?
4610 <RIGHTPAREN>
4611 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004612 WindowExpression winExpr = new WindowExpression(signature, argList, aggFilterExpr, partitionExprs, orderbyList,
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004613 orderbyModifierList, orderbyNullModifierList, frameMode, frameStartKind, frameStartExpr, frameEndKind,
4614 frameEndExpr, frameExclusionKind, windowVar, windowFieldList, ignoreNulls, fromLast);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004615 return addSourceLocation(winExpr, startToken);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004616 }
4617}
4618
4619WindowExpression.FrameMode WindowFrameMode() throws ParseException:
4620{
4621}
4622{
4623 <IDENTIFIER>
4624 {
4625 if (isToken(RANGE)) {
4626 return WindowExpression.FrameMode.RANGE;
4627 } else if (isToken(ROWS)) {
4628 return WindowExpression.FrameMode.ROWS;
4629 } else if (isToken(GROUPS)) {
4630 return WindowExpression.FrameMode.GROUPS;
4631 } else {
4632 throw createUnexpectedTokenError();
4633 }
4634 }
4635}
4636
4637Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
4638{
4639 boolean current = false;
4640 Expression expr = null;
4641}
4642{
4643 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004644 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004645 | expr = Expression()
4646 )
4647 <IDENTIFIER>
4648 {
4649 WindowExpression.FrameBoundaryKind kind;
4650 if (current && isToken(ROW)) {
4651 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4652 } else if (!current && isToken(PRECEDING)) {
4653 kind = expr == null
4654 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
4655 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
4656 } else if (!current && isToken(FOLLOWING)) {
4657 kind = expr == null
4658 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
4659 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
4660 } else {
4661 throw createUnexpectedTokenError();
4662 }
4663 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
4664 }
4665}
4666
4667WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
4668{
4669 boolean current = false, no = false;
4670}
4671{
4672 <IDENTIFIER>
4673 {
4674 expectToken(EXCLUDE);
4675 }
4676 (
4677 <GROUP>
4678 {
4679 return WindowExpression.FrameExclusionKind.GROUP;
4680 }
4681 |
4682 (
4683 <IDENTIFIER>
4684 {
4685 if (isToken(TIES)) {
4686 return WindowExpression.FrameExclusionKind.TIES;
4687 } else if (isToken(CURRENT)) {
4688 current = true;
4689 } else if (isToken(NO)) {
4690 no = true;
4691 } else {
4692 throw createUnexpectedTokenError();
4693 }
4694 }
4695 <IDENTIFIER>
4696 {
4697 if (current && isToken(ROW)) {
4698 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
4699 } else if (no && isToken(OTHERS)) {
4700 return WindowExpression.FrameExclusionKind.NO_OTHERS;
4701 } else {
4702 throw createUnexpectedTokenError();
4703 }
4704 }
4705 )
4706 )
4707}
4708
Yingyi Bu391f09e2015-10-29 13:49:39 -07004709Expression ParenthesizedExpression() throws ParseException:
4710{
Dmitry Lychagine9630592021-08-30 15:29:30 -07004711 Token startToken = null;
4712 Expression expr = null, expr2 = null;
4713 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004714}
4715{
4716 (
4717 LOOKAHEAD(2)
Dmitry Lychagine9630592021-08-30 15:29:30 -07004718 <LEFTPAREN> { startToken = token; } expr = Expression()
4719 (
4720 <COMMA> expr2 = Expression()
4721 {
4722 if (exprList == null) {
4723 exprList = new ArrayList<Expression>();
4724 exprList.add(expr);
4725 }
4726 exprList.add(expr2);
4727 }
4728 )*
4729 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004730 |
4731 expr = Subquery()
4732 )
4733 {
Dmitry Lychagine9630592021-08-30 15:29:30 -07004734 if (exprList != null) {
4735 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
4736 return addSourceLocation(listExpr, startToken);
4737 } else {
4738 return expr;
4739 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004740 }
4741}
4742
Yingyi Buc8c067c2016-07-25 23:37:19 -07004743Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004744{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004745 Token startToken = null;
4746 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004747 List<Expression> whenExprs = new ArrayList<Expression>();
4748 List<Expression> thenExprs = new ArrayList<Expression>();
4749 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004750 Expression whenExpr = null;
4751 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004752}
4753{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004754 <CASE> { startToken = token; }
4755 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07004756 (
4757 <WHEN> whenExpr = Expression()
4758 {
4759 whenExprs.add(whenExpr);
4760 }
4761 <THEN> thenExpr = Expression()
4762 {
4763 thenExprs.add(thenExpr);
4764 }
4765 )*
4766 (<ELSE> elseExpr = Expression() )?
4767 <END>
4768 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004769 if (conditionExpr == null) {
4770 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004771 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004772 }
4773 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004774 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07004775 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004776}
4777
Yingyi Buab817482016-08-19 21:29:31 -07004778SelectExpression SelectExpression(boolean subquery) throws ParseException:
4779{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004780 List<LetClause> letClauses = new ArrayList<LetClause>();
4781 SelectSetOperation selectSetOperation;
4782 OrderbyClause orderbyClause = null;
4783 LimitClause limitClause = null;
4784 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07004785}
4786{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004787 ( letClauses = LetClause() )?
4788 selectSetOperation = SelectSetOperation()
4789 (orderbyClause = OrderbyClause() {})?
4790 (limitClause = LimitClause() {})?
4791 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004792 SelectExpression selectExpr =
4793 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
4794 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
4795 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004796 }
4797}
4798
Yingyi Buab817482016-08-19 21:29:31 -07004799SelectSetOperation SelectSetOperation() throws ParseException:
4800{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004801 SetOperationInput setOperationInputLeft;
4802 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
4803}
4804{
4805 {
4806 SelectBlock selectBlockLeft = null;
4807 SelectExpression subqueryLeft = null;
4808 Expression expr = null;
4809 }
4810 selectBlockLeft = SelectBlock()
4811 {
4812 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
4813 }
4814 (
4815 {
4816 SetOpType opType = SetOpType.UNION;
4817 boolean setSemantics = true;
4818 SelectBlock selectBlockRight = null;
4819 SelectExpression subqueryRight = null;
4820 }
4821 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
4822 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
4823 {
4824 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
4825 }
4826 )*
4827 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004828 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
4829 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
4830 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004831 }
4832}
4833
Yingyi Buab817482016-08-19 21:29:31 -07004834SelectExpression Subquery() throws ParseException:
4835{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004836 SelectExpression selectExpr = null;
4837}
4838{
4839 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
4840 {
4841 return selectExpr;
4842 }
4843}
4844
Yingyi Buab817482016-08-19 21:29:31 -07004845SelectBlock SelectBlock() throws ParseException:
4846{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004847 SelectClause selectClause = null;
4848 FromClause fromClause = null;
4849 List<LetClause> fromLetClauses = null;
4850 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004851 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004852 GroupbyClause groupbyClause = null;
4853 List<LetClause> gbyLetClauses = null;
4854 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004855 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004856 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004857}
4858{
4859 (
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004860 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004861 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004862 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004863 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004864 fromClause = FromClause()
4865 (
4866 fromLetClauses = LetClause()
4867 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004868 (whereClause = WhereClause())?
4869 (
4870 groupbyClause = GroupbyClause()
4871 (
4872 gbyLetClauses = LetClause()
4873 )?
4874 (havingClause = HavingClause())?
4875 )?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004876 )
4877 |
4878 (
4879 fromLetClauses = LetClause()
4880 {
4881 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
4882 SourceLocation sourceLoc = getSourceLocation(token);
4883 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
4884 missingExpr.setSourceLocation(sourceLoc);
4885 List<Expression> list = new ArrayList<Expression>(1);
4886 list.add(missingExpr);
4887 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
4888 listExpr.setSourceLocation(sourceLoc);
4889 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
4890 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
4891 fromVar.setSourceLocation(sourceLoc);
4892 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
4893 fromClause = new FromClause(fromTerms);
4894 }
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004895 (whereClause = WhereClause())?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004896 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004897 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004898 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004899 |
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004900 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004901 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004902 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004903 fromLetClauses = LetClause()
4904 )?
4905 (whereClause = WhereClause())?
4906 (
4907 groupbyClause = GroupbyClause()
4908 (
4909 gbyLetClauses = LetClause()
4910 )?
4911 (havingClause = HavingClause())?
4912 )?
4913 selectClause = SelectClause()
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004914 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004915 )
4916 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004917 if (fromLetClauses != null) {
4918 fromLetWhereClauses.addAll(fromLetClauses);
4919 }
4920 if (whereClause != null) {
4921 fromLetWhereClauses.add(whereClause);
4922 }
4923 if (gbyLetClauses != null) {
4924 gbyLetHavingClauses.addAll(gbyLetClauses);
4925 }
4926 if (havingClause != null) {
4927 gbyLetHavingClauses.add(havingClause);
4928 }
4929 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
4930 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004931 selectBlock.setSourceLocation(startSrcLoc);
4932 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004933 }
4934}
4935
Yingyi Buab817482016-08-19 21:29:31 -07004936SelectClause SelectClause() throws ParseException:
4937{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004938 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004939 SelectRegular selectRegular = null;
4940 SelectElement selectElement = null;
ggalvizo82499442022-10-24 12:46:15 -07004941 List<List<String>> fieldExclusions = new ArrayList<List<String>>();
4942 List<String> nestedField = new ArrayList<String>();
4943 String identifier;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004944 boolean distinct = false;
4945}
4946{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004947 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004948 (
ggalvizo82499442022-10-24 12:46:15 -07004949 (
4950 selectRegular = SelectRegular()
4951 ( LOOKAHEAD({laIdentifier(EXCLUDE)}) <IDENTIFIER>
4952 identifier = Identifier() { nestedField.add(identifier); }
4953 ( <DOT> identifier = Identifier() { nestedField.add(identifier); } )*
4954 {
4955 fieldExclusions.add(nestedField);
4956 nestedField = new ArrayList<String>();
4957 }
4958 ( LOOKAHEAD(1) // Force <COMMA> to be recognized for a nested field in our EXCLUDE list.
4959 <COMMA> identifier = Identifier() { nestedField.add(identifier); }
4960 ( <DOT> identifier = Identifier() { nestedField.add(identifier); } )*
4961 {
4962 fieldExclusions.add(nestedField);
4963 nestedField = new ArrayList<String>();
4964 }
4965 )*
4966 )?
4967 )
Michael Blowd6cf6412016-06-30 02:44:35 -04004968 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07004969 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07004970 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004971 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004972 SourceLocation sourceLoc = getSourceLocation(startToken);
4973 if (selectRegular == null && selectElement == null){
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004974 Projection projection = new Projection(Projection.Kind.STAR, null, null);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004975 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004976 List<Projection> projections = new ArrayList<Projection>();
4977 projections.add(projection);
4978 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004979 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004980 }
ggalvizo82499442022-10-24 12:46:15 -07004981 SelectClause selectClause = new SelectClause(selectElement, selectRegular, fieldExclusions, distinct);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004982 selectClause.setSourceLocation(sourceLoc);
4983 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004984 }
4985}
4986
Yingyi Buab817482016-08-19 21:29:31 -07004987SelectRegular SelectRegular() throws ParseException:
4988{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004989 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004990 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004991 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004992}
4993{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004994 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004995 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004996 projections.add(projection);
4997 startSrcLoc = projection.getSourceLocation();
4998 }
4999 ( LOOKAHEAD(2) <COMMA> projection = Projection()
5000 {
5001 projections.add(projection);
5002 }
5003 )*
5004 {
5005 SelectRegular selectRegular = new SelectRegular(projections);
5006 selectRegular.setSourceLocation(startSrcLoc);
5007 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005008 }
5009}
5010
Yingyi Buab817482016-08-19 21:29:31 -07005011SelectElement SelectElement() throws ParseException:
5012{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005013 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005014 Expression expr = null;
5015 String name = null;
5016}
5017{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005018 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005019 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005020 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005021 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005022 }
5023}
5024
Yingyi Buab817482016-08-19 21:29:31 -07005025Projection Projection() throws ParseException :
5026{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005027 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005028 Expression expr = null;
5029 Identifier identifier = null;
5030 String name = null;
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07005031 Projection.Kind kind = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005032 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08005033 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005034}
5035{
5036 (
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07005037 <MUL> { kind = Projection.Kind.STAR; startSrcLoc = getSourceLocation(token); }
5038 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { kind = Projection.Kind.VAR_STAR; }
ggalvizo82499442022-10-24 12:46:15 -07005039 | expr = Expression()
5040 ( // EXCLUDE is a soft-keyword-- we want to avoid mistaking EXCLUDE as an identifier here.
5041 LOOKAHEAD({ getToken(1).kind == AS || getToken(1).kind == QUOTED_STRING
5042 || (getToken(1).kind == IDENTIFIER && !laIdentifier(1, EXCLUDE)) })
5043 (<AS>)? name = Identifier()
5044 )?
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08005045 {
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07005046 kind = Projection.Kind.NAMED_EXPR;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08005047 if (name == null) {
5048 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
5049 if (generatedColumnIdentifier != null) {
5050 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
5051 }
5052 }
5053 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005054 )
5055 {
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07005056 Projection projection = new Projection(kind, expr, name);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005057 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
5058 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005059 }
5060}
5061
5062FromClause FromClause() throws ParseException :
5063{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005064 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005065 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
5066 extendCurrentScope();
5067}
5068{
5069 {
5070 FromTerm fromTerm = null;
5071 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005072 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005073 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
5074 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005075 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005076 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005077 }
5078}
5079
5080FromTerm FromTerm() throws ParseException :
5081{
5082 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04005083 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005084 VariableExpr posVar = null;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005085 AbstractBinaryCorrelateClause correlateClause = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005086 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
5087}
5088{
Ali Alsulimanba91b1c2023-02-22 21:33:55 -08005089 leftExpr = Expression()
5090 {
5091 if (leftExpr.getKind() == Expression.Kind.VARIABLE_EXPRESSION) {
5092 Token hintToken = fetchHint(token, SqlppHint.SUBPATH_HINT);
5093 if (hintToken != null) {
5094 String subPath = hintToken.hintParams;
5095 ((VariableExpr) leftExpr).addHint(new ExternalSubpathAnnotation(subPath));
5096 }
5097 }
5098 } ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005099 (
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005100 (
5101 correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER)
5102 | ( <INNER> correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER) )
5103 | ( <LEFT> ( <OUTER> )? correlateClause = JoinOrUnnestClause(JoinType.LEFTOUTER, UnnestType.LEFTOUTER) )
5104 | ( <RIGHT> ( <OUTER> )? correlateClause = JoinClause(JoinType.RIGHTOUTER) )
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005105 | ( <CROSS> correlateClause = CrossJoinClause() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005106 )
5107 {
5108 correlateClauses.add(correlateClause);
5109 }
5110 )*
5111 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005112 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07005113 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07005114 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005115 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
5116 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
5117 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005118 }
5119}
5120
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005121AbstractBinaryCorrelateClause JoinOrUnnestClause(JoinType joinType, UnnestType unnestType) throws ParseException :
5122{
5123 AbstractBinaryCorrelateClause correlateClause = null;
5124}
5125{
5126 ( correlateClause = JoinClause(joinType) | correlateClause = UnnestClause(unnestType) )
5127 {
5128 return correlateClause;
5129 }
5130}
5131
Yingyi Bu391f09e2015-10-29 13:49:39 -07005132JoinClause JoinClause(JoinType joinType) throws ParseException :
5133{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005134 Token startToken = null;
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005135 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005136 Expression conditionExpr = null;
5137}
5138{
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005139 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput() <ON> conditionExpr = Expression()
5140 {
5141 JoinClause joinClause = new JoinClause(joinType, rightInput.first, rightInput.second, rightInput.third,
Dmitry Lychagin34b72ac2021-11-29 16:25:40 -08005142 conditionExpr, joinType == JoinType.INNER ? null : Literal.Type.MISSING);
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005143 return addSourceLocation(joinClause, startToken);
5144 }
5145}
5146
5147JoinClause CrossJoinClause() throws ParseException :
5148{
5149 Token startToken = null;
5150 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
5151 Expression conditionExpr = null;
5152}
5153{
5154 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput()
5155 {
5156 JoinClause joinClause = new JoinClause(JoinType.INNER, rightInput.first, rightInput.second, rightInput.third,
Dmitry Lychagin34b72ac2021-11-29 16:25:40 -08005157 new LiteralExpr(TrueLiteral.INSTANCE), null);
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005158 return addSourceLocation(joinClause, startToken);
5159 }
5160}
5161
5162Triple<Expression, VariableExpr, VariableExpr> JoinClauseRightInput() throws ParseException :
5163{
5164 Expression rightExpr = null;
5165 VariableExpr rightVar = null;
5166 VariableExpr posVar = null;
5167}
5168{
Ali Alsulimanba91b1c2023-02-22 21:33:55 -08005169 rightExpr = Expression()
5170 {
5171 if (rightExpr.getKind() == Expression.Kind.VARIABLE_EXPRESSION) {
5172 Token hintToken = fetchHint(token, SqlppHint.SUBPATH_HINT);
5173 if (hintToken != null) {
5174 String subPath = hintToken.hintParams;
5175 ((VariableExpr) rightExpr).addHint(new ExternalSubpathAnnotation(subPath));
5176 }
5177 }
5178 } ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005179 {
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005180 if (rightVar == null) {
5181 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07005182 }
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005183 return new Triple<Expression, VariableExpr, VariableExpr>(rightExpr, rightVar, posVar);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005184 }
5185}
5186
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005187UnnestClause UnnestClause(UnnestType unnestType) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07005188{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005189 Token startToken = null;
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08005190 Expression rightExpr = null;
5191 VariableExpr rightVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005192 VariableExpr posVar = null;
5193}
5194{
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08005195 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005196 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005197 if (rightVar == null) {
5198 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07005199 }
Dmitry Lychagin34b72ac2021-11-29 16:25:40 -08005200 UnnestClause unnestClause = new UnnestClause(unnestType, rightExpr, rightVar, posVar,
5201 unnestType == UnnestType.INNER ? null : Literal.Type.MISSING);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005202 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005203 }
5204}
5205
Yingyi Bu391f09e2015-10-29 13:49:39 -07005206List<LetClause> LetClause() throws ParseException:
5207{
5208 List<LetClause> letList = new ArrayList<LetClause>();
5209 LetClause letClause;
5210}
5211{
5212 (
5213 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
5214 |
5215 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
5216 )
5217 {
5218 return letList;
5219 }
5220}
5221
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005222WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07005223{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005224 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005225 Expression whereExpr;
5226}
5227{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005228 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005229 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005230 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005231 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005232 }
5233}
5234
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005235OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07005236{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005237 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005238 OrderbyClause oc = new OrderbyClause();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005239 Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> orderbyExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005240 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005241 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005242 List<OrderbyClause.NullOrderModifier> nullModifierList = new ArrayList<OrderbyClause.NullOrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07005243}
5244{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005245 <ORDER>
5246 {
5247 startToken = token;
5248 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
5249 if (hintToken != null) {
5250 switch (hintToken.hint) {
5251 case INMEMORY_HINT:
5252 String[] splits = hintToken.hintParams.split("\\s+");
5253 int numFrames = Integer.parseInt(splits[0]);
5254 int numTuples = Integer.parseInt(splits[1]);
5255 oc.setNumFrames(numFrames);
5256 oc.setNumTuples(numTuples);
5257 break;
5258 case RANGE_HINT:
5259 try {
5260 Expression rangeExpr = parseExpression(hintToken.hintParams);
5261 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
5262 oc.setRangeMap(rangeMap);
5263 } catch (CompilationException e) {
5264 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
Ali Alsuliman80225e22018-10-15 14:17:07 -07005265 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005266 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005267 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005268 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005269 }
5270 <BY> orderbyExpr = OrderByExpression()
5271 {
5272 orderbyList.add(orderbyExpr.first);
5273 modifierList.add(orderbyExpr.second);
5274 nullModifierList.add(orderbyExpr.third);
5275 }
5276 (
5277 LOOKAHEAD(2) <COMMA> orderbyExpr = OrderByExpression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005278 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005279 orderbyList.add(orderbyExpr.first);
5280 modifierList.add(orderbyExpr.second);
5281 nullModifierList.add(orderbyExpr.third);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005282 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005283 )*
5284 {
5285 oc.setOrderbyList(orderbyList);
5286 oc.setModifierList(modifierList);
5287 oc.setNullModifierList(nullModifierList);
5288 return addSourceLocation(oc, startToken);
5289 }
5290}
Yingyi Bu391f09e2015-10-29 13:49:39 -07005291
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005292Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> OrderByExpression()
5293 throws ParseException:
5294{
5295 Expression orderbyExpr = null;
5296 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
5297 OrderbyClause.NullOrderModifier nullModif = null;
5298}
5299{
5300 orderbyExpr = Expression()
5301 (
5302 <ASC> { modif = OrderbyClause.OrderModifier.ASC; }
5303 |
5304 <DESC> { modif = OrderbyClause.OrderModifier.DESC; }
5305 )?
5306 (
5307 LOOKAHEAD({ laIdentifier(NULLS) }) <IDENTIFIER> { expectToken(NULLS); } <IDENTIFIER>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005308 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005309 if (isToken(FIRST)) {
5310 nullModif = OrderbyClause.NullOrderModifier.FIRST;
5311 } else if (isToken(LAST)) {
5312 nullModif = OrderbyClause.NullOrderModifier.LAST;
5313 } else {
5314 throw createUnexpectedTokenError();
5315 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005316 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005317 )?
5318 {
5319 return new Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier>(orderbyExpr, modif,
5320 nullModif);
5321 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005322}
5323
5324GroupbyClause GroupbyClause()throws ParseException :
5325{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005326 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005327 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005328 List<List<GbyVariableExpressionPair>> gbyList = null;
5329 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08005330 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07005331 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08005332 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005333}
5334{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005335 {
5336 Scope newScope = extendCurrentScopeNoPush(true);
5337 // extendCurrentScope(true);
5338 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005339 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005340 {
5341 startToken = token;
5342 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
5343 if (hintToken != null) {
5344 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005345 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005346 }
5347 <BY> groupingElementList = GroupingElementList()
5348 (
5349 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08005350 {
5351 groupVar = groupVarWithFieldList.first;
5352 groupFieldList = groupVarWithFieldList.second;
5353 }
Yingyi Buacc12a92016-03-26 17:25:05 -07005354 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005355 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005356 if (groupingSetsParser == null) {
5357 groupingSetsParser = new SqlppGroupingSetsParser();
5358 }
5359 SourceLocation sourceLoc = getSourceLocation(startToken);
5360 try {
5361 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
5362 } catch (CompilationException e) {
5363 throw new SqlppParseException(sourceLoc, e.getMessage());
5364 }
5365 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005366 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07005367 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07005368 gbc.setGroupVar(groupVar);
5369 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005370 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005371 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005372 }
5373}
5374
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005375List<GroupingElement> GroupingElementList() throws ParseException:
5376{
5377 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
5378 GroupingElement groupingElement = null;
5379}
5380{
5381 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
5382 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
5383 {
5384 return groupingElementList;
5385 }
5386}
5387
5388GroupingElement GroupingElement() throws ParseException:
5389{
5390 GroupingElement groupingElement = null;
5391 List<GroupingSet> groupingSets = null;
5392 List<GroupingElement> groupingElements = null;
5393}
5394{
5395 (
5396 LOOKAHEAD(2)
5397 <LEFTPAREN> <RIGHTPAREN>
5398 {
5399 groupingElement = GroupingSet.EMPTY;
5400 }
5401 |
5402 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
5403 <IDENTIFIER> { expectToken(ROLLUP); }
5404 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
5405 {
5406 groupingElement = new RollupCube(groupingSets, false);
5407 }
5408 |
5409 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
5410 <IDENTIFIER> { expectToken(CUBE); }
5411 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
5412 {
5413 groupingElement = new RollupCube(groupingSets, true);
5414 }
5415 |
5416 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
5417 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
5418 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
5419 {
5420 groupingElement = new GroupingSets(groupingElements);
5421 }
5422 |
5423 groupingElement = OrdinaryGroupingSet()
5424 )
5425 {
5426 return groupingElement;
5427 }
5428}
5429
5430GroupingSet OrdinaryGroupingSet() throws ParseException:
5431{
5432 GbyVariableExpressionPair gbyExprPair = null;
5433 List<GbyVariableExpressionPair> items = null;
5434}
5435{
5436 (
5437 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
5438 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
5439 )
5440 {
5441 return new GroupingSet(items);
5442 }
5443}
5444
5445List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
5446{
5447 GroupingSet groupingSet = null;
5448 List<GroupingSet> items = new ArrayList<GroupingSet>();
5449}
5450{
5451 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
5452 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
5453 {
5454 return items;
5455 }
5456}
5457
5458List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
5459{
5460 GbyVariableExpressionPair gbyExprPair = null;
5461 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
5462}
5463{
5464 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
5465 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
5466 {
5467 return items;
5468 }
5469}
5470
5471GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
5472{
5473 Expression expr = null;
5474 VariableExpr var = null;
5475}
5476{
5477 expr = Expression() ( (<AS>)? var = Variable() )?
5478 {
5479 return new GbyVariableExpressionPair(var, expr);
5480 }
5481}
5482
Yingyi Bu391f09e2015-10-29 13:49:39 -07005483HavingClause HavingClause() throws ParseException:
5484{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005485 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005486 Expression filterExpr = null;
5487}
5488{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005489 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005490 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005491 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005492 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005493 }
5494}
5495
5496LimitClause LimitClause() throws ParseException:
5497{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005498 Token startToken = null;
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005499 Expression limitExpr = null, offsetExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005500}
5501{
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005502 (
5503 (
5504 <LIMIT> { startToken = token; pushForbiddenScope(getCurrentScope()); } limitExpr = Expression()
5505 ( <OFFSET> offsetExpr = Expression() )?
5506 { popForbiddenScope(); }
5507 )
5508 |
5509 (
5510 <OFFSET> { startToken = token; pushForbiddenScope(getCurrentScope()); } offsetExpr = Expression()
5511 { popForbiddenScope(); }
5512 )
5513 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005514 {
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005515 LimitClause lc = new LimitClause(limitExpr, offsetExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005516 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005517 }
5518}
5519
5520QuantifiedExpression QuantifiedExpression()throws ParseException:
5521{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005522 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005523 QuantifiedExpression qc = new QuantifiedExpression();
5524 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
5525 Expression satisfiesExpr;
5526 VariableExpr var;
5527 Expression inExpr;
5528 QuantifiedPair pair;
5529}
5530{
5531 {
5532 createNewScope();
5533 }
5534
ggalvizo21e52822021-07-27 10:26:31 -10005535 ( LOOKAHEAD(2)
5536 (<ANY>|<SOME>)<AND><EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME_AND_EVERY); }
5537 | (<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); }
5538 | <EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005539 var = Variable() <IN> inExpr = Expression()
5540 {
5541 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005542 quantifiedList.add(pair);
5543 }
5544 (
5545 <COMMA> var = Variable() <IN> inExpr = Expression()
5546 {
5547 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005548 quantifiedList.add(pair);
5549 }
5550 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07005551 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005552 {
5553 qc.setSatisfiesExpr(satisfiesExpr);
5554 qc.setQuantifiedList(quantifiedList);
5555 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005556 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005557 }
5558}
5559
5560LetClause LetElement() throws ParseException:
5561{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005562 VariableExpr varExp;
5563 Expression beExp;
5564 extendCurrentScope();
5565}
5566{
5567 varExp = Variable() <EQ> beExp = Expression()
5568 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005569 LetClause lc = new LetClause(varExp, beExp);
5570 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005571 return lc;
5572 }
5573}
5574
5575LetClause WithElement() throws ParseException:
5576{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005577 VariableExpr varExp;
5578 Expression beExp;
5579 extendCurrentScope();
5580}
5581{
5582 varExp = Variable() <AS> beExp = Expression()
5583 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005584 LetClause lc = new LetClause(varExp, beExp);
5585 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005586 return lc;
5587 }
5588}
5589
5590TOKEN_MGR_DECLS:
5591{
5592 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07005593 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005594 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005595
5596 public void pushState() {
5597 lexerStateStack.push( curLexState );
5598 }
5599
5600 public void popState(String token) {
5601 if (lexerStateStack.size() > 0) {
5602 SwitchTo( lexerStateStack.pop() );
5603 } else {
5604 int errorLine = input_stream.getEndLine();
5605 int errorColumn = input_stream.getEndColumn();
5606 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
5607 + "\" but state stack is empty.";
5608 throw new TokenMgrError(msg, -1);
5609 }
5610 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005611
5612 void CommonTokenAction(Token token) {
5613 Token hintToken = token.specialToken;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07005614 while (hintToken != null) { // make this a while loop
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005615 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
5616 String text = hintToken.image.substring(1).trim();
5617 boolean hintFound = hintToken.parseHint(text);
5618 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07005619 hintToken = hintToken.specialToken;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005620 }
5621 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005622}
5623
5624<DEFAULT,IN_DBL_BRACE>
5625TOKEN [IGNORE_CASE]:
5626{
Ian Maxon38fe9402020-01-29 19:27:40 -08005627 <ADAPTER: "adapter">
5628 | <ALL : "all">
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07005629 | <ANALYZE: "analyze">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005630 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07005631 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005632 | <APPLY : "apply">
5633 | <AS : "as">
5634 | <ASC : "asc">
5635 | <AT : "at">
5636 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005637 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005638 | <BTREE : "btree">
5639 | <BY : "by">
5640 | <CASE : "case">
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07005641 | <CAST : "cast">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005642 | <CLOSED : "closed">
5643 | <CREATE : "create">
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005644 | <CROSS : "cross">
Dmitry Lychagin0ebc4202021-01-25 13:12:41 -08005645 | <COMPACTION : "compaction"> // no longer used
Yingyi Bu391f09e2015-10-29 13:49:39 -07005646 | <COMPACT : "compact">
5647 | <CONNECT : "connect">
5648 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07005649 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07005650 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07005651 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005652 | <DECLARE : "declare">
5653 | <DEFINITION : "definition">
5654 | <DELETE : "delete">
5655 | <DESC : "desc">
5656 | <DISCONNECT : "disconnect">
5657 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005658 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005659 | <DROP : "drop">
5660 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07005661 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005662 | <ELSE : "else">
5663 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07005664 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005665 | <EVERY : "every">
5666 | <EXCEPT : "except">
5667 | <EXISTS : "exists">
5668 | <EXTERNAL : "external">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005669 | <FALSE : "false">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005670 | <FEED : "feed">
5671 | <FILTER : "filter">
5672 | <FLATTEN : "flatten">
5673 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005674 | <FROM : "from">
5675 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08005676 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005677 | <FUNCTION : "function">
5678 | <GROUP : "group">
5679 | <HAVING : "having">
5680 | <HINTS : "hints">
5681 | <IF : "if">
5682 | <INTO : "into">
5683 | <IN : "in">
5684 | <INDEX : "index">
5685 | <INGESTION : "ingestion">
5686 | <INNER : "inner">
5687 | <INSERT : "insert">
5688 | <INTERNAL : "internal">
5689 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07005690 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005691 | <JOIN : "join">
5692 | <KEYWORD : "keyword">
5693 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07005694 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005695 | <LEFT : "left">
5696 | <LETTING : "letting">
5697 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005698 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005699 | <LIMIT : "limit">
5700 | <LOAD : "load">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005701 | <MISSING : "missing">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005702 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005703 | <NODEGROUP : "nodegroup">
5704 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07005705 | <NOT : "not">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005706 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005707 | <OFFSET : "offset">
5708 | <ON : "on">
5709 | <OPEN : "open">
5710 | <OR : "or">
5711 | <ORDER : "order">
5712 | <OUTER : "outer">
5713 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07005714 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005715 | <PATH : "path">
5716 | <POLICY : "policy">
5717 | <PRESORTED : "pre-sorted">
5718 | <PRIMARY : "primary">
5719 | <RAW : "raw">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005720 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005721 | <RETURNING : "returning">
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005722 | <RIGHT : "right">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005723 | <RTREE : "rtree">
5724 | <RUN : "run">
5725 | <SATISFIES : "satisfies">
5726 | <SECONDARY : "secondary">
5727 | <SELECT : "select">
5728 | <SET : "set">
5729 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08005730 | <START : "start">
5731 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08005732 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03005733 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07005734 | <THEN : "then">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005735 | <TO : "to">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005736 | <TRUE : "true">
5737 | <TYPE : "type">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005738 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08005739 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005740 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07005741 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005742 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07005743 | <USE : "use">
5744 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005745 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08005746 | <VALUED : "valued">
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07005747 | <VIEW : "view">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005748 | <WHEN : "when">
5749 | <WHERE : "where">
5750 | <WITH : "with">
5751 | <WRITE : "write">
Peeyush Gupta066fd562023-04-30 13:03:00 -07005752 | <COPY : "copy">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005753}
5754
5755<DEFAULT,IN_DBL_BRACE>
5756TOKEN :
5757{
5758 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07005759 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005760 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005761 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005762 | <MUL : "*">
5763 | <PLUS : "+">
5764
5765 | <LEFTPAREN : "(">
5766 | <RIGHTPAREN : ")">
5767 | <LEFTBRACKET : "[">
5768 | <RIGHTBRACKET : "]">
5769
5770 | <ATT : "@">
5771 | <COLON : ":">
5772 | <COMMA : ",">
5773 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005774 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005775 | <QUES : "?">
5776 | <SEMICOLON : ";">
5777 | <SHARP : "#">
5778
5779 | <LT : "<">
5780 | <GT : ">">
5781 | <LE : "<=">
5782 | <GE : ">=">
5783 | <EQ : "=">
5784 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07005785 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005786 | <SIMILAR : "~=">
5787}
5788
5789<DEFAULT,IN_DBL_BRACE>
5790TOKEN :
5791{
5792 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
5793}
5794
5795<DEFAULT>
5796TOKEN :
5797{
5798 <RIGHTBRACE : "}"> { popState("}"); }
5799}
5800
5801<DEFAULT,IN_DBL_BRACE>
5802TOKEN :
5803{
5804 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
5805}
5806
5807<IN_DBL_BRACE>
5808TOKEN :
5809{
5810 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
5811}
5812
5813<DEFAULT,IN_DBL_BRACE>
5814TOKEN :
5815{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005816 <#DIGIT : ["0" - "9"]>
5817}
5818
5819<DEFAULT,IN_DBL_BRACE>
5820TOKEN:
5821{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005822 <INTEGER_LITERAL : <DIGITS> >
5823 | <DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07005824 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
5825 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005826 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005827 | <FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
Yingyi Bue4d919e2016-10-30 10:47:03 -07005828 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
5829 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005830 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005831 | <#DIGITS : (<DIGIT>)+ >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005832}
5833
5834<DEFAULT,IN_DBL_BRACE>
5835TOKEN :
5836{
5837 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005838 | <#IDENTIFIER_START_SPECIALCHAR : ["_"]>
5839 | <#IDENTIFIER_REST_SPECIALCHAR : ["$"]>
5840 | <#IDENTIFIER_START : <LETTER> | <IDENTIFIER_START_SPECIALCHAR> >
5841 | <#IDENTIFIER_REST : <LETTER> | <DIGIT> | <IDENTIFIER_START_SPECIALCHAR> | <IDENTIFIER_REST_SPECIALCHAR> >
5842 | <IDENTIFIER : <IDENTIFIER_START> (<IDENTIFIER_REST>)* >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005843}
5844
5845<DEFAULT,IN_DBL_BRACE>
5846TOKEN :
5847{
5848 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07005849 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005850 <EscapeQuot>
5851 | <EscapeBslash>
5852 | <EscapeSlash>
5853 | <EscapeBspace>
5854 | <EscapeFormf>
5855 | <EscapeNl>
5856 | <EscapeCr>
5857 | <EscapeTab>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305858 | <EscapeBTickWithBslash>
5859 | <EscapeBtickWithBtick>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005860 | ~["`","\\"])* "`">
Dmitry Lychagin984bc522021-10-13 16:22:25 -07005861 | <STRING_LITERAL : ( ("E")? "\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005862 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005863 | <EscapeBslash>
5864 | <EscapeSlash>
5865 | <EscapeBspace>
5866 | <EscapeFormf>
5867 | <EscapeNl>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305868 | <EscapeBTickWithBslash>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005869 | <EscapeCr>
5870 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005871 | ~["\"","\\"])* "\"")
Dmitry Lychagin984bc522021-10-13 16:22:25 -07005872 | ( ("E")? "\'" (
Yingyi Bu6d57e492016-06-06 21:24:42 -07005873 <EscapeApos>
5874 | <EscapeBslash>
5875 | <EscapeSlash>
5876 | <EscapeBspace>
5877 | <EscapeFormf>
5878 | <EscapeNl>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305879 | <EscapeBTickWithBslash>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005880 | <EscapeCr>
5881 | <EscapeTab>
5882 | ~["\'","\\"])* "\'")>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305883 | < #EscapeBTickWithBslash: "\\`" >
5884 | < #EscapeBtickWithBtick: "``" >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005885 | < #EscapeQuot: "\\\"" >
5886 | < #EscapeApos: "\\\'" >
5887 | < #EscapeBslash: "\\\\" >
5888 | < #EscapeSlash: "\\/" >
5889 | < #EscapeBspace: "\\b" >
5890 | < #EscapeFormf: "\\f" >
5891 | < #EscapeNl: "\\n" >
5892 | < #EscapeCr: "\\r" >
5893 | < #EscapeTab: "\\t" >
5894}
5895
5896<DEFAULT,IN_DBL_BRACE>
5897TOKEN :
5898{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005899 <DOLLAR_INTEGER_LITERAL : "$" <INTEGER_LITERAL> >
5900 | <DOLLAR_IDENTIFIER : "$" <IDENTIFIER> >
5901 | <DOLLAR_QUOTED_STRING: "$" <QUOTED_STRING> >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005902}
5903
5904<DEFAULT,IN_DBL_BRACE>
5905SKIP:
5906{
5907 " "
5908 | "\t"
5909 | "\r"
5910 | "\n"
5911}
5912
5913<DEFAULT,IN_DBL_BRACE>
5914SKIP:
5915{
5916 <"//" (~["\n"])* "\n">
5917}
5918
5919<DEFAULT,IN_DBL_BRACE>
5920SKIP:
5921{
5922 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5923}
5924
5925<DEFAULT,IN_DBL_BRACE>
5926SKIP:
5927{
Yingyi Bu93846a72016-09-13 16:30:39 -07005928 <"--" (~["\n"])* "\n">
5929}
5930
5931
5932<DEFAULT,IN_DBL_BRACE>
5933SKIP:
5934{
5935 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5936}
5937
5938<DEFAULT,IN_DBL_BRACE>
5939SKIP:
5940{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005941 <"/*"> { pushState(); } : INSIDE_COMMENT
5942}
5943
5944<INSIDE_COMMENT>
5945SPECIAL_TOKEN:
5946{
5947 <"+"(" ")*(~["*"])*>
5948}
5949
5950<INSIDE_COMMENT>
5951SKIP:
5952{
5953 <"/*"> { pushState(); }
5954}
5955
5956<INSIDE_COMMENT>
5957SKIP:
5958{
5959 <"*/"> { popState("*/"); }
5960 | <~[]>
5961}