blob: 464efb46d9bb5590b74a54fd4039e9dc8ad4aecf [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;
162import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
163import org.apache.asterix.lang.common.statement.NodegroupDecl;
164import org.apache.asterix.lang.common.statement.Query;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700165import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800166import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700167import org.apache.asterix.lang.common.statement.TypeDecl;
168import org.apache.asterix.lang.common.statement.TypeDropStatement;
169import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800170import org.apache.asterix.lang.common.statement.UpsertStatement;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700171import org.apache.asterix.lang.common.statement.ViewDecl;
172import org.apache.asterix.lang.common.statement.ViewDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700173import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800174import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700175import org.apache.asterix.lang.common.struct.QuantifiedPair;
176import org.apache.asterix.lang.common.struct.VarIdentifier;
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -0800177import org.apache.asterix.lang.common.util.DatasetDeclParametersUtil;
Tin Vu4f1090d2021-09-21 02:06:32 -0700178import org.apache.asterix.lang.common.util.ExpressionUtils;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700179import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700180import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
181import org.apache.asterix.lang.sqlpp.clause.FromClause;
182import org.apache.asterix.lang.sqlpp.clause.FromTerm;
183import org.apache.asterix.lang.sqlpp.clause.HavingClause;
184import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700185import org.apache.asterix.lang.sqlpp.clause.Projection;
186import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
187import org.apache.asterix.lang.sqlpp.clause.SelectClause;
188import org.apache.asterix.lang.sqlpp.clause.SelectElement;
189import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
190import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
191import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800192import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700193import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700194import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700195import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700196import org.apache.asterix.lang.sqlpp.optype.JoinType;
197import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -0700198import org.apache.asterix.lang.sqlpp.optype.UnnestType;
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700199import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser;
200import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingElement;
201import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSet;
202import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSets;
203import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.RollupCube;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700204import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700205import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
206import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700207import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700208import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700209import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Peeyush Gupta6ba5a262023-04-13 11:08:53 -0700210import org.apache.asterix.metadata.bootstrap.MetadataBuiltinEntities;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -0700211import org.apache.asterix.metadata.utils.MetadataConstants;
Tin Vu4f1090d2021-09-21 02:06:32 -0700212import org.apache.asterix.om.exceptions.TypeMismatchException;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800213import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700214import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700215import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700216import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700217import org.apache.hyracks.algebricks.common.utils.Pair;
218import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800219import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700220import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700221import org.apache.hyracks.algebricks.core.algebra.expressions.HashJoinExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700222import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700223import org.apache.hyracks.algebricks.core.algebra.expressions.JoinProductivityAnnotation;
224import org.apache.hyracks.algebricks.core.algebra.expressions.PredicateCardinalityAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700225import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800226import org.apache.hyracks.api.exceptions.IWarningCollector;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700227import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700228import org.apache.hyracks.api.exceptions.Warning;
Dmitry Lychagin70bf47a2020-10-22 13:25:57 -0700229import org.apache.hyracks.dataflow.common.data.partition.range.RangeMap;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800230import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700231import org.apache.hyracks.util.StringUtil;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700232import java.util.regex.Matcher;
233import java.util.regex.Pattern;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700234
Yingyi Bucaea8f02015-11-16 15:12:15 -0800235class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700236
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800237 // tokens parsed as identifiers
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700238 private static final String CUBE = "CUBE";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800239 private static final String CURRENT = "CURRENT";
Dmitry Lychagin33c77f92021-07-21 15:59:04 -0700240 private static final String DEFAULT = "DEFAULT";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800241 private static final String EXCLUDE = "EXCLUDE";
Ali Alsuliman931e7382021-08-08 21:55:59 +0300242 private static final String INCLUDE = "INCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700243 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800244 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychagin914983f2021-09-30 15:11:11 -0700245 private static final String FOREIGN = "FOREIGN";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700246 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800247 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700248 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700249 private static final String LAST = "LAST";
Glenn67fd1f32021-02-25 16:04:49 -0800250 private static final String META = "META";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800251 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700252 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800253 private static final String OTHERS = "OTHERS";
254 private static final String PARTITION = "PARTITION";
255 private static final String PRECEDING = "PRECEDING";
256 private static final String RANGE = "RANGE";
Dmitry Lychagin914983f2021-09-30 15:11:11 -0700257 private static final String REFERENCES = "REFERENCES";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700258 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700259 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800260 private static final String ROW = "ROW";
261 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700262 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800263 private static final String TIES = "TIES";
264 private static final String UNBOUNDED = "UNBOUNDED";
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700265 private static final String REPLACE = "REPLACE";
Ian Maxon38fe9402020-01-29 19:27:40 -0800266 private static final String RETURNS = "RETURNS";
Rui Guoe6986dd2020-11-25 19:50:06 -0800267 private static final String CONFIG = "CONFIG";
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -0700268 private static final String STATISTICS = "STATISTICS";
Rui Guoe6986dd2020-11-25 19:50:06 -0800269
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800270
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700271 private static final String INT_TYPE_NAME = "int";
Dmitry Lychagin541652082020-11-09 14:04:53 -0800272 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 -0700273
Till Westmann7199a562016-09-17 16:07:32 -0700274 // error configuration
275 protected static final boolean REPORT_EXPECTED_TOKENS = false;
276
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700277 private int externalVarCounter;
278
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800279 private DataverseName defaultDataverse;
280
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700281 private SqlppGroupingSetsParser groupingSetsParser;
282
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700283 private final WarningCollector warningCollector = new WarningCollector();
284
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700285 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
286
Yingyi Bu391f09e2015-10-29 13:49:39 -0700287 private static class IndexParams {
288 public IndexType type;
289 public int gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800290 public String fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700291
Rui Guoe6986dd2020-11-25 19:50:06 -0800292 public IndexParams(IndexType type, int gramLength, String fullTextConfig) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700293 this.type = type;
294 this.gramLength = gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800295 this.fullTextConfig = fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700296 }
Ali Alsuliman69ce7d82021-11-01 15:00:16 -0700297 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700298
299 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800300 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700301 public String library;
302 public String function;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -0600303 public Token hintToken;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700304 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700305 }
306
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700307 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
308 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
309 switch (hintToken.hint) {
310 case VAL_FILE_HINT:
311 File[] valFiles = new File[splits.length];
312 for (int k=0; k<splits.length; k++) {
313 valFiles[k] = new File(splits[k]);
314 }
315 return new FieldValFileDataGen(valFiles);
316 case VAL_FILE_SAME_INDEX_HINT:
317 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
318 case LIST_VAL_FILE_HINT:
319 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
320 case LIST_HINT:
321 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
322 case INTERVAL_HINT:
323 FieldIntervalDataGen.ValueType vt;
324 switch (splits[0]) {
325 case "int":
326 vt = FieldIntervalDataGen.ValueType.INT;
327 break;
328 case "long":
329 vt = FieldIntervalDataGen.ValueType.LONG;
330 break;
331 case "float":
332 vt = FieldIntervalDataGen.ValueType.FLOAT;
333 break;
334 case "double":
335 vt = FieldIntervalDataGen.ValueType.DOUBLE;
336 break;
337 default:
338 throw new SqlppParseException(getSourceLocation(hintToken),
339 "Unknown type for interval data gen: " + splits[0]);
340 }
341 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
342 case INSERT_RAND_INT_HINT:
343 return new InsertRandIntDataGen(splits[0], splits[1]);
344 case DATE_BETWEEN_YEARS_HINT:
345 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
346 case DATETIME_BETWEEN_YEARS_HINT:
347 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
348 case DATETIME_ADD_RAND_HOURS_HINT:
349 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
350 case AUTO_HINT:
351 return new AutoDataGen(splits[0]);
352 default:
353 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700354 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700355 }
356
Till Westmann7199a562016-09-17 16:07:32 -0700357 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700358 this(new StringReader(s));
359 super.setInput(s);
360 }
361
Ali Alsuliman69ce7d82021-11-01 15:00:16 -0700362 public static void main(String[] args) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700363 File file = new File(args[0]);
364 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
365 SQLPPParser parser = new SQLPPParser(fis);
366 List<Statement> st = parser.parse();
367 //st.accept(new SQLPPPrintVisitor(), 0);
368 }
369
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800370 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800371 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700372 return parseImpl(new ParseFunction<List<Statement>>() {
373 @Override
374 public List<Statement> parse() throws ParseException {
375 return SQLPPParser.this.Statement();
376 }
377 });
378 }
379
Dmitry Lychagin62f0beb2020-09-08 11:44:19 -0700380 @Override
381 public Expression parseExpression() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700382 return parseImpl(new ParseFunction<Expression>() {
383 @Override
384 public Expression parse() throws ParseException {
385 return SQLPPParser.this.Expression();
386 }
387 });
388 }
389
390 private static Expression parseExpression(String text) throws CompilationException {
391 return new SQLPPParser(text).parseExpression();
392 }
393
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800394 @Override
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700395 public List<String> parseMultipartIdentifier() throws CompilationException {
396 return parseImpl(new ParseFunction<List<String>>() {
397 @Override
398 public List<String> parse() throws ParseException {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700399 return SQLPPParser.this.MultipartIdentifier().first;
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700400 }
401 });
402 }
403
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800404 private List<String> parseParenthesizedIdentifierList() throws CompilationException {
405 return parseImpl(new ParseFunction<List<String>>() {
406 @Override
407 public List<String> parse() throws ParseException {
408 return SQLPPParser.this.ParenthesizedIdentifierList();
409 }
410 });
411 }
412
413 private static List<String> parseParenthesizedIdentifierList(String text) throws CompilationException {
414 return new SQLPPParser(text).parseParenthesizedIdentifierList();
415 }
416
Vijay Sarathy691e93b2022-11-20 14:31:47 -0800417 private Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> parseHashJoinParams() throws CompilationException {
418 return parseImpl(new ParseFunction<Pair<HashJoinExpressionAnnotation.BuildOrProbe, String>>() {
419 @Override
420 public Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> parse() throws ParseException {
421 return SQLPPParser.this.buildOrProbeParenthesizedIdentifier();
422 }
423 });
424 }
425
426 private static Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> parseHashJoinParams(String text) throws CompilationException {
427 return new SQLPPParser(text).parseHashJoinParams();
428 }
429
430 private String parseBroadcastJoinParams() throws CompilationException {
431 return parseImpl(new ParseFunction<String>() {
432 @Override
433 public String parse() throws ParseException {
434 return SQLPPParser.this.parenthesizedIdentifier();
435 }
436 });
437 }
438
439 private static String parseBroadcastJoinParams(String text) throws CompilationException {
440 return new SQLPPParser(text).parseBroadcastJoinParams();
441 }
442
Tin Vu4f1090d2021-09-21 02:06:32 -0700443 private List<Literal> parseParenthesizedLiteralList() throws CompilationException {
444 return parseImpl(new ParseFunction<List<Literal>>() {
445 @Override
446 public List<Literal> parse() throws ParseException {
447 return SQLPPParser.this.ParenthesizedLiteralList();
448 }
449 });
450 }
451
452 private static List<Literal> parseParenthesizedLiteralList(String text) throws CompilationException {
453 return new SQLPPParser(text).parseParenthesizedLiteralList();
454 }
455
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700456 @Override
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700457 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames, boolean isStored)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800458 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800459 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800460 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800461 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800462 DataverseName dataverse = defaultDataverse;
463 defaultDataverse = signature.getDataverseName();
464 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800465 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
466 for (String paramName : paramNames) {
467 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
468 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800469 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
470 removeCurrentScope();
471 defaultDataverse = dataverse;
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700472 return new FunctionDecl(signature, paramVars, functionBodyExpr, isStored);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800473 }
474 });
475 }
476
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700477 @Override
478 public ViewDecl parseViewBody(DatasetFullyQualifiedName viewName) throws CompilationException {
479 return parseImpl(new ParseFunction<ViewDecl>() {
480 @Override
481 public ViewDecl parse() throws ParseException {
482 DataverseName dataverse = defaultDataverse;
483 defaultDataverse = viewName.getDataverseName();
484 createNewScope();
485 Expression viewBodyExpr = SQLPPParser.this.ViewBody();
486 removeCurrentScope();
487 defaultDataverse = dataverse;
488 return new ViewDecl(viewName, viewBodyExpr);
489 }
490 });
491 }
492
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700493 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700494 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700495 hintCollector.clear();
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800496 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700497 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700498 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700499 } catch (SqlppParseException e) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700500 if (e.getCause() instanceof AlgebricksException) {
501 AlgebricksException cause = (AlgebricksException) e.getCause();
502 if (cause.getError().isPresent() && cause.getError().get() instanceof ErrorCode) {
503 throw new CompilationException((ErrorCode) cause.getError().get(), e.getSourceLocation(),
504 cause.getParams());
505 }
506 }
507 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(),
508 LogRedactionUtil.userData(getMessage(e)));
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700509 } catch (ParseException e) {
510 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700511 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700512 // 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 -0700513 // 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 -0700514 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700515 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700516 } finally {
517 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700518 }
519 }
Till Westmann7199a562016-09-17 16:07:32 -0700520
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700521 @FunctionalInterface
522 private interface ParseFunction<T> {
523 T parse() throws ParseException;
524 }
525
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700526 @Override
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800527 public void getWarnings(IWarningCollector outWarningCollector) {
528 warningCollector.getWarnings(outWarningCollector);
529 }
530
531 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700532 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
533 warningCollector.getWarnings(outWarnings, maxWarnings);
534 }
535
536 @Override
537 public long getTotalWarningsCount() {
538 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700539 }
540
Till Westmann7199a562016-09-17 16:07:32 -0700541 protected String getMessage(ParseException pe) {
542 Token currentToken = pe.currentToken;
543 if (currentToken == null) {
544 return pe.getMessage();
545 }
546 int[][] expectedTokenSequences = pe.expectedTokenSequences;
547 String[] tokenImage = pe.tokenImage;
548 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
549 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
550 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
551 Token tok = currentToken.next;
552 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700553 StringBuilder message = new StringBuilder(128);
554 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700555 for (int i = 0; i < maxSize; i++) {
556 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700557 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700558 }
559 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700560 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700561 break;
562 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700563 final String fixedTokenImage = tokenImage[tok.kind];
564 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700565 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700566 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700567 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700568 tok = tok.next;
569 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700570 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700571 if (REPORT_EXPECTED_TOKENS) {
572 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700573 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700574 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700575 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700576 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700577 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700578 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700579 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700580 }
581
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700582 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700583 return
584 token == null ? null :
585 token.sourceLocation != null ? token.sourceLocation :
586 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700587 }
588
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700589 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
590 expr.setSourceLocation(getSourceLocation(token));
591 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700592 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800593
594 private boolean isToken(String image) {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700595 return isToken(token, image);
596 }
597
598 private static boolean isToken(Token token, String image) {
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800599 return token.image.equalsIgnoreCase(image);
600 }
601
602 private void expectToken(String image) throws SqlppParseException {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700603 expectToken(token, image);
604 }
605
606 private static void expectToken(Token token, String image) throws SqlppParseException {
607 if (!isToken(token, image)) {
608 throw createUnexpectedTokenError(token);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800609 }
610 }
611
612 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700613 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700614 }
615
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700616 private static SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700617 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800618 }
619
620 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700621 return createUnexpectedTokenError(token, expected);
622 }
623
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700624 private static SqlppParseException createUnexpectedTokenError(Token t, String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700625 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800626 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700627 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800628 }
629
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700630 private boolean laToken(int idx, int kind) {
631 Token t = getToken(idx);
632 return t.kind == kind;
633 }
634
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800635 private boolean laToken(int idx, int kind, String image) {
636 Token t = getToken(idx);
637 return t.kind == kind && t.image.equalsIgnoreCase(image);
638 }
639
640 private boolean laIdentifier(int idx, String image) {
641 return laToken(idx, IDENTIFIER, image);
642 }
643
644 private boolean laIdentifier(String image) {
645 return laIdentifier(1, image);
646 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700647
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700648 private Token fetchHint(Token token, SqlppHint... expectedHints) {
649 Token hintToken = token.specialToken;
650 if (hintToken == null) {
651 return null;
652 }
653 SourceLocation sourceLoc = getSourceLocation(hintToken);
654 hintCollector.remove(sourceLoc);
655 if (hintToken.hint == null) {
656 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
657 return null;
658 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
659 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
660 return null;
661 } else {
662 return hintToken;
663 }
664 }
665
666 private void reportUnclaimedHints() {
667 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
668 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
669 }
670 }
671
672 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
673 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
674 }
675
676 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700677 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500678 warningCollector.warn(Warning.of(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
Ali Alsulimanaa118862019-09-10 20:55:45 -0700679 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700680 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700681
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800682 private IExpressionAnnotation parseExpressionAnnotation(Token hintToken) {
683 // placeholder for the annotation that should be returned if this hint's parameters cannot be parsed
684 IExpressionAnnotation onParseErrorReturn = null;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700685 double selectivity, cardinality, productivity;
686 Pattern number = Pattern.compile("\\d+\\.\\d+");
687 Pattern stringNumber = Pattern.compile("\\w+\\s+\\d+\\.\\d+");
688 Pattern lessThanOnePat = Pattern.compile("0\\.\\d+");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800689 try {
690 switch (hintToken.hint) {
Vijay Sarathy7ba271a2023-03-09 08:07:43 -0800691 case SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT:
692 if (hintToken.hintParams == null) {
693 throw new SqlppParseException(getSourceLocation(hintToken), "Expected selectivity value");
694 }
695 else {
696 selectivity = 1.0; // uninitialized
697 Matcher mat = lessThanOnePat.matcher(hintToken.hintParams);
698 if (mat.find()) {
699 selectivity = Double.parseDouble (mat.group());
700 }
701 else {
Vijay Sarathyc70bae02023-03-13 13:24:05 -0700702 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 -0800703 }
704 return new PredicateCardinalityAnnotation(selectivity);
705 }
706 case JOIN_PREDICATE_PRODUCTIVITY_HINT:
707 if (hintToken.hintParams == null) {
708 throw new SqlppParseException(getSourceLocation(hintToken), "Expected productivity collection name and value");
709 }
710 else {
711 productivity = 1.0; // uninitialized
712 String leftSideDataSet = null;
713 Matcher StringNum = stringNumber.matcher(hintToken.hintParams);
714 if (StringNum.find()) {
715 String matchedGroup = StringNum.group();
716 Pattern var = Pattern.compile("[a-zA-Z]\\w*"); // any word character [a-zA-Z_0-9]
717 Matcher matVar = var.matcher(matchedGroup);
718 if (matVar.find()) {
719 leftSideDataSet = matVar.group();
720 }
721 else {
722 throw new SqlppParseException(getSourceLocation(hintToken), "Expected productivity collection name");
723 }
724 Matcher numMat = number.matcher(matchedGroup);
725 if (numMat.find()) {
726 productivity = Double.parseDouble (numMat.group());
727 }
728 else {
Vijay Sarathyc70bae02023-03-13 13:24:05 -0700729 throw new SqlppParseException(getSourceLocation(hintToken), "Productivity has to be a decimal value greater than 0");
Vijay Sarathy7ba271a2023-03-09 08:07:43 -0800730 }
731 }
732 else {
733 throw new SqlppParseException(getSourceLocation(hintToken), "Invalid format for productivity values");
734 }
735 // attach hint to global scope
736 return new JoinProductivityAnnotation (productivity, leftSideDataSet);
737 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800738 case HASH_BROADCAST_JOIN_HINT:
Vijay Sarathy691e93b2022-11-20 14:31:47 -0800739 if (hintToken.hintParams == null) {
740 return new BroadcastExpressionAnnotation(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
741 }
742 else {
743 // if parameter parsing fails then ignore this hint.
744 String name = parseBroadcastJoinParams(hintToken.hintParams);
745 return new BroadcastExpressionAnnotation(name);
746 }
Vijay Sarathy243ef2a2022-08-23 11:56:56 -0700747 case HASH_JOIN_HINT:
Vijay Sarathy691e93b2022-11-20 14:31:47 -0800748 if (hintToken.hintParams == null) {
749 throw new SqlppParseException(getSourceLocation(hintToken), "Expected hash join build/probe collection name");
750 }
751 else {
752 // if parameter parsing fails then ignore this hint.
753 Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> pair = parseHashJoinParams(hintToken.hintParams);
754 return new HashJoinExpressionAnnotation(pair);
755 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800756 case INDEXED_NESTED_LOOP_JOIN_HINT:
757 if (hintToken.hintParams == null) {
758 return IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
759 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800760 // if parameter parsing fails then return hint annotation without parameters
761 onParseErrorReturn = IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800762 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
763 return IndexedNLJoinExpressionAnnotation.newInstance(indexNames);
764 }
765 case RANGE_HINT:
766 Expression rangeExpr = parseExpression(hintToken.hintParams);
767 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
768 return new RangeAnnotation(rangeMap);
769 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
770 if (hintToken.hintParams == null) {
771 return SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE_ANY_INDEX;
772 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800773 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800774 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
775 return SkipSecondaryIndexSearchExpressionAnnotation.newInstance(indexNames);
776 }
Tin Vu4f1090d2021-09-21 02:06:32 -0700777 case SPATIAL_JOIN_HINT:
778 List<Literal> hintValues = parseParenthesizedLiteralList(hintToken.hintParams);
779
780 // Handle exceptions
781 if (hintValues.size() != 6) {
782 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. 6 arguments are required.",
783 hintToken.hint.toString()));
784 }
785
786 for (int i = 0; i < 4; i++) {
787 Literal lit = hintValues.get(i);
788 if ((lit.getLiteralType() != Literal.Type.DOUBLE)
789 && (lit.getLiteralType() != Literal.Type.FLOAT)
790 && (lit.getLiteralType() != Literal.Type.LONG)
791 && (lit.getLiteralType() != Literal.Type.INTEGER)){
792 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. Numeric value is required for first 4 arguments.",
793 hintToken.hint.toString()));
794 }
795 }
796
797 for (int i = 4; i < 6; i++) {
798 Literal lit = hintValues.get(i);
799 if ((lit.getLiteralType() != Literal.Type.LONG)
800 && (lit.getLiteralType() != Literal.Type.INTEGER)) {
801 throw new SqlppParseException(getSourceLocation(hintToken), String.format("Unexpected hint: %s. Long/int is required for last 2 arguments.",
802 hintToken.hint.toString()));
803 }
804 }
805
806 try {
807 double minX = ExpressionUtils.getDoubleValue(hintValues.get(0));
808 double minY = ExpressionUtils.getDoubleValue(hintValues.get(1));
809 double maxX = ExpressionUtils.getDoubleValue(hintValues.get(2));
810 double maxY = ExpressionUtils.getDoubleValue(hintValues.get(3));
811 int numRows = (int) ExpressionUtils.getLongValue(hintValues.get(4));
812 int numColumns = (int) ExpressionUtils.getLongValue(hintValues.get(5));
813 SpatialJoinAnnotation spatialJoinAnn = new SpatialJoinAnnotation(minX, minY, maxX, maxY, numRows, numColumns);
814 return spatialJoinAnn;
815 } catch (TypeMismatchException e) {
816 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
817 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800818 case USE_SECONDARY_INDEX_SEARCH_HINT:
819 if (hintToken.hintParams == null) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800820 throw new SqlppParseException(getSourceLocation(hintToken), "Expected index name(s)");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800821 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800822 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800823 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
824 return SecondaryIndexSearchPreferenceAnnotation.newInstance(indexNames);
825 }
826 default:
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800827 throw new SqlppParseException(getSourceLocation(hintToken), "Unexpected hint");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800828 }
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800829 } catch (SqlppParseException e) {
830 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500831 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800832 hintToken.hint.toString(), e.getMessage()));
833 }
834 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800835 } catch (CompilationException e) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800836 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500837 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800838 hintToken.hint.toString(), e.getMessage()));
839 }
840 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800841 }
842 }
843
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700844 private void ensureNoTypeDeclsInFunction(String fnName, List<Pair<VarIdentifier, TypeExpression>> paramList,
845 TypeExpression returnType, Token startToken) throws SqlppParseException {
846 for (Pair<VarIdentifier, TypeExpression> p : paramList) {
847 if (p.second != null) {
848 String paramName = SqlppVariableUtil.toUserDefinedName(p.first.getValue());
849 throw new SqlppParseException(getSourceLocation(startToken),
850 "Unexpected type declaration for parameter " + paramName + " in function " + fnName);
851 }
852 }
853 if (returnType != null) {
854 throw new SqlppParseException(getSourceLocation(startToken),
855 "Unexpected return type declaration for function " + fnName);
856 }
857 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -0700858
859 private void ensureIntegerLiteral(LiteralExpr expr, String errorPrefix) throws SqlppParseException {
860 Literal lit = expr.getValue();
861 if (lit.getLiteralType() != Literal.Type.INTEGER && lit.getLiteralType() != Literal.Type.LONG) {
862 throw new SqlppParseException(expr.getSourceLocation(), errorPrefix + " should be an INTEGER");
863 }
864 }
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700865
866 private DataverseName createDataverseName(List<String> parts, int fromIndex, int toIndex, Token startToken)
867 throws SqlppParseException {
868 try {
869 return DataverseName.create(parts, fromIndex, toIndex);
870 } catch (AsterixException e) {
871 SqlppParseException pe = new SqlppParseException(getSourceLocation(startToken), e.getMessage());
872 pe.initCause(e);
873 throw pe;
874 }
875 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700876}
877
878PARSER_END(SQLPPParser)
879
Yingyi Bu391f09e2015-10-29 13:49:39 -0700880List<Statement> Statement() throws ParseException:
881{
882 scopeStack.push(RootScopeFactory.createRootScope(this));
883 List<Statement> decls = new ArrayList<Statement>();
884 Statement stmt = null;
885}
886{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300887 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700888 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300889 {
890 decls.add(stmt);
891 }
892 )?
893 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700894 )*
895 <EOF>
896 {
897 return decls;
898 }
899}
900
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700901Statement ExplainStatement() throws ParseException:
902{
903 Statement stmt = null;
904 Token explainToken = null;
905}
906{
907 ( <EXPLAIN> { explainToken = token; } )?
908 stmt = SingleStatement()
909 {
910 if (explainToken != null) {
911 if (stmt.getKind() == Statement.Kind.QUERY) {
912 ((Query)stmt).setExplain(true);
913 } else {
914 throw new SqlppParseException(getSourceLocation(explainToken),
915 "EXPLAIN is not supported for this kind of statement");
916 }
917 }
918 return stmt;
919 }
920}
921
Yingyi Bu391f09e2015-10-29 13:49:39 -0700922Statement SingleStatement() throws ParseException:
923{
924 Statement stmt = null;
925}
926{
927 (
928 stmt = DataverseDeclaration()
929 | stmt = FunctionDeclaration()
930 | stmt = CreateStatement()
931 | stmt = LoadStatement()
932 | stmt = DropStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700933 | stmt = SetStatement()
934 | stmt = InsertStatement()
935 | stmt = DeleteStatement()
936 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800937 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700938 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700939 | stmt = CompactStatement()
Dmitry Lychaginf3eec282022-06-10 12:44:25 -0700940 | stmt = AnalyzeStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700941 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700942 )
943 {
944 return stmt;
945 }
946}
947
948DataverseDecl DataverseDeclaration() throws ParseException:
949{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700950 Token startToken = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700951 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700952}
953{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700954 <USE> { startToken = token; } dvName = DataverseName()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700955 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700956 defaultDataverse = dvName;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800957 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700958 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700959 }
960}
961
962Statement CreateStatement() throws ParseException:
963{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700964 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700965 Statement stmt = null;
966}
967{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700968 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700969 (
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700970 stmt = CreateOrReplaceStatement(startToken)
971 | stmt = CreateTypeStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800972 | stmt = CreateNodegroupStatement(startToken)
973 | stmt = CreateDatasetStatement(startToken)
974 | stmt = CreateIndexStatement(startToken)
975 | stmt = CreateDataverseStatement(startToken)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700976 | stmt = CreateFunctionStatement(startToken, false)
Ian Maxon38fe9402020-01-29 19:27:40 -0800977 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800978 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800979 | stmt = CreateFeedStatement(startToken)
980 | stmt = CreateFeedPolicyStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -0800981 | stmt = CreateFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700982 | stmt = CreateViewStatement(startToken, false)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700983 )
984 {
985 return stmt;
986 }
987}
988
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700989Statement CreateOrReplaceStatement(Token startStmtToken) throws ParseException:
990{
991 Statement stmt = null;
992 Token replaceToken = null;
993}
994{
995 <OR> <IDENTIFIER> { replaceToken = token; }
996 (
997 stmt = CreateFunctionStatement(startStmtToken, true)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700998 | stmt = CreateViewStatement(startStmtToken, true)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700999 )
1000 {
1001 // check expected token here to make the grammar extension plugin happy
1002 expectToken(replaceToken, REPLACE);
1003 return stmt;
1004 }
1005}
1006
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001007TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
1008{
1009 TypeDecl stmt = null;
1010}
1011{
1012 <TYPE> stmt = TypeSpecification(startStmtToken)
1013 {
1014 return stmt;
1015 }
1016}
1017
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07001018TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001019{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001020 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001021 boolean ifNotExists = false;
1022 TypeExpression typeExpr = null;
1023}
1024{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001025 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -07001026 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001027 {
1028 boolean dgen = false;
1029 long numValues = -1;
1030 String filename = null;
1031 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
1032 if (hintToken != null) {
1033 String hintParams = hintToken.hintParams;
1034 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
1035 if (splits == null || splits.length != 2) {
1036 throw new SqlppParseException(getSourceLocation(hintToken),
1037 "Expecting /*+ dgen <filename> <numberOfItems> */");
1038 }
1039 dgen = true;
1040 filename = splits[0];
1041 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001042 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001043 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
1044 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
1045 return addSourceLocation(stmt, startStmtToken);
1046 }
1047}
1048
1049NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
1050{
1051 NodegroupDecl stmt = null;
1052}
1053{
1054 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
1055 {
1056 return stmt;
1057 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001058}
1059
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001060NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001061{
1062 String name = null;
1063 String tmp = null;
1064 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001065 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001066}
1067{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001068 name = Identifier() ifNotExists = IfNotExists()
1069 <ON> tmp = Identifier()
1070 {
1071 ncNames.add(new Identifier(tmp));
1072 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001073 ( <COMMA> tmp = Identifier()
1074 {
1075 ncNames.add(new Identifier(tmp));
1076 }
1077 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001078 {
1079 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
1080 return addSourceLocation(stmt, startStmtToken);
1081 }
1082}
1083
1084void Dataset() throws ParseException:
1085{
1086}
1087{
1088 (<DATASET>|<COLLECTION>)
1089}
1090
Ali Alsuliman7e5c5742022-11-18 12:21:13 -08001091void DatasetToken() throws ParseException:
1092{
1093}
1094{
1095 Dataset()
1096}
1097
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001098DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
1099{
1100 DatasetDecl stmt = null;
1101}
1102{
1103 (
1104 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
1105 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
1106 )
1107 {
1108 return stmt;
1109 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001110}
1111
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001112DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001113{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001114 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001115 boolean ifNotExists = false;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001116 TypeExpression typeExpr = null;
1117 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001118 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001119 Triple<List<Integer>, List<List<String>>, List<TypeExpression>> primaryKeyFieldsWithTypes = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001120 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001121 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001122 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -08001123 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -08001124 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001125}
1126{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001127 nameComponents = QualifiedName()
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001128 (typeExpr = DatasetTypeSpecification())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001129 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001130 { String name; }
1131 <WITH>
1132 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001133 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001134 if (!name.equalsIgnoreCase("meta")){
1135 throw new SqlppParseException(getSourceLocation(startStmtToken),
1136 "We can only support one additional associated field called \"meta\".");
1137 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001138 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001139 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001140 )?
1141 ifNotExists = IfNotExists()
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001142 (LOOKAHEAD(3) primaryKeyFieldsWithTypes = PrimaryKeyWithType()
1143 | primaryKeyFields = PrimaryKey())
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001144 (<AUTOGENERATED> { autogenerated = true; } )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001145 ( <HINTS> hints = Properties() )?
1146 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
1147 ( <WITH> withRecord = RecordConstructor() )?
1148 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001149 try {
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07001150 if (typeExpr == null) {
1151 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFieldsWithTypes.second,
1152 primaryKeyFieldsWithTypes.first, autogenerated, filterField == null? null : filterField.first,
1153 filterField == null? null : filterField.second, primaryKeyFieldsWithTypes.third);
1154 final TypeReferenceExpression anyObjectReference = new TypeReferenceExpression(
1155 new Pair(MetadataBuiltinEntities.ANY_OBJECT_DATATYPE.getDataverseName(),
1156 new Identifier(MetadataBuiltinEntities.ANY_OBJECT_DATATYPE.getDatatypeName())));
1157 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, anyObjectReference, null, hints,
1158 DatasetType.INTERNAL, idd, withRecord, ifNotExists);
1159 return addSourceLocation(stmt, startStmtToken);
1160 } else {
1161 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
1162 filterField == null? null : filterField.first, filterField == null? null : filterField.second);
1163 DatasetDeclParametersUtil.adjustInlineTypeDecl(typeExpr, primaryKeyFields.second, primaryKeyFields.first, false);
1164 if (metaTypeExpr != null) {
1165 DatasetDeclParametersUtil.adjustInlineTypeDecl(metaTypeExpr, primaryKeyFields.second, primaryKeyFields.first,
1166 true);
1167 }
1168 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr, hints,
1169 DatasetType.INTERNAL, idd, withRecord, ifNotExists);
1170 return addSourceLocation(stmt, startStmtToken);
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -08001171 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001172 } catch (CompilationException e) {
1173 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1174 }
1175 }
1176}
1177
1178DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
1179{
1180 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001181 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001182 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001183 String adapterName = null;
1184 Map<String,String> properties = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001185 Map<String,String> hints = new HashMap<String,String>();
1186 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001187 RecordConstructor withRecord = null;
1188}
1189{
1190 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001191 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001192 ifNotExists = IfNotExists()
1193 <USING> adapterName = AdapterName() properties = Configuration()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001194 ( <HINTS> hints = Properties() )?
1195 ( <WITH> withRecord = RecordConstructor() )?
1196 {
1197 ExternalDetailsDecl edd = new ExternalDetailsDecl();
1198 edd.setAdapter(adapterName);
1199 edd.setProperties(properties);
1200 try {
Murtadha Hubail353e95f2020-08-24 22:18:04 +03001201 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null, hints, DatasetType.EXTERNAL,
1202 edd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001203 return addSourceLocation(stmt, startStmtToken);
1204 } catch (CompilationException e) {
1205 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1206 }
1207 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001208}
1209
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001210TypeExpression DatasetTypeSpecification() throws ParseException:
1211{
1212 TypeExpression typeExpr = null;
1213}
1214{
1215 (
1216 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
1217 | typeExpr = DatasetReferenceTypeSpecification()
1218 )
1219 {
1220 return typeExpr;
1221 }
1222}
1223
1224TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
1225{
1226 TypeExpression typeExpr = null;
1227}
1228{
1229 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
1230 {
1231 return typeExpr;
1232 }
1233}
1234
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001235RecordTypeDefinition DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001236{
1237 RecordTypeDefinition recordTypeDef = null;
1238 RecordTypeDefinition.RecordKind recordKind = null;
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001239 Token startToken = null, recordKindToken = null;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001240}
1241{
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001242 <LEFTPAREN> { startToken = token; } recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001243 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
1244 {
1245 if (recordKind == null) {
1246 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
1247 } else if (!allowRecordKindModifier) {
1248 throw createUnexpectedTokenError(recordKindToken);
1249 }
1250 recordTypeDef.setRecordKind(recordKind);
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001251 return addSourceLocation(recordTypeDef, startToken);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001252 }
1253}
1254
1255RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
1256{
1257 RecordTypeDefinition recType = new RecordTypeDefinition();
1258}
1259{
1260 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
1261 {
1262 return recType;
1263 }
1264}
1265
1266void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
1267{
1268 String fieldName;
1269 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001270 boolean nullable = true, missable = true;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001271}
1272{
1273 fieldName = Identifier()
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001274 type = TypeReference() ( <NOT> <UNKNOWN> { nullable = false; missable = false; } )?
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001275 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001276 recType.addField(fieldName, type, nullable, missable);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001277 }
1278}
1279
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001280CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
1281{
1282 CreateIndexStatement stmt = null;
1283}
1284{
1285 (
1286 <INDEX> stmt = IndexSpecification(startStmtToken)
1287 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
1288 )
1289 {
1290 return stmt;
1291 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001292}
1293
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001294CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001295{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001296 Pair<DataverseName,Identifier> nameComponents = null;
Glenn67fd1f32021-02-25 16:04:49 -08001297 String indexName = null;
1298 IndexParams indexParams = null;
1299 CreateIndexStatement.IndexedElement indexedElement = null;
1300 List<CreateIndexStatement.IndexedElement> indexedElementList = new ArrayList<CreateIndexStatement.IndexedElement>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001301 boolean enforced = false;
Glenn67fd1f32021-02-25 16:04:49 -08001302 boolean ifNotExists = false;
1303 boolean hasUnnest = false;
1304 String fullTextConfigName = null;
1305 Token startElementToken = null;
Ali Alsuliman931e7382021-08-08 21:55:59 +03001306 Boolean excludeUnknown = null;
Ali Alsuliman465a7282021-11-24 12:51:32 -08001307 Pair<Map<String, String>, Boolean> castConfigDefaultNull;
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001308 Boolean castDefaultNull = null;
Ali Alsuliman465a7282021-11-24 12:51:32 -08001309 Map<String, String> castConfig = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001310}
1311{
Ali Alsuliman8351d252017-09-24 00:43:15 -07001312 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001313 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -07001314 <ON> nameComponents = QualifiedName()
Glenn67fd1f32021-02-25 16:04:49 -08001315 <LEFTPAREN> { startElementToken = token; }
1316 indexedElement = IndexedElement(startElementToken) {
1317 indexedElementList.add(indexedElement);
1318 hasUnnest |= indexedElement.hasUnnest();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001319 }
Glenn67fd1f32021-02-25 16:04:49 -08001320 (<COMMA> { startElementToken = token; }
1321 indexedElement = IndexedElement(startElementToken) {
1322 indexedElementList.add(indexedElement);
1323 hasUnnest |= indexedElement.hasUnnest();
1324 }
1325 )*
1326 <RIGHTPAREN>
1327 ( <TYPE> indexParams = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001328 ( LOOKAHEAD({laIdentifier(EXCLUDE) || laIdentifier(INCLUDE)}) <IDENTIFIER>
1329 {
1330 if (isToken(EXCLUDE)) {
1331 excludeUnknown = true;
1332 } else if (isToken(INCLUDE)) {
1333 excludeUnknown = false;
1334 } else {
1335 throw createUnexpectedTokenError();
1336 }
1337 } <UNKNOWN> <KEY>
Ali Alsuliman931e7382021-08-08 21:55:59 +03001338 )?
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07001339
Ali Alsuliman465a7282021-11-24 12:51:32 -08001340 ( <CAST><LEFTPAREN> castConfigDefaultNull = CastDefaultNull() <RIGHTPAREN>
1341 {
1342 castConfig = castConfigDefaultNull.first;
1343 castDefaultNull = castConfigDefaultNull.second;
1344 }
1345 )?
Ali Alsuliman8351d252017-09-24 00:43:15 -07001346 )
1347 {
Glenn67fd1f32021-02-25 16:04:49 -08001348 IndexType indexType;
1349 int gramLength;
1350 if (indexParams != null) {
1351 indexType = indexParams.type;
1352 gramLength = indexParams.gramLength;
1353 fullTextConfigName = indexParams.fullTextConfig;
1354 } else {
1355 indexType = hasUnnest ? IndexType.ARRAY : IndexType.BTREE;
1356 gramLength = -1;
1357 fullTextConfigName = null;
Ali Alsuliman8351d252017-09-24 00:43:15 -07001358 }
Glenn67fd1f32021-02-25 16:04:49 -08001359 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman931e7382021-08-08 21:55:59 +03001360 new Identifier(indexName), indexType, indexedElementList, enforced, gramLength, fullTextConfigName, ifNotExists,
Ali Alsuliman465a7282021-11-24 12:51:32 -08001361 excludeUnknown, castDefaultNull, castConfig);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001362 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001363 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001364}
1365
Glenn67fd1f32021-02-25 16:04:49 -08001366CreateIndexStatement.IndexedElement IndexedElement(Token startElementToken) throws ParseException:
1367{
1368 Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1369 Pair<List<String>, IndexedTypeExpression> elementSimple = null;
1370 int elementSimpleSource = 0;
1371}
1372{
1373 (
1374 element = IndexedElementUnnestSelect()
1375 | (
1376 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1377 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1378 <DOT> elementSimple = IndexedField()
1379 { elementSimpleSource = 1; }
1380 )
1381 | elementSimple = IndexedField()
1382 | <LEFTPAREN> ( element = IndexedElementUnnestSelect() | elementSimple = IndexedField() ) <RIGHTPAREN>
1383 )
1384 {
1385 int source;
1386 List<List<String>> unnestList;
1387 List<Pair<List<String>, IndexedTypeExpression>> projectList;
1388 if (elementSimple != null) {
1389 source = elementSimpleSource;
1390 unnestList = null;
1391 projectList = Collections.singletonList(elementSimple);
1392 } else {
1393 source = element.first;
1394 unnestList = element.second;
1395 projectList = element.third;
1396 }
1397 CreateIndexStatement.IndexedElement ie = new CreateIndexStatement.IndexedElement(source, unnestList, projectList);
1398 ie.setSourceLocation(getSourceLocation(startElementToken));
1399 return ie;
1400 }
1401}
1402
1403Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelect()
1404 throws ParseException:
1405{
1406 int source = 0;
1407 Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1408}
1409{
1410 <UNNEST>
1411 (
1412 (
1413 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1414 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1415 <DOT> element = IndexedElementUnnestSelectBody() { source = 1; }
1416 ) | element = IndexedElementUnnestSelectBody()
1417 )
1418 {
1419 return new Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(
1420 source, element.first, element.second
1421 );
1422 }
1423}
1424
1425Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelectBody()
1426 throws ParseException:
1427{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001428 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001429 IndexedTypeExpression type = null;
1430 List<List<String>> unnestList = new ArrayList();
Glenn67fd1f32021-02-25 16:04:49 -08001431 List<Pair<List<String>, IndexedTypeExpression>> projectList = new ArrayList();
1432}
1433{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001434 path = MultipartIdentifier() { unnestList.add(path.first); }
1435 ( <UNNEST> path = MultipartIdentifier() { unnestList.add(path.first); })*
Glenn67fd1f32021-02-25 16:04:49 -08001436 (
Glenn58329202021-04-09 12:03:46 -07001437 ( <COLON> type = IndexedTypeExpr(false)
Glenn67fd1f32021-02-25 16:04:49 -08001438 {
1439 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, type));
1440 }
1441 ) |
1442 (
Glenn58329202021-04-09 12:03:46 -07001443 <SELECT> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1444 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001445 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001446 }
1447 ( <COMMA> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1448 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001449 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001450 }
1451 )*
Glenn67fd1f32021-02-25 16:04:49 -08001452 )
1453 )?
1454 {
1455 if (projectList.isEmpty()) {
1456 // To support the case (<UNNEST> IDENTIFIER)* IDENTIFIER w/o any type specification.
1457 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, null));
1458 }
1459
1460 return new Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(unnestList, projectList);
1461 }
1462}
1463
1464Pair<List<String>, IndexedTypeExpression> IndexedField() throws ParseException:
1465{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001466 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001467 IndexedTypeExpression type = null;
1468}
1469{
Glenn58329202021-04-09 12:03:46 -07001470 path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(true) )?
Glenn67fd1f32021-02-25 16:04:49 -08001471 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001472 return new Pair<List<String>, IndexedTypeExpression>(path.first, type);
Glenn67fd1f32021-02-25 16:04:49 -08001473 }
1474}
1475
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001476CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
1477{
Glenn67fd1f32021-02-25 16:04:49 -08001478 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001479 String indexName = null;
1480 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001481}
1482{
1483 (indexName = Identifier())? ifNotExists = IfNotExists()
1484 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
1485 {
1486 if (indexName == null) {
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07001487 indexName = MetadataConstants.PRIMARY_INDEX_PREFIX + nameComponents.second;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001488 }
Glenn67fd1f32021-02-25 16:04:49 -08001489 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman465a7282021-11-24 12:51:32 -08001490 new Identifier(indexName), IndexType.BTREE, Collections.emptyList(), false, -1, null, ifNotExists, null, null,
1491 Collections.emptyMap());
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001492 return addSourceLocation(stmt, startStmtToken);
1493 }
1494}
1495
Yingyi Bu391f09e2015-10-29 13:49:39 -07001496String FilterField() throws ParseException :
1497{
1498 String filterField = null;
1499}
1500{
1501 filterField = Identifier()
1502 {
1503 return filterField;
1504 }
1505}
1506
1507IndexParams IndexType() throws ParseException:
1508{
1509 IndexType type = null;
1510 int gramLength = 0;
Rui Guoe6986dd2020-11-25 19:50:06 -08001511 String fullTextConfig = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001512}
1513{
1514 (<BTREE>
1515 {
1516 type = IndexType.BTREE;
1517 }
1518 | <RTREE>
1519 {
1520 type = IndexType.RTREE;
1521 }
1522 | <KEYWORD>
1523 {
1524 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1525 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001526 | <FULLTEXT>
Taewoo Kimc49405a2017-01-04 00:30:43 -08001527 {
1528 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1529 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001530 // For now we don't allow inverted index creation using a full-text config in another data verse.
1531 // We may want to support corss-dataverse full-text config access later
1532 // If so, replace the Identifier() with QualifiedName() to get the dataverse name
1533 ( <USING> Identifier()
1534 {
1535 fullTextConfig = token.image;
1536 }
1537 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001538 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1539 {
1540 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1541 gramLength = Integer.valueOf(token.image);
1542 }
1543 <RIGHTPAREN>)
1544 {
Rui Guoe6986dd2020-11-25 19:50:06 -08001545 return new IndexParams(type, gramLength, fullTextConfig);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001546 }
1547}
1548
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001549CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1550{
1551 CreateDataverseStatement stmt = null;
1552}
1553{
1554 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1555 {
1556 return stmt;
1557 }
1558}
1559
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001560CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001561{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001562 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001563 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001564}
1565{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001566 dvName = DataverseName() ifNotExists = IfNotExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001567 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001568 CreateDataverseStatement stmt = new CreateDataverseStatement(dvName, null, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001569 return addSourceLocation(stmt, startStmtToken);
1570 }
1571}
1572
Ian Maxon38fe9402020-01-29 19:27:40 -08001573CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1574{
1575 CreateAdapterStatement stmt = null;
1576}
1577{
1578 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1579 {
1580 return stmt;
1581 }
1582}
1583
1584CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1585{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001586 Pair<DataverseName,Identifier> adapterName = null;
1587 Pair<DataverseName,Identifier> libraryName = null;
1588 List<String> externalIdentifier = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001589 boolean ifNotExists = false;
1590}
1591{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001592 adapterName = QualifiedName()
1593 ifNotExists = IfNotExists()
1594 <AS> externalIdentifier = FunctionExternalIdentifier()
1595 <AT> libraryName = QualifiedName()
1596 {
1597 CreateAdapterStatement stmt = new CreateAdapterStatement(adapterName.first, adapterName.second.getValue(),
1598 libraryName.first, libraryName.second.getValue(), externalIdentifier, ifNotExists);
1599 return addSourceLocation(stmt, startStmtToken);
1600 }
1601}
Ian Maxon38fe9402020-01-29 19:27:40 -08001602
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001603CreateViewStatement CreateViewStatement(Token startStmtToken, boolean orReplace) throws ParseException:
1604{
1605 CreateViewStatement stmt = null;
1606}
1607{
1608 <VIEW> stmt = ViewSpecification(startStmtToken, orReplace)
1609 {
1610 return stmt;
1611 }
1612}
1613
1614CreateViewStatement ViewSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
1615{
1616 Pair<DataverseName, Identifier> nameComponents = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001617 TypeExpression typeExpr = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001618 boolean ifNotExists = false;
1619 Token beginPos = null, endPos = null;
1620 Expression viewBodyExpr = null;
Ali Alsuliman465a7282021-11-24 12:51:32 -08001621 Pair<Map<String, String>, Boolean> viewConfigDefaultNull;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001622 Boolean defaultNull = null;
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001623 Map<String, String> viewConfig = null;
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001624 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001625 Pair<List<Integer>, List<List<String>>> foreignKeyFields = null;
1626 Pair<DataverseName, Identifier> refNameComponents = null;
1627 List<CreateViewStatement.ForeignKeyDecl> foreignKeyDecls = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001628 DataverseName currentDataverse = defaultDataverse;
1629}
1630{
1631 nameComponents = QualifiedName()
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001632 (
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001633 (
1634 typeExpr = DatasetTypeSpecification()
1635 ifNotExists = IfNotExists()
Ali Alsuliman465a7282021-11-24 12:51:32 -08001636 viewConfigDefaultNull = CastDefaultNull()
1637 {
1638 viewConfig = viewConfigDefaultNull.first;
1639 defaultNull = viewConfigDefaultNull.second;
1640 }
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001641 (
1642 <PRIMARY> <KEY> <LEFTPAREN> primaryKeyFields = PrimaryKeyFields() <RIGHTPAREN>
1643 <NOT> <ENFORCED>
1644 )?
1645 (
1646 <IDENTIFIER> { expectToken(FOREIGN); } <KEY> <LEFTPAREN> foreignKeyFields = PrimaryKeyFields() <RIGHTPAREN>
1647 <IDENTIFIER> { expectToken(REFERENCES); } refNameComponents = QualifiedName()
1648 <NOT> <ENFORCED>
1649 {
1650 if (foreignKeyDecls == null) {
1651 foreignKeyDecls = new ArrayList<CreateViewStatement.ForeignKeyDecl>();
1652 }
1653 foreignKeyDecls.add(new CreateViewStatement.ForeignKeyDecl(foreignKeyFields.second,
1654 foreignKeyFields.first, refNameComponents.first, refNameComponents.second));
1655 }
1656 )*
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001657 )
1658 |
1659 ( ifNotExists = IfNotExists() )
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001660 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001661 {
1662 if (orReplace && ifNotExists) {
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001663 throw new SqlppParseException(getSourceLocation(startStmtToken), "Unexpected IF NOT EXISTS");
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001664 }
1665 }
1666 <AS>
1667 {
1668 beginPos = token;
1669 createNewScope();
1670 if (nameComponents.first != null) {
1671 defaultDataverse = nameComponents.first;
1672 }
1673 }
1674 viewBodyExpr = ViewBody()
1675 {
1676 endPos = token;
1677 String viewBody = extractFragment(beginPos.beginLine, beginPos.beginColumn + 1, endPos.endLine,
1678 endPos.endColumn + 1);
1679 removeCurrentScope();
1680 defaultDataverse = currentDataverse;
Dmitry Lychagin4fb583d2021-11-10 17:25:28 -08001681 if (typeExpr != null && primaryKeyFields != null) {
1682 DatasetDeclParametersUtil.adjustInlineTypeDecl(typeExpr, primaryKeyFields.second, primaryKeyFields.first, false);
1683 }
Dmitry Lychagin914983f2021-09-30 15:11:11 -07001684 CreateViewStatement.KeyDecl primaryKeyDecl = primaryKeyFields != null ?
1685 new CreateViewStatement.KeyDecl(primaryKeyFields.second, primaryKeyFields.first) : null;
1686 CreateViewStatement stmt = new CreateViewStatement(nameComponents.first, nameComponents.second.getValue(),
1687 typeExpr, viewBody, viewBodyExpr, defaultNull, viewConfig, primaryKeyDecl, foreignKeyDecls, orReplace,
1688 ifNotExists);
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001689 return addSourceLocation(stmt, startStmtToken);
1690 }
1691}
1692
1693Expression ViewBody() throws ParseException:
1694{
1695 Expression viewBodyExpr = null;
1696}
1697{
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001698 (
1699 ( viewBodyExpr = VariableRef() ( viewBodyExpr = FieldAccessor(viewBodyExpr) )* )
1700 | viewBodyExpr = SelectExpression(true)
1701 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001702 {
1703 return viewBodyExpr;
1704 }
1705}
1706
Ali Alsuliman465a7282021-11-24 12:51:32 -08001707Pair<Map<String, String>, Boolean> CastDefaultNull() throws ParseException:
1708{
1709 Map<String, String> castConfig = null;
1710 Boolean defaultNull = null;
1711 String propertyName = null, propertyValue = null;
1712}
1713{
1714 <IDENTIFIER> { expectToken(DEFAULT); } <NULL> { defaultNull = true; }
1715 (
1716 LOOKAHEAD(2) <IDENTIFIER> { propertyName = token.image.toLowerCase(); } propertyValue = StringLiteral()
1717 {
1718 if (castConfig == null) {
1719 castConfig = new HashMap<String, String>();
1720 }
1721 castConfig.put(propertyName, propertyValue);
1722 }
1723 )*
1724 {
1725 return new Pair<Map<String, String>, Boolean>(castConfig, defaultNull);
1726 }
1727}
1728
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001729CreateFunctionStatement CreateFunctionStatement(Token startStmtToken, boolean orReplace) throws ParseException:
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001730{
1731 CreateFunctionStatement stmt = null;
1732}
1733{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001734 <FUNCTION> stmt = FunctionSpecification(startStmtToken, orReplace)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001735 {
1736 return stmt;
1737 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001738}
1739
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001740CreateFunctionStatement FunctionSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001741{
Ian Maxon38fe9402020-01-29 19:27:40 -08001742 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001743 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001744 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001745 List<Pair<VarIdentifier,TypeExpression>> params = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001746 int arity = 0;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001747 TypeExpression returnType = null;
1748 Token beginPos = null, endPos = null;
1749 Expression functionBodyExpr = null;
1750 Pair<DataverseName,Identifier> libraryName = null;
1751 List<String> externalIdentifier = null;
1752 RecordConstructor withOptions = null;
1753 boolean ifNotExists = false;
1754 CreateFunctionStatement stmt = null;
1755 DataverseName currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001756}
1757{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001758 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001759 {
1760 defaultDataverse = fctName.dataverse;
1761 }
Dmitry Lychagin541652082020-11-09 14:04:53 -08001762 paramsWithArity = FunctionParameters()
1763 {
1764 arity = paramsWithArity.first;
1765 params = paramsWithArity.second;
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001766 signature = new FunctionSignature(fctName.dataverse, fctName.function, arity);
Dmitry Lychagin541652082020-11-09 14:04:53 -08001767 }
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001768 ifNotExists = IfNotExists()
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001769 {
1770 if (orReplace && ifNotExists) {
1771 throw new SqlppParseException(getSourceLocation(token), "Unexpected IF NOT EXISTS");
1772 }
1773 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001774 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001775 (
1776 (
1777 <LEFTBRACE>
1778 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001779 beginPos = token;
1780 createNewScope();
Ian Maxon38fe9402020-01-29 19:27:40 -08001781 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001782 functionBodyExpr = FunctionBody()
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001783 <RIGHTBRACE>
1784 {
1785 endPos = token;
1786 String functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine,
1787 endPos.beginColumn);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001788 getCurrentScope().addFunctionDescriptor(signature, false);
1789 removeCurrentScope();
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001790 ensureNoTypeDeclsInFunction(fctName.function, params, returnType, startStmtToken);
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001791 stmt = new CreateFunctionStatement(signature, params, functionBody, functionBodyExpr, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001792 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001793 )
1794 |
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001795 (
1796 <AS> externalIdentifier = FunctionExternalIdentifier()
1797 <AT> libraryName = QualifiedName()
1798 (<WITH> withOptions = RecordConstructor())?
1799 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001800 try {
1801 stmt = new CreateFunctionStatement(signature, params, returnType, libraryName.first,
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001802 libraryName.second.getValue(), externalIdentifier, withOptions, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001803 } catch (AlgebricksException e) {
1804 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1805 }
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001806 }
1807 )
Ian Maxon38fe9402020-01-29 19:27:40 -08001808 )
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001809 {
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001810 defaultDataverse = currentDataverse;
1811 return addSourceLocation(stmt, startStmtToken);
1812 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001813}
1814
Dmitry Lychagin541652082020-11-09 14:04:53 -08001815Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameters() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001816{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001817 Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> params = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001818}
1819{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001820 <LEFTPAREN> (params = FunctionParameterList())? <RIGHTPAREN>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001821 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001822 if (params == null) {
1823 params = new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(
1824 0, Collections.<Pair<VarIdentifier, TypeExpression>>emptyList()
1825 );
1826 }
1827 return params;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001828 }
1829}
1830
Dmitry Lychagin541652082020-11-09 14:04:53 -08001831Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameterList() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001832{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001833 List<Pair<VarIdentifier, TypeExpression>> paramList = null;
1834 Pair<VarIdentifier, TypeExpression> param = null;
1835 int arity = 0;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001836}
1837{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001838 (
1839 (
1840 <DOT> <DOT> <DOT>
1841 {
1842 param = new Pair<VarIdentifier, TypeExpression>(
1843 SqlppVariableUtil.toInternalVariableIdentifier(UDF_VARARGS_PARAM_NAME), null
1844 );
1845 paramList = Collections.<Pair<VarIdentifier, TypeExpression>>singletonList(param);
1846 arity = FunctionIdentifier.VARARGS;
1847 }
1848 )
1849 |
1850 (
1851 param = FunctionParameter()
1852 {
1853 paramList = new ArrayList<Pair<VarIdentifier, TypeExpression>>();
1854 paramList.add(param);
1855 }
1856 ( <COMMA> param = FunctionParameter() { paramList.add(param); } )*
1857 {
1858 arity = paramList.size();
1859 }
1860 )
1861 )
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001862 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001863 return new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(arity, paramList);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001864 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001865}
1866
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001867Pair<VarIdentifier,TypeExpression> FunctionParameter() throws ParseException:
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001868{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001869 String name = null;
1870 TypeExpression type = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001871}
1872{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001873 name = VariableIdentifier()
1874 ( LOOKAHEAD(3) (<COLON>)? type = TypeExpr(false) )?
1875 {
1876 return new Pair<VarIdentifier,TypeExpression>(new VarIdentifier(name), type);
1877 }
1878}
1879
1880TypeExpression FunctionReturnType() throws ParseException:
1881{
1882 TypeExpression returnType = null;
1883}
1884{
1885 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = TypeExpr(false) )?
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001886 {
1887 return returnType;
1888 }
1889}
1890
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001891Expression FunctionBody() throws ParseException:
1892{
1893 Expression functionBodyExpr = null;
1894}
1895{
1896 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1897 {
1898 return functionBodyExpr;
1899 }
1900}
1901
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001902List<String> FunctionExternalIdentifier() throws ParseException:
1903{
1904 String ident = null;
1905 List<String> identList = new ArrayList(2);
1906}
1907{
1908 ident = StringLiteral() { identList.add(ident.trim()); }
1909 ( <COMMA> ident = StringLiteral() { identList.add(ident.trim()); } )*
1910 {
1911 return identList;
1912 }
1913}
1914
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001915CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1916{
1917 CreateFeedStatement stmt = null;
1918}
1919{
1920 <FEED> stmt = FeedSpecification(startStmtToken)
1921 {
1922 return stmt;
1923 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001924}
1925
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001926CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001927{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001928 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001929 boolean ifNotExists = false;
1930 String adapterName = null;
1931 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001932 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001933 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001934 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001935}
1936{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001937 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001938 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001939 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001940 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001941 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001942 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001943 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001944 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001945 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001946 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001947}
1948
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001949CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1950{
1951 CreateFeedPolicyStatement stmt = null;
1952}
1953{
1954 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1955 {
1956 return stmt;
1957 }
1958}
1959
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001960CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001961{
Michael Blowd6cf6412016-06-30 02:44:35 -04001962 String policyName = null;
1963 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001964 String sourcePolicyFile = null;
1965 String definition = null;
1966 boolean ifNotExists = false;
1967 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001968 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001969}
1970{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001971 policyName = Identifier() ifNotExists = IfNotExists()
1972 <FROM>
1973 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001974 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001975 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001976 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001977 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1978 {
1979 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1980 }
1981 )
1982 {
1983 return addSourceLocation(stmt, startStmtToken);
1984 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001985}
1986
Rui Guoe6986dd2020-11-25 19:50:06 -08001987Statement CreateFullTextStatement(Token startStmtToken) throws ParseException:
1988{
1989 Statement stmt = null;
1990}
1991{
1992 (
1993 <FULLTEXT>
1994 (
1995 <FILTER> stmt = CreateFullTextFilterSpec(startStmtToken)
1996 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = CreateFullTextConfigSpec(startStmtToken))
1997 )
1998 )
1999 {
2000 return stmt;
2001 }
2002}
2003
2004CreateFullTextFilterStatement CreateFullTextFilterSpec(Token startStmtToken) throws ParseException:
2005{
2006 CreateFullTextFilterStatement stmt = null;
2007 Pair<DataverseName,Identifier> nameComponents = null;
2008 boolean ifNotExists = false;
2009 RecordConstructor expr = null;
2010}
2011{
2012 (
2013 nameComponents = QualifiedName() ifNotExists = IfNotExists()
2014 <AS>
2015 expr = RecordConstructor()
2016 )
2017 {
2018 try {
2019 stmt = new CreateFullTextFilterStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
2020 return addSourceLocation(stmt, startStmtToken);
2021 } catch (CompilationException e) {
2022 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
2023 }
2024 }
2025}
2026
2027CreateFullTextConfigStatement CreateFullTextConfigSpec(Token startStmtToken) throws ParseException:
2028{
2029 CreateFullTextConfigStatement stmt = null;
2030 Pair<DataverseName,Identifier> nameComponents = null;
2031 boolean ifNotExists = false;
2032 RecordConstructor expr = null;
2033}
2034{
2035 (
2036 nameComponents = QualifiedName() ifNotExists = IfNotExists()
2037 <AS>
2038 expr = RecordConstructor()
2039 )
2040 {
2041 try {
2042 stmt = new CreateFullTextConfigStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
2043 return addSourceLocation(stmt, startStmtToken);
2044 } catch (CompilationException e) {
2045 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
2046 }
2047 }
2048}
2049
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002050CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
2051{
2052 CreateSynonymStatement stmt = null;
2053}
2054{
2055 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
2056 {
2057 return stmt;
2058 }
2059}
2060
2061CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
2062{
2063 Pair<DataverseName,Identifier> nameComponents = null;
2064 Pair<DataverseName,Identifier> objectNameComponents = null;
2065 boolean ifNotExists = false;
2066}
2067{
2068 nameComponents = QualifiedName() ifNotExists = IfNotExists()
2069 <FOR> objectNameComponents = QualifiedName()
2070 {
2071 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
2072 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
2073 return addSourceLocation(stmt, startStmtToken);
2074 }
2075}
2076
Yingyi Bu391f09e2015-10-29 13:49:39 -07002077boolean IfNotExists() throws ParseException:
2078{
2079}
2080{
Yingyi Budaa549c2016-06-28 22:30:52 -07002081 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002082 {
2083 return true;
2084 }
2085 )?
2086 {
2087 return false;
2088 }
2089}
2090
Xikui Wang9d63f622017-05-18 17:50:44 -07002091void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002092{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002093 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07002094 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002095}
2096{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002097 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002098 {
2099 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
2100 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
2101 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07002102 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002103 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07002104 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002105 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
2106 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07002107 }
2108 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07002109}
2110
2111String GetPolicy() throws ParseException:
2112{
2113 String policy = null;
2114}
2115{
2116 <USING> <POLICY> policy = Identifier()
2117 {
2118 return policy;
2119 }
2120
2121}
2122
2123FunctionSignature FunctionSignature() throws ParseException:
2124{
2125 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08002126 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> params = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002127 int arity = 0;
2128}
2129{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002130 fctName = FunctionName()
2131 (
Dmitry Lychagin541652082020-11-09 14:04:53 -08002132 LOOKAHEAD(2) params = FunctionParameters() { arity = params.first; }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002133 | ( <LEFTPAREN> arity = FunctionArity() <RIGHTPAREN> )
2134 | ( <ATT> arity = FunctionArity() ) // back-compat
2135 )
2136 {
2137 return new FunctionSignature(fctName.dataverse, fctName.function, arity);
2138 }
2139}
Yingyi Bu391f09e2015-10-29 13:49:39 -07002140
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002141int FunctionArity() throws ParseException:
2142{
2143 int arity;
2144}
2145{
2146 <INTEGER_LITERAL>
2147 {
2148 try {
2149 arity = Integer.parseInt(token.image);
2150 } catch (NumberFormatException e) {
2151 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002152 }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07002153 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
2154 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + arity);
2155 }
2156 return arity;
2157 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002158}
2159
Peeyush Gupta6ba5a262023-04-13 11:08:53 -07002160Triple<List<Integer>, List<List<String>>, List<TypeExpression>> PrimaryKeyWithType() throws ParseException:
2161{
2162 Triple<List<Integer>, List<List<String>>, List<TypeExpression>> primaryKeyFieldsWithTypes = null;
2163}
2164{
2165 <PRIMARY> <KEY> <LEFTPAREN> primaryKeyFieldsWithTypes = PrimaryKeyFieldsWithType() <RIGHTPAREN>
2166 {
2167 return primaryKeyFieldsWithTypes;
2168 }
2169}
2170
2171Triple<List<Integer>, List<List<String>>, List<TypeExpression>> PrimaryKeyFieldsWithType() throws ParseException:
2172{
2173 Pair<Integer, List<String>> tmp = null;
2174 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
2175 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
2176 List<TypeExpression> primaryKeyFieldTypes = new ArrayList<TypeExpression>();
2177 TypeExpression type = null;
2178}
2179{
2180 tmp = NestedField() <COLON> type = TypeReference()
2181 {
2182 keyFieldSourceIndicators.add(tmp.first);
2183 primaryKeyFields.add(tmp.second);
2184 primaryKeyFieldTypes.add(type);
2185 }
2186 ( <COMMA> tmp = NestedField() <COLON> type = TypeReference()
2187 {
2188 keyFieldSourceIndicators.add(tmp.first);
2189 primaryKeyFields.add(tmp.second);
2190 primaryKeyFieldTypes.add(type);
2191 }
2192 )*
2193 {
2194 return new Triple<List<Integer>, List<List<String>>, List<TypeExpression>> (keyFieldSourceIndicators,
2195 primaryKeyFields, primaryKeyFieldTypes);
2196 }
2197}
2198
Yingyi Buc9bfe252016-03-01 00:02:40 -08002199Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002200{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07002201 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
2202}
2203{
2204 <PRIMARY> <KEY> primaryKeyFields = PrimaryKeyFields()
2205 {
2206 return primaryKeyFields;
2207 }
2208}
2209
2210Pair<List<Integer>, List<List<String>>> PrimaryKeyFields() throws ParseException:
2211{
Yingyi Buc9bfe252016-03-01 00:02:40 -08002212 Pair<Integer, List<String>> tmp = null;
2213 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002214 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
2215}
2216{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07002217 tmp = NestedField()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002218 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002219 keyFieldSourceIndicators.add(tmp.first);
2220 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002221 }
2222 ( <COMMA> tmp = NestedField()
2223 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002224 keyFieldSourceIndicators.add(tmp.first);
2225 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002226 }
2227 )*
2228 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002229 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002230 }
2231}
2232
2233Statement DropStatement() throws ParseException:
2234{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002235 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002236 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002237}
2238{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002239 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002240 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002241 stmt = DropDatasetStatement(startToken)
2242 | stmt = DropIndexStatement(startToken)
2243 | stmt = DropNodeGroupStatement(startToken)
2244 | stmt = DropTypeStatement(startToken)
2245 | stmt = DropDataverseStatement(startToken)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002246 | stmt = DropAdapterStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002247 | stmt = DropFunctionStatement(startToken)
2248 | stmt = DropFeedStatement(startToken)
2249 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002250 | stmt = DropSynonymStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -08002251 | stmt = DropFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002252 | stmt = DropViewStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002253 )
2254 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002255 return stmt;
2256 }
2257}
2258
2259DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
2260{
2261 DropDatasetStatement stmt = null;
2262}
2263{
2264 Dataset() stmt = DropDatasetSpecification(startStmtToken)
2265 {
2266 return stmt;
2267 }
2268}
2269
2270DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
2271{
2272 Pair<DataverseName,Identifier> pairId = null;
2273 boolean ifExists = false;
2274}
2275{
2276 pairId = QualifiedName() ifExists = IfExists()
2277 {
2278 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
2279 return addSourceLocation(stmt, startStmtToken);
2280 }
2281}
2282
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002283ViewDropStatement DropViewStatement(Token startStmtToken) throws ParseException:
2284{
2285 ViewDropStatement stmt = null;
2286}
2287{
2288 <VIEW> stmt = DropViewSpecification(startStmtToken)
2289 {
2290 return stmt;
2291 }
2292}
2293
2294ViewDropStatement DropViewSpecification(Token startStmtToken) throws ParseException:
2295{
2296 Pair<DataverseName,Identifier> pairId = null;
2297 boolean ifExists = false;
2298}
2299{
2300 pairId = QualifiedName() ifExists = IfExists()
2301 {
2302 ViewDropStatement stmt = new ViewDropStatement(pairId.first, pairId.second, ifExists);
2303 return addSourceLocation(stmt, startStmtToken);
2304 }
2305}
2306
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002307IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
2308{
2309 IndexDropStatement stmt = null;
2310}
2311{
2312 <INDEX> stmt = DropIndexSpecification(startStmtToken)
2313 {
2314 return stmt;
2315 }
2316}
2317
2318IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
2319{
2320 Triple<DataverseName,Identifier,Identifier> tripleId = null;
2321 boolean ifExists = false;
2322}
2323{
2324 tripleId = DoubleQualifiedName() ifExists = IfExists()
2325 {
2326 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
2327 return addSourceLocation(stmt, startStmtToken);
2328 }
2329}
2330
Rui Guoe6986dd2020-11-25 19:50:06 -08002331Statement DropFullTextStatement(Token startStmtToken) throws ParseException:
2332{
2333 Statement stmt = null;
2334}
2335{
2336 <FULLTEXT>
2337 (
2338 <FILTER> stmt = DropFullTextFilterSpec(startStmtToken)
2339 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = DropFullTextConfigSpec(startStmtToken))
2340 )
2341 {
2342 return stmt;
2343 }
2344}
2345
2346FullTextFilterDropStatement DropFullTextFilterSpec(Token startStmtToken) throws ParseException:
2347{
2348 FullTextFilterDropStatement stmt = null;
2349 Pair<DataverseName,Identifier> nameComponents = null;
2350 boolean ifExists = false;
2351}
2352{
2353 nameComponents = QualifiedName() ifExists = IfExists()
2354 {
2355 stmt = new FullTextFilterDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2356 return addSourceLocation(stmt, startStmtToken);
2357 }
2358}
2359
2360FullTextConfigDropStatement DropFullTextConfigSpec(Token startStmtToken) throws ParseException:
2361{
2362 FullTextConfigDropStatement stmt = null;
2363 Pair<DataverseName,Identifier> nameComponents = null;
2364 boolean ifExists = false;
2365}
2366{
2367 nameComponents = QualifiedName() ifExists = IfExists()
2368 {
2369 stmt = new FullTextConfigDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2370 return addSourceLocation(stmt, startStmtToken);
2371 }
2372}
2373
2374
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002375NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
2376{
2377 NodeGroupDropStatement stmt = null;
2378}
2379{
2380 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
2381 {
2382 return stmt;
2383 }
2384}
2385
2386NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
2387{
2388 String id = null;
2389 boolean ifExists = false;
2390}
2391{
2392 id = Identifier() ifExists = IfExists()
2393 {
2394 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
2395 return addSourceLocation(stmt, startStmtToken);
2396 }
2397}
2398
2399TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
2400{
2401 TypeDropStatement stmt = null;
2402}
2403{
2404 <TYPE> stmt = DropTypeSpecification(startStmtToken)
2405 {
2406 return stmt;
2407 }
2408}
2409
2410TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
2411{
2412 Pair<DataverseName,Identifier> pairId = null;
2413 boolean ifExists = false;
2414}
2415{
2416 pairId = TypeName() ifExists = IfExists()
2417 {
2418 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
2419 return addSourceLocation(stmt, startStmtToken);
2420 }
2421}
2422
2423DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
2424{
2425 DataverseDropStatement stmt = null;
2426}
2427{
2428 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
2429 {
2430 return stmt;
2431 }
2432}
2433
2434DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
2435{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002436 DataverseName dvName = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002437 boolean ifExists = false;
2438}
2439{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002440 dvName = DataverseName() ifExists = IfExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002441 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002442 DataverseDropStatement stmt = new DataverseDropStatement(dvName, ifExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002443 return addSourceLocation(stmt, startStmtToken);
2444 }
2445}
2446
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002447AdapterDropStatement DropAdapterStatement(Token startStmtToken) throws ParseException:
2448{
2449 AdapterDropStatement stmt = null;
2450}
2451{
2452 <ADAPTER> stmt = DropAdapterSpecification(startStmtToken)
2453 {
2454 return stmt;
2455 }
2456}
2457
2458AdapterDropStatement DropAdapterSpecification(Token startStmtToken) throws ParseException:
2459{
2460 Pair<DataverseName,Identifier> adapterName = null;
2461 boolean ifExists = false;
2462}
2463{
2464 adapterName = QualifiedName() ifExists = IfExists()
2465 {
2466 AdapterDropStatement stmt = new AdapterDropStatement(adapterName.first, adapterName.second.getValue(), ifExists);
2467 return addSourceLocation(stmt, startStmtToken);
2468 }
2469}
2470
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002471FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
2472{
2473 FunctionDropStatement stmt = null;
2474}
2475{
2476 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
2477 {
2478 return stmt;
2479 }
2480}
2481
2482FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
2483{
2484 FunctionSignature funcSig = null;
2485 boolean ifExists = false;
2486}
2487{
2488 funcSig = FunctionSignature() ifExists = IfExists()
2489 {
2490 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
2491 return addSourceLocation(stmt, startStmtToken);
2492 }
2493}
2494
2495FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
2496{
2497 FeedDropStatement stmt = null;
2498}
2499{
2500 <FEED> stmt = DropFeedSpecification(startStmtToken)
2501 {
2502 return stmt;
2503 }
2504}
2505
2506FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
2507{
2508 Pair<DataverseName,Identifier> pairId = null;
2509 boolean ifExists = false;
2510}
2511{
2512 pairId = QualifiedName() ifExists = IfExists()
2513 {
2514 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
2515 return addSourceLocation(stmt, startStmtToken);
2516 }
2517}
2518
2519FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
2520{
2521 FeedPolicyDropStatement stmt = null;
2522}
2523{
2524 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
2525 {
2526 return stmt;
2527 }
2528}
2529
2530FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
2531{
2532 Pair<DataverseName,Identifier> pairId = null;
2533 boolean ifExists = false;
2534}
2535{
2536 pairId = QualifiedName() ifExists = IfExists()
2537 {
2538 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
2539 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002540 }
2541}
2542
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002543SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
2544{
2545 SynonymDropStatement stmt = null;
2546}
2547{
2548 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
2549 {
2550 return stmt;
2551 }
2552}
2553
2554SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
2555{
2556 Pair<DataverseName,Identifier> pairId = null;
2557 boolean ifExists = false;
2558}
2559{
2560 pairId = QualifiedName() ifExists = IfExists()
2561 {
2562 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
2563 return addSourceLocation(stmt, startStmtToken);
2564 }
2565}
2566
Yingyi Bu391f09e2015-10-29 13:49:39 -07002567boolean IfExists() throws ParseException :
2568{
2569}
2570{
2571 ( LOOKAHEAD(1) <IF> <EXISTS>
2572 {
2573 return true;
2574 }
2575 )?
2576 {
2577 return false;
2578 }
2579}
2580
2581InsertStatement InsertStatement() throws ParseException:
2582{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002583 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002584 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002585 VariableExpr var = null;
2586 Query query = null;
2587 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002588}
2589{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002590 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002591 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002592 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002593 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002594 if (var == null && returnExpression != null) {
2595 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2596 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002597 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08002598 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002599 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2600 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002601 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002602 }
2603}
2604
2605UpsertStatement UpsertStatement() throws ParseException:
2606{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002607 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002608 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002609 VariableExpr var = null;
2610 Query query = null;
2611 Expression returnExpression = null;
2612}
2613{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002614 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002615 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002616 ( <RETURNING> returnExpression = Expression())?
2617 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002618 if (var == null && returnExpression != null) {
2619 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2620 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002621 }
2622 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002623 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2624 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002625 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002626 }
2627}
2628
2629DeleteStatement DeleteStatement() throws ParseException:
2630{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002631 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002632 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002633 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002634 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002635}
2636{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002637 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002638 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002639 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07002640 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002641 if (var == null) {
2642 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2643 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002644 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002645 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07002646 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002647 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002648 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002649}
2650
2651UpdateStatement UpdateStatement() throws ParseException:
2652{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002653 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002654 VariableExpr vars;
2655 Expression target;
2656 Expression condition;
2657 UpdateClause uc;
2658 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
2659}
2660{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002661 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002662 <WHERE> condition = Expression()
2663 <LEFTPAREN> (uc = UpdateClause()
2664 {
2665 ucs.add(uc);
2666 }
2667 (<COMMA> uc = UpdateClause()
2668 {
2669 ucs.add(uc);
2670 }
2671 )*) <RIGHTPAREN>
2672 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002673 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002674 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002675 }
2676}
2677
2678UpdateClause UpdateClause() throws ParseException:
2679{
2680 Expression target = null;
2681 Expression value = null ;
2682 InsertStatement is = null;
2683 DeleteStatement ds = null;
2684 UpdateStatement us = null;
2685 Expression condition = null;
2686 UpdateClause ifbranch = null;
2687 UpdateClause elsebranch = null;
2688}
2689{
2690 (<SET> target = Expression() <EQ> value = Expression()
2691 | is = InsertStatement()
2692 | ds = DeleteStatement()
2693 | us = UpdateStatement()
2694 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
2695 <THEN> ifbranch = UpdateClause()
2696 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
2697 {
2698 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
2699 }
2700 )
2701}
2702
2703Statement SetStatement() throws ParseException:
2704{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002705 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002706 String pn = null;
2707 String pv = null;
2708}
2709{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002710 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002711 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002712 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002713 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002714 }
2715}
2716
Yingyi Bu391f09e2015-10-29 13:49:39 -07002717LoadStatement LoadStatement() throws ParseException:
2718{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002719 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002720 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002721 Identifier datasetName = null;
2722 boolean alreadySorted = false;
2723 String adapterName;
2724 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002725 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002726}
2727{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002728 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002729 {
2730 dataverseName = nameComponents.first;
2731 datasetName = nameComponents.second;
2732 }
2733 <USING> adapterName = AdapterName() properties = Configuration()
2734 (<PRESORTED>
2735 {
2736 alreadySorted = true;
2737 }
2738 )?
2739 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002740 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
2741 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002742 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002743 }
2744}
2745
2746
2747String AdapterName() throws ParseException :
2748{
2749 String adapterName = null;
2750}
2751{
2752 adapterName = Identifier()
2753 {
2754 return adapterName;
2755 }
2756}
2757
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002758Statement AnalyzeStatement() throws ParseException:
2759{
2760 Token startToken = null;
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002761 Statement stmt = null;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002762 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002763}
2764{
Ali Alsuliman7e5c5742022-11-18 12:21:13 -08002765 <ANALYZE> { startToken = token; } DatasetToken() nameComponents = QualifiedName()
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002766 (
2767 stmt = AnalyzeDatasetDropStatement(startToken, nameComponents.first, nameComponents.second)
2768 | stmt = AnalyzeDatasetStatement(startToken, nameComponents.first, nameComponents.second)
2769 )
2770 {
2771 return stmt;
2772 }
2773}
2774
2775Statement AnalyzeDatasetStatement(Token startToken, DataverseName dvName, Identifier identifier) throws ParseException:
2776{
2777 RecordConstructor withRecord = null;
2778}
2779{
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002780 ( <WITH> withRecord = RecordConstructor() )?
2781 {
2782 try {
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002783 AnalyzeStatement stmt = new AnalyzeStatement(dvName, identifier.getValue(), withRecord);
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07002784 return addSourceLocation(stmt, startToken);
2785 } catch (CompilationException e) {
2786 throw new SqlppParseException(getSourceLocation(startToken), e.getMessage());
2787 }
2788 }
2789}
2790
Ali Alsuliman6d72d5e2022-08-08 13:41:08 -07002791Statement AnalyzeDatasetDropStatement(Token startToken, DataverseName dvName, Identifier identifier) throws ParseException:
2792{
2793}
2794{
2795 <DROP> <IDENTIFIER> { expectToken(STATISTICS); }
2796 {
2797 AnalyzeDropStatement stmt = new AnalyzeDropStatement(dvName, identifier.getValue());
2798 return addSourceLocation(stmt, startToken);
2799 }
2800}
2801
Yingyi Bu391f09e2015-10-29 13:49:39 -07002802Statement CompactStatement() throws ParseException:
2803{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002804 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002805 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002806}
2807{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002808 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002809 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002810 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002811 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002812 }
2813}
2814
Yingyi Buab817482016-08-19 21:29:31 -07002815Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002816{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002817 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002818 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002819}
2820{
2821 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002822 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
2823 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
2824 | <START> { startToken = token; } stmt = StartStatement(startToken)
2825 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07002826 )
2827 {
2828 return stmt;
2829 }
2830}
2831
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002832Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002833{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002834 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002835 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002836}
2837{
2838 <FEED> feedNameComponents = QualifiedName()
2839 {
2840 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002841 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002842 }
2843}
2844
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002845AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002846{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002847 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002848 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002849}
2850{
2851 <FEED> feedNameComponents = QualifiedName()
2852 {
2853 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002854 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002855 }
2856}
2857
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002858AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002859{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002860 Pair<DataverseName,Identifier> feedNameComponents = null;
2861 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002862
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002863 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002864}
2865{
2866 (
2867 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002868 {
2869 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
2870 }
2871 )
Yingyi Buab817482016-08-19 21:29:31 -07002872 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002873 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002874 }
2875}
2876
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002877AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002878{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002879 Pair<DataverseName,Identifier> feedNameComponents = null;
2880 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002881
2882 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002883 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002884 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002885 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002886 String whereClauseBody = null;
2887 WhereClause whereClause = null;
2888 Token beginPos = null;
2889 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002890}
2891{
2892 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002893 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002894 (ApplyFunction(appliedFunctions))?
2895 (policy = GetPolicy())?
2896 (
2897 <WHERE>
2898 {
2899 beginPos = token;
2900 whereClause = new WhereClause();
2901 Expression whereExpr;
2902 }
2903 whereExpr = Expression()
2904 {
2905 whereClause.setWhereExpr(whereExpr);
2906 }
2907 )?
2908 {
2909 if (whereClause != null) {
2910 endPos = token;
2911 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2912 }
2913 }
Yingyi Buab817482016-08-19 21:29:31 -07002914 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002915 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002916 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002917 }
2918 )
2919 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002920 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002921 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002922}
2923
2924Map<String,String> Configuration() throws ParseException :
2925{
2926 Map<String,String> configuration = new LinkedHashMap<String,String>();
2927 Pair<String, String> keyValuePair = null;
2928}
2929{
2930 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2931 {
2932 configuration.put(keyValuePair.first, keyValuePair.second);
2933 }
2934 ( <COMMA> keyValuePair = KeyValuePair()
2935 {
2936 configuration.put(keyValuePair.first, keyValuePair.second);
2937 }
2938 )* )? <RIGHTPAREN>
2939 {
2940 return configuration;
2941 }
2942}
2943
2944Pair<String, String> KeyValuePair() throws ParseException:
2945{
2946 String key;
2947 String value;
2948}
2949{
Ali Alsulimane2986012020-05-14 18:14:22 -07002950 <LEFTPAREN> key = ConstantString()
2951 <EQ> ( value = ConstantString() | (<TRUE> | <FALSE>) {value = token.image.toLowerCase();} )
2952 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002953 {
2954 return new Pair<String, String>(key, value);
2955 }
2956}
2957
2958Map<String,String> Properties() throws ParseException:
2959{
2960 Map<String,String> properties = new HashMap<String,String>();
2961 Pair<String, String> property;
2962}
2963{
2964 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
2965 {
2966 properties.put(property.first, property.second);
2967 }
2968 ( <COMMA> property = Property()
2969 {
2970 properties.put(property.first, property.second);
2971 }
2972 )* <RIGHTPAREN> )?
2973 {
2974 return properties;
2975 }
2976}
2977
2978Pair<String, String> Property() throws ParseException:
2979{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002980 String key = null;
2981 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002982}
2983{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002984 (key = Identifier() | key = StringLiteral())
2985 <EQ>
2986 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002987 {
2988 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002989 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002990 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002991 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002992 }
2993 }
2994 )
2995 {
2996 return new Pair<String, String>(key.toUpperCase(), value);
2997 }
2998}
2999
Glenn58329202021-04-09 12:03:46 -07003000IndexedTypeExpression IndexedTypeExpr(boolean allowQues) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003001{
3002 TypeExpression typeExpr = null;
Glenn58329202021-04-09 12:03:46 -07003003 boolean isUnknownable = !allowQues;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003004}
3005{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003006 typeExpr = TypeExpr(false)
Glenn58329202021-04-09 12:03:46 -07003007 ( <QUES>
3008 {
3009 if (!allowQues) {
3010 throw new SqlppParseException(getSourceLocation(token), "'?' quantifier is illegal for the type expression.");
3011 }
3012 isUnknownable = true;
3013 }
3014 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003015 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003016 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003017 }
3018}
3019
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003020TypeExpression TypeExpr(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003021{
3022 TypeExpression typeExpr = null;
3023}
3024{
3025 (
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003026 typeExpr = TypeReference()
3027 | typeExpr = OrderedListTypeDef(allowRecordTypeDef)
3028 | typeExpr = UnorderedListTypeDef(allowRecordTypeDef)
3029 | typeExpr = RecordTypeDef() {
3030 if (!allowRecordTypeDef) {
3031 throw new SqlppParseException(typeExpr.getSourceLocation(), "Unexpected record type declaration");
3032 }
3033 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003034 )
3035 {
3036 return typeExpr;
3037 }
3038}
3039
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07003040RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
3041{
3042 RecordTypeDefinition.RecordKind recordKind = null;
3043}
3044{
3045 (
3046 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
3047 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
3048 )
3049 {
3050 return recordKind;
3051 }
3052}
3053
Yingyi Bu391f09e2015-10-29 13:49:39 -07003054RecordTypeDefinition RecordTypeDef() throws ParseException:
3055{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003056 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003057 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07003058 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003059}
3060{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07003061 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003062 <LEFTBRACE>
3063 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003064 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003065 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
3066 if (hintToken != null) {
3067 String hintParams = hintToken.hintParams;
3068 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
3069 if (splits == null || splits.length != 4) {
3070 throw new SqlppParseException(getSourceLocation(hintToken),
3071 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07003072 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003073 if (!splits[0].equals("int")) {
3074 throw new SqlppParseException(getSourceLocation(hintToken),
3075 "The only supported type for gen-fields is int.");
3076 }
3077 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
3078 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
3079 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003080 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003081 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003082 (
3083 RecordField(recType)
3084 ( <COMMA> RecordField(recType) )*
3085 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003086 <RIGHTBRACE>
3087 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07003088 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003089 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003090 }
3091}
3092
3093void RecordField(RecordTypeDefinition recType) throws ParseException:
3094{
3095 String fieldName;
3096 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07003097 boolean nullable = false, missable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003098}
3099{
3100 fieldName = Identifier()
3101 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003102 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
3103 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
3104 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
3105 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003106 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003107 <COLON> type = TypeExpr(true) ( <QUES> { nullable = true; missable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003108 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07003109 recType.addField(fieldName, type, nullable, missable, rfdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003110 }
3111}
3112
3113TypeReferenceExpression TypeReference() throws ParseException:
3114{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003115 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003116}
3117{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003118 id = QualifiedName()
3119 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003120 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
3121 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003122 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003123
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003124 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003125 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003126 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003127}
3128
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003129OrderedListTypeDefinition OrderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003130{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003131 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003132 TypeExpression type = null;
3133}
3134{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003135 <LEFTBRACKET> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003136 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003137 <RIGHTBRACKET>
3138 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003139 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003140 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003141 }
3142}
3143
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003144UnorderedListTypeDefinition UnorderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003145{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003146 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003147 TypeExpression type = null;
3148}
3149{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003150 <LEFTDBLBRACE> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003151 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003152 <RIGHTDBLBRACE>
3153 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003154 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003155 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003156 }
3157}
3158
3159FunctionName FunctionName() throws ParseException:
3160{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003161 Triple<List<String>, Token, Token> prefix = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003162 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003163}
3164{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003165 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
3166 // that copy must be kept in sync with this code
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003167 prefix = MultipartIdentifierWithHints(
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003168 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 -07003169 SqlppHint.SPATIAL_JOIN_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003170 )
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003171 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003172 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003173 Token startToken = prefix.second;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003174 FunctionName result = new FunctionName();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003175 result.sourceLoc = getSourceLocation(startToken);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06003176 result.hintToken = prefix.third;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003177 List<String> list = prefix.first;
3178 int ln = list.size();
3179 String last = list.get(ln - 1);
3180 if (suffix == null) {
3181 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
3182 // no library name
3183 result.function = last;
3184 } else {
3185 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
3186 // suffix = func_name
3187 result.library = last;
3188 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003189 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003190 if (ln > 1) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003191 result.dataverse = createDataverseName(list, 0, ln - 1, startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003192 } else {
3193 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003194 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003195
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003196 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
3197 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003198 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003199 return result;
3200 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003201}
3202
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003203Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003204{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003205 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003206}
3207{
3208 name = QualifiedName()
3209 {
3210 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003211 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003212 }
3213 return name;
3214 }
3215}
3216
3217String Identifier() throws ParseException:
3218{
3219 String lit = null;
3220}
3221{
3222 (<IDENTIFIER>
3223 {
3224 return token.image;
3225 }
3226 | lit = QuotedString()
3227 {
3228 return lit;
3229 }
3230 )
3231}
3232
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003233List<String> ParenthesizedIdentifierList() throws ParseException:
3234{
3235 List<String> list = new ArrayList<String>();
3236 String ident = null;
3237}
3238{
3239 <LEFTPAREN>
3240 ident = Identifier() { list.add(ident); }
3241 ( <COMMA> ident = Identifier() { list.add(ident); } )*
3242 <RIGHTPAREN>
3243 {
3244 return list;
3245 }
3246}
3247
Vijay Sarathy691e93b2022-11-20 14:31:47 -08003248Pair<HashJoinExpressionAnnotation.BuildOrProbe, String> buildOrProbeParenthesizedIdentifier() throws ParseException:
3249{
3250 String ident1 = null;
3251 String ident2 = null;
3252}
3253{
3254 ident1 = Identifier() <LEFTPAREN> ident2 = Identifier() <RIGHTPAREN>
3255 {
3256 // check
3257 if (ident1.equals("build")) {
3258 return new Pair<HashJoinExpressionAnnotation.BuildOrProbe, String>(HashJoinExpressionAnnotation.BuildOrProbe.BUILD, ident2);
3259 }
3260 else if (ident1.equals("probe")) {
3261 return new Pair<HashJoinExpressionAnnotation.BuildOrProbe, String>(HashJoinExpressionAnnotation.BuildOrProbe.PROBE, ident2);
3262 }
3263 else {
3264 throw new SqlppParseException(getSourceLocation(token), "The string after hashjoin has to be \"build\" or \"probe\".");
3265 }
3266 return null;
3267 }
3268}
3269
3270String parenthesizedIdentifier() throws ParseException:
3271{
3272 String ident = null;
3273}
3274{
3275 <LEFTPAREN> ident = Identifier() <RIGHTPAREN>
3276 {
3277 return ident;
3278 }
3279}
3280
Tin Vu4f1090d2021-09-21 02:06:32 -07003281List<Literal> ParenthesizedLiteralList() throws ParseException:
3282{
3283 List<Literal> list = new ArrayList<Literal>();
3284 boolean minus = false;
3285 Expression litExpr = null;
3286 Literal lit = null;
3287}
3288{
3289 <LEFTPAREN>
3290 (<MINUS> { minus = true; })? litExpr = Literal()
3291 {
3292 lit = ((LiteralExpr) litExpr).getValue();
3293 if (minus)
3294 {
3295 try {
3296 lit = ExpressionUtils.reverseSign(lit);
3297 } catch (TypeMismatchException e) {
3298 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
3299 }
3300 minus = false;
3301 }
3302 list.add(lit);
3303 }
3304 ( <COMMA> (<MINUS> { minus = true; })? litExpr = Literal()
3305 {
3306 lit = ((LiteralExpr) litExpr).getValue();
3307 if (minus)
3308 {
3309 try {
3310 lit = ExpressionUtils.reverseSign(lit);
3311 } catch (TypeMismatchException e) {
3312 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
3313 }
3314 minus = false;
3315 }
3316 list.add(lit);
3317 } )*
3318 <RIGHTPAREN>
3319 {
3320 return list;
3321 }
3322}
3323
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003324Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003325{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003326 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003327 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003328}
3329{
3330 fieldList = NestedField()
Glenn58329202021-04-09 12:03:46 -07003331 ( <COLON> fieldType = IndexedTypeExpr(true) )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07003332 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07003333 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
3334 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07003335 }
3336}
3337
Yingyi Buc9bfe252016-03-01 00:02:40 -08003338Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003339{
3340 List<String> exprList = new ArrayList<String>();
3341 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003342 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003343 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003344}
3345{
3346 lit = Identifier()
3347 {
Yingyi Bub9169b62016-02-26 21:21:49 -08003348 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003349 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08003350 }
3351 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08003352 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08003353 <LEFTPAREN><RIGHTPAREN>
3354 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003355 if(!lit.equalsIgnoreCase("meta")){
3356 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08003357 }
3358 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003359 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08003360 }
3361 )?
3362 {
3363 if(!meetParens){
3364 exprList.add(lit);
3365 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003366 }
3367 (<DOT>
3368 lit = Identifier()
3369 {
3370 exprList.add(lit);
3371 }
3372 )*
3373 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08003374 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003375 }
3376}
3377
Yingyi Bu6d57e492016-06-06 21:24:42 -07003378String ConstantString() throws ParseException:
3379{
3380 String value = null;
3381}
3382{
3383 (value = QuotedString() | value = StringLiteral())
3384 {
3385 return value;
3386 }
3387}
3388
Yingyi Bu391f09e2015-10-29 13:49:39 -07003389String QuotedString() throws ParseException:
3390{
3391}
3392{
3393 <QUOTED_STRING>
3394 {
3395 return removeQuotesAndEscapes(token.image);
3396 }
3397}
3398
Yingyi Bu391f09e2015-10-29 13:49:39 -07003399String StringLiteral() throws ParseException:
3400{
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003401 StringBuilder str = null;
3402 char quote = 0;
3403 boolean ext = false;
3404 Token litToken = null;
3405 String lit = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003406}
3407{
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003408 ( <STRING_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07003409 {
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003410 String quoted = token.image;
3411 char q = quoted.charAt(0);
3412 boolean e = q == 'E';
3413 if (e) {
3414 q = quoted.charAt(1);
3415 quoted = quoted.substring(1);
3416 }
3417 if (lit == null) {
3418 quote = q;
3419 ext = e;
3420 } else {
3421 boolean isAdjacent = litToken.endLine == token.beginLine && litToken.endColumn + 1 == token.beginColumn;
3422 if (!isAdjacent || ext || e || (q != quote)) {
3423 throw new SqlppParseException(getSourceLocation(token), "Invalid string literal");
3424 }
3425 if (str == null) {
3426 str = new StringBuilder();
3427 }
3428 str.append(lit);
3429 str.append(quote);
3430 }
3431 lit = removeQuotesAndEscapes(quoted);
3432 litToken = token;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003433 }
Dmitry Lychagin984bc522021-10-13 16:22:25 -07003434 )+
3435 {
3436 if (str == null) {
3437 return lit;
3438 } else {
3439 str.append(lit);
3440 return str.toString();
3441 }
3442 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003443}
3444
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003445Triple<List<String>, Token, Token> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003446{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003447 Triple<List<String>, Token, Token> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003448}
3449{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003450 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003451 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003452 return result;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003453 }
3454}
3455
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003456Triple<List<String>, Token, Token> MultipartIdentifierWithHints(SqlppHint... expectedHints)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003457 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003458{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003459 List<String> list = new ArrayList<String>();
3460 SourceLocation sourceLoc = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003461 Token startToken, hintToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003462 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003463}
3464{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003465 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003466 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003467 list.add(item);
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003468 startToken = token;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003469 if (expectedHints != null && expectedHints.length > 0) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003470 hintToken = fetchHint(token, expectedHints);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003471 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003472 }
3473 (<DOT> item = Identifier() { list.add(item); } )*
3474 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003475 return new Triple<List<String>, Token, Token>(list, startToken, hintToken);
3476 }
3477}
3478
3479DataverseName DataverseName() throws ParseException:
3480{
3481 Triple<List<String>, Token, Token> ident = null;
3482}
3483{
3484 ident = MultipartIdentifier()
3485 {
3486 List<String> list = ident.first;
3487 Token startToken = ident.second;
3488 return createDataverseName(list, 0, list.size(), startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003489 }
3490}
3491
3492Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
3493{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003494 Triple<List<String>, Token, Token> ident = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003495}
3496{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003497 ident = MultipartIdentifier()
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003498 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003499 List<String> list = ident.first;
3500 Token startToken = ident.second;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003501 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003502 DataverseName id1 = len > 1 ? createDataverseName(list, 0, len - 1, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003503 Identifier id2 = new Identifier(list.get(len - 1));
3504 return new Pair<DataverseName,Identifier>(id1, id2);
3505 }
3506}
3507
3508Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
3509{
3510 List<String> list = new ArrayList<String>();
3511 String item = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003512 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003513}
3514{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003515 item = Identifier() { list.add(item); startToken = token; }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003516 (<DOT> item = Identifier() { list.add(item); } )+
3517 {
3518 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003519 DataverseName id1 = len > 2 ? createDataverseName(list, 0, len - 2, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003520 Identifier id2 = new Identifier(list.get(len - 2));
3521 Identifier id3 = new Identifier(list.get(len - 1));
3522 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003523 }
3524}
3525
3526FunctionDecl FunctionDeclaration() throws ParseException:
3527{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003528 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003529 String functionName;
Dmitry Lychagin541652082020-11-09 14:04:53 -08003530 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003531 Expression funcBody;
3532 createNewScope();
3533}
3534{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003535 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07003536 functionName = Identifier()
Dmitry Lychagin541652082020-11-09 14:04:53 -08003537 paramsWithArity = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07003538 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003539 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07003540 <RIGHTBRACE>
3541 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08003542 int arity = paramsWithArity.first;
3543 List<Pair<VarIdentifier,TypeExpression>> paramList = paramsWithArity.second;
3544 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, arity);
Xikui Wang3de700a2018-03-15 16:32:55 -07003545 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003546 ensureNoTypeDeclsInFunction(functionName, paramList, null, startToken);
3547 List<VarIdentifier> params = new ArrayList<VarIdentifier>(paramList.size());
3548 for (Pair<VarIdentifier,TypeExpression> p: paramList) {
3549 params.add(p.getFirst());
Ian Maxon38fe9402020-01-29 19:27:40 -08003550 }
Dmitry Lychagin9ba74872021-04-05 14:38:20 -07003551 FunctionDecl stmt = new FunctionDecl(signature, params, funcBody, false);
Xikui Wang3de700a2018-03-15 16:32:55 -07003552 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003553 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07003554 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003555}
3556
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003557Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003558{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003559 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003560 Expression expr;
3561}
3562{
3563 (
3564 expr = Expression()
3565 |
3566 expr = SelectExpression(false)
3567 )
3568 {
3569 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003570 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003571 return query;
3572 }
3573}
3574
3575
Yingyi Bu391f09e2015-10-29 13:49:39 -07003576Expression Expression():
3577{
3578 Expression expr = null;
3579 Expression exprP = null;
3580}
3581{
3582(
3583 LOOKAHEAD(2)
3584 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003585 | expr = QuantifiedExpression()
3586)
3587 {
3588 return (exprP==null) ? expr : exprP;
3589 }
3590}
3591
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003592Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003593{
3594 OperatorExpr op = null;
3595 Expression operand = null;
3596}
3597{
3598 operand = AndExpr()
3599 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003600 <OR>
3601 {
3602 if (op == null) {
3603 op = new OperatorExpr();
3604 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07003605 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003606 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003607 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003608 try{
3609 op.addOperator(token.image.toLowerCase());
3610 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003611 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003612 }
Xikui Wang3de700a2018-03-15 16:32:55 -07003613 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003614
Xikui Wang3de700a2018-03-15 16:32:55 -07003615 operand = AndExpr()
3616 {
3617 op.addOperand(operand);
3618 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003619
3620 )*
3621
3622 {
3623 return op==null? operand: op;
3624 }
3625}
3626
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003627Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003628{
3629 OperatorExpr op = null;
3630 Expression operand = null;
3631}
3632{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003633 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003634 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003635 <AND>
3636 {
3637 if (op == null) {
3638 op = new OperatorExpr();
3639 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003640 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003641 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003642 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003643 try{
3644 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003645 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003646 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003647 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003648 }
3649
Yingyi Bu196db5d2016-07-15 19:07:20 -07003650 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003651 {
3652 op.addOperand(operand);
3653 }
3654
3655 )*
3656
3657 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003658 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003659 }
3660}
3661
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003662Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07003663{
3664 Expression inputExpr;
3665 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003666 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07003667}
3668{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003669 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07003670 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003671 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003672 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003673 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003674 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003675 } else {
3676 return inputExpr;
3677 }
3678 }
3679}
Yingyi Bu391f09e2015-10-29 13:49:39 -07003680
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003681Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003682{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003683 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003684 OperatorExpr op = null;
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003685 Token opToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003686 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003687 IExpressionAnnotation annotation = null;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003688 List<IExpressionAnnotation> annotationList = new ArrayList<IExpressionAnnotation>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003689}
3690{
Yingyi Bu6c638342016-09-02 17:54:34 -07003691 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003692
3693 (
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003694 LOOKAHEAD(3)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN> |
3695 <IS> (<NOT> { not = true; })? <DISTINCT> { opToken = token; } <FROM> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003696 {
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003697 if (opToken == null) {
3698 opToken = token;
3699 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003700 Token hintToken = fetchHint(token,
3701 SqlppHint.HASH_BROADCAST_JOIN_HINT, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003702 SqlppHint.HASH_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT,
3703 SqlppHint.SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT, SqlppHint.JOIN_PREDICATE_PRODUCTIVITY_HINT
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003704 );
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003705 while (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003706 annotation = parseExpressionAnnotation(hintToken);
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003707 annotationList.add(annotation);
3708 hintToken = hintToken.specialToken;
Vijay Sarathy8f4fccd2022-12-05 14:29:02 -08003709 if (hintToken != null) {
3710 SourceLocation sourceLoc = getSourceLocation(hintToken);
3711 hintCollector.remove(sourceLoc);
3712 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003713 }
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003714 String operator = opToken.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07003715 if (operator.equals("<>")){
3716 operator = "!=";
3717 }
3718 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003719 operator = "not_" + operator;
3720 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003721 if (op == null) {
3722 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07003723 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003724 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003725 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003726 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003727 try{
3728 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003729 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003730 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003731 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003732 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003733
Yingyi Bu6c638342016-09-02 17:54:34 -07003734 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07003735 {
Shiva2ea73232019-10-09 18:04:05 -07003736 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07003737 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003738 )?
3739
3740 {
3741 if (annotation != null) {
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003742 op.addHints(annotationList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003743 }
3744 return op==null? operand: op;
3745 }
3746}
3747
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003748Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07003749{
3750 boolean not = false;
3751 OperatorExpr op = null;
3752 Expression operand = null;
3753 IExpressionAnnotation annotation = null;
3754}
3755{
3756 operand = IsExpr()
3757 (
3758 LOOKAHEAD(2)
3759 (<NOT> { not = true; })? <BETWEEN>
3760 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003761 Token hintToken = fetchHint(token,
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003762 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.HASH_JOIN_HINT,
3763 SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003764 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003765 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003766 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu6c638342016-09-02 17:54:34 -07003767 }
3768 String operator = token.image.toLowerCase();
3769 if(not){
3770 operator = "not_" + operator;
3771 }
3772 if (op == null) {
3773 op = new OperatorExpr();
3774 op.addOperand(operand);
3775 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003776 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07003777 }
3778 try{
3779 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003780 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003781 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07003782 }
3783 }
3784
3785 operand = IsExpr()
3786 {
3787 op.addOperand(operand);
3788 }
3789
3790 <AND>
3791 operand = IsExpr()
3792 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08003793 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07003794 op.addOperand(operand);
3795 }
3796 )?
3797
3798 {
3799 if (annotation != null) {
3800 op.addHint(annotation);
3801 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003802 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07003803 }
3804}
3805
Yingyi Budaa549c2016-06-28 22:30:52 -07003806Expression IsExpr() throws ParseException:
3807{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003808 Token notToken = null;
3809 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003810 Expression operand = null;
3811 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003812 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003813}
3814{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003815 operand = LikeExpr()
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003816 (
3817 LOOKAHEAD(3)
3818 <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003819 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003820 (
3821 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
3822 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08003823 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07003824 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003825 )
Yingyi Budaa549c2016-06-28 22:30:52 -07003826 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003827 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07003828 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003829 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003830 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003831 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07003832 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003833 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07003834 }
3835 }
3836 )?
3837 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003838 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07003839 }
3840}
3841
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003842Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003843{
3844 boolean not = false;
3845 OperatorExpr op = null;
3846 Expression operand = null;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003847 IExpressionAnnotation annotation = null;
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003848}
3849{
3850 operand = ConcatExpr()
3851 (
3852 LOOKAHEAD(2)
3853 (<NOT> { not = true; })? <LIKE>
3854 {
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003855 Token hintToken = fetchHint(token, SqlppHint.SINGLE_DATASET_PREDICATE_SELECTIVITY_HINT);
3856 if (hintToken != null) {
3857 annotation = parseExpressionAnnotation(hintToken);
3858 }
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003859 op = new OperatorExpr();
3860 op.addOperand(operand);
3861 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003862 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003863
3864 String operator = token.image.toLowerCase();
3865 if (not) {
3866 operator = "not_" + operator;
3867 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003868 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003869 op.addOperator(operator);
3870 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003871 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003872 }
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07003873 if (annotation != null) {
3874 op.addHint(annotation);
3875 }
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003876 }
3877
3878 operand = ConcatExpr()
3879 {
3880 op.addOperand(operand);
3881 }
3882 )?
3883
3884 {
3885 return op == null ? operand : op;
3886 }
3887}
3888
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003889Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003890{
3891 OperatorExpr op = null;
3892 Expression operand = null;
3893}
3894{
3895 operand = AddExpr()
3896 (
3897 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003898 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003899 {
3900 if (op == null) {
3901 op = new OperatorExpr();
3902 op.addOperand(operand);
3903 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003904 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003905 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003906 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003907 }
3908 operand = AddExpr()
3909 {
3910 op.addOperand(operand);
3911 }
3912 )*
3913
3914 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003915 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003916 }
3917}
Yingyi Budaa549c2016-06-28 22:30:52 -07003918
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003919Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003920{
3921 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003922 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003923 Expression operand = null;
3924}
3925{
3926 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07003927 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003928 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003929 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003930 {
3931 if (op == null) {
3932 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003933 op.addOperand(operand);
3934 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003935 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003936 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003937 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003938 }
3939
3940 operand = MultExpr()
3941 {
3942 op.addOperand(operand);
3943 }
3944 )*
3945
3946 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003947 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003948 }
3949}
3950
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003951Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003952{
3953 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003954 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003955 Expression operand = null;
3956}
3957{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003958 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003959
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003960 ( (
3961 <MUL> { opType = OperatorType.MUL; } |
3962 <DIVIDE> { opType = OperatorType.DIVIDE; } |
3963 <DIV> { opType = OperatorType.DIV; } |
3964 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
3965 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003966 {
3967 if (op == null) {
3968 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003969 op.addOperand(operand);
3970 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003971 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003972 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003973 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003974 }
3975 operand = ExponentExpr()
3976 {
3977 op.addOperand(operand);
3978 }
3979 )*
3980
3981 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003982 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003983 }
3984}
3985
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003986Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003987{
3988 OperatorExpr op = null;
3989 Expression operand = null;
3990}
3991{
3992 operand = UnaryExpr()
3993 (<CARET>
3994 {
3995 if (op == null) {
3996 op = new OperatorExpr();
3997 op.addOperand(operand);
3998 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003999 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004000 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004001 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004002 }
4003 operand = UnaryExpr()
4004 {
4005 op.addOperand(operand);
4006 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004007 )?
4008 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004009 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07004010 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004011}
4012
4013Expression UnaryExpr() throws ParseException:
4014{
Yingyi Bu196db5d2016-07-15 19:07:20 -07004015 boolean not = false;
4016 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004017 Expression expr = null;
4018}
Yingyi Budaa549c2016-06-28 22:30:52 -07004019{
Yingyi Bu196db5d2016-07-15 19:07:20 -07004020 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004021 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07004022 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004023 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07004024 exprType = "not_" + exprType;
4025 }
4026 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004027 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004028 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07004029 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08004030 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004031 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07004032 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004033 }
4034 )?
4035
4036 expr = ValueExpr()
4037 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004038 if (uexpr == null) {
4039 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004040 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004041 uexpr.setExpr(expr);
4042 return uexpr;
4043 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004044 }
4045}
4046
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004047Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004048{
4049 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004050 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004051}
4052{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03004053 expr = PrimaryExpr()
4054 (
4055 accessor = FieldAccessor(accessor != null ? accessor : expr)
4056 |
4057 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004058 )*
4059 {
4060 return accessor == null ? expr : accessor;
4061 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004062}
4063
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004064FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004065{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004066 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004067 String ident = null;
4068}
4069{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004070 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004071 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004072 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004073 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004074 }
4075}
4076
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03004077AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004078{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004079 Token startToken = null;
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004080 boolean star = false, slice = false;
4081 Expression expr1 = null, expr2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004082}
4083{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004084 <LEFTBRACKET> { startToken = token; }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004085 (
4086 <MUL> { star = true; }
4087 |
4088 ( expr1 = Expression() ( <COLON> { slice = true; } ( expr2 = Expression() )? )? )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004089 )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004090 <RIGHTBRACKET>
4091 {
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004092 if (expr1 != null && expr1.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
4093 ensureIntegerLiteral( (LiteralExpr) expr1, "Index");
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03004094 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07004095 if (expr2 != null && expr2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
4096 ensureIntegerLiteral( (LiteralExpr) expr2, "Index");
4097 }
4098 AbstractAccessor resultAccessor;
4099 if (slice) {
4100 resultAccessor = new ListSliceExpression(inputExpr, expr1, expr2);
4101 } else if (star) {
4102 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.STAR, null);
4103 } else {
4104 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.ELEMENT, expr1);
4105 }
4106 return addSourceLocation(resultAccessor, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004107 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004108}
4109
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004110Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004111{
4112 Expression expr = null;
4113}
4114{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08004115 (
4116 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07004117 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004118 | expr = Literal()
4119 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004120 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004121 | expr = ListConstructor()
4122 | expr = RecordConstructor()
4123 | expr = ParenthesizedExpression()
4124 )
4125 {
4126 return expr;
4127 }
4128}
4129
4130Expression Literal() throws ParseException:
4131{
4132 LiteralExpr lit = new LiteralExpr();
4133 String str = null;
4134}
4135{
4136 ( str = StringLiteral()
4137 {
4138 lit.setValue(new StringLiteral(str));
4139 }
4140 | <INTEGER_LITERAL>
4141 {
Till Westmann68c6a992016-09-30 00:19:12 -07004142 try {
4143 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
4144 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08004145 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07004146 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004147 }
4148 | <FLOAT_LITERAL>
4149 {
Till Westmann68c6a992016-09-30 00:19:12 -07004150 try {
4151 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
4152 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08004153 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07004154 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004155 }
4156 | <DOUBLE_LITERAL>
4157 {
Till Westmann68c6a992016-09-30 00:19:12 -07004158 try {
4159 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
4160 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08004161 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07004162 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004163 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07004164 | <MISSING>
4165 {
4166 lit.setValue(MissingLiteral.INSTANCE);
4167 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004168 | <NULL>
4169 {
4170 lit.setValue(NullLiteral.INSTANCE);
4171 }
4172 | <TRUE>
4173 {
4174 lit.setValue(TrueLiteral.INSTANCE);
4175 }
4176 | <FALSE>
4177 {
4178 lit.setValue(FalseLiteral.INSTANCE);
4179 }
4180 )
4181 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004182 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004183 }
4184}
4185
Yingyi Bu391f09e2015-10-29 13:49:39 -07004186VariableExpr VariableRef() throws ParseException:
4187{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004188 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004189}
4190{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08004191 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004192 {
4193 Identifier ident = lookupSymbol(id);
4194 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004195 throw new SqlppParseException(getSourceLocation(token),
4196 "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 -07004197 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004198 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004199 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004200 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004201 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004202 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07004203 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004204 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004205 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004206 }
4207}
4208
Yingyi Bu391f09e2015-10-29 13:49:39 -07004209VariableExpr Variable() throws ParseException:
4210{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004211 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004212}
4213{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08004214 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004215 {
4216 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07004217 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
4218 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07004219 varExp.setIsNewVar(false);
4220 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004221 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004222 }
4223}
4224
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08004225String VariableIdentifier() throws ParseException:
4226{
4227 String id = null;
4228}
4229{
4230 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
4231 {
4232 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
4233 }
4234}
4235
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004236Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
4237{
4238 VariableExpr var = null;
4239 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
4240}
4241{
4242 var = Variable()
4243 ( LOOKAHEAD(1)
4244 {
4245 VariableExpr fieldVarExpr = null;
4246 String fieldIdentifierStr = null;
4247 }
4248 <LEFTPAREN>
4249 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
4250 {
4251 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
4252 }
4253 (<COMMA>
4254 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
4255 {
4256 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
4257 }
4258 )*
4259 <RIGHTPAREN>
4260 )?
4261 {
4262 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
4263 }
4264}
4265
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004266VariableExpr ExternalVariableRef() throws ParseException:
4267{
4268 String name = null;
4269}
4270{
4271 (
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08004272 <DOLLAR_IDENTIFIER> { name = token.image.substring(1); }
4273 | <DOLLAR_INTEGER_LITERAL> { name = token.image.substring(1); }
4274 | <DOLLAR_QUOTED_STRING> { name = removeQuotesAndEscapes(token.image.substring(1)); }
4275 | <QUES> { name = String.valueOf(++externalVarCounter); }
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07004276 )
4277 {
4278 String idName = SqlppVariableUtil.toExternalVariableName(name);
4279 VarIdentifier id = new VarIdentifier(idName);
4280 VariableExpr varExp = new VariableExpr(id);
4281 return addSourceLocation(varExp, token);
4282 }
4283}
4284
Yingyi Bu391f09e2015-10-29 13:49:39 -07004285Expression ListConstructor() throws ParseException:
4286{
4287 Expression expr = null;
4288}
4289{
4290 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004291 expr = OrderedListConstructor() |
4292 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004293 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004294 {
4295 return expr;
4296 }
4297}
4298
Yingyi Bu391f09e2015-10-29 13:49:39 -07004299ListConstructor OrderedListConstructor() throws ParseException:
4300{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004301 Token startToken = null;
4302 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004303}
4304{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004305 <LEFTBRACKET> { startToken = token; }
4306 exprList = ExpressionList()
4307 <RIGHTBRACKET>
4308 {
4309 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004310 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004311 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004312}
4313
4314ListConstructor UnorderedListConstructor() throws ParseException:
4315{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004316 Token startToken = null;
4317 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004318}
4319{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004320 <LEFTDBLBRACE> { startToken = token; }
4321 exprList = ExpressionList()
4322 <RIGHTDBLBRACE>
4323 {
4324 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004325 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004326 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004327}
4328
4329List<Expression> ExpressionList() throws ParseException:
4330{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004331 Expression expr = null;
4332 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004333}
4334{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004335 (
4336 expr = Expression()
4337 {
4338 exprList.add(expr);
4339 }
4340 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07004341 {
4342 exprList.add(expr);
4343 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004344 )*
4345 )?
4346 {
4347 return exprList;
4348 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004349}
4350
Yingyi Bu391f09e2015-10-29 13:49:39 -07004351RecordConstructor RecordConstructor() throws ParseException:
4352{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004353 Token startToken = null;
4354 FieldBinding fb = null;
4355 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004356}
4357{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004358 <LEFTBRACE> { startToken = token; }
4359 (
4360 fb = FieldBinding() { fbList.add(fb); }
4361 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
4362 )?
4363 <RIGHTBRACE>
4364 {
4365 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004366 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004367 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004368}
4369
4370FieldBinding FieldBinding() throws ParseException:
4371{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004372 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004373}
4374{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004375 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004376 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004377 if (right == null) {
4378 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
4379 if (generatedIdentifier == null) {
4380 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
4381 }
4382 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07004383 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
4384 generatedNameExpr.setSourceLocation(left.getSourceLocation());
4385 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07004386 } else {
4387 return new FieldBinding(left, right);
4388 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004389 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004390}
4391
Yingyi Bu391f09e2015-10-29 13:49:39 -07004392Expression FunctionCallExpr() throws ParseException:
4393{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004394 List<Expression> argList = new ArrayList<Expression>();
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004395 Expression argExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004396 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07004397 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07004398 boolean distinct = false;
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004399 Expression filterExpr = null;
4400 WindowExpression windowExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004401}
4402{
4403 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07004404 <LEFTPAREN> (
4405 ( <DISTINCT> { distinct = true; } )?
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004406 ( argExpr = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004407 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004408 if (star) {
4409 if (funcName.function.equalsIgnoreCase(BuiltinFunctions.SCALAR_COUNT.getName())) {
4410 argExpr = new LiteralExpr(new LongIntegerLiteral(1L));
4411 } else {
4412 throw new SqlppParseException(getSourceLocation(token),
4413 "The parameter * can only be used in " + BuiltinFunctions.SCALAR_COUNT.getName() + "().");
Yingyi Buf4d09842016-08-26 00:03:52 -07004414 }
Yingyi Buf4d09842016-08-26 00:03:52 -07004415 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004416 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004417 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004418 (<COMMA> argExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004419 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004420 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004421 }
4422 )*)? <RIGHTPAREN>
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004423
4424 {
4425 String name = funcName.function;
4426 if (distinct) {
4427 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
4428 }
4429 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
4430 int arity = argList.size();
4431 FunctionSignature signature = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
4432 if (signature == null) {
4433 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
4434 }
4435 }
4436
4437 ( <FILTER> <LEFTPAREN> <WHERE> filterExpr = Expression() <RIGHTPAREN> )?
4438
4439 ( LOOKAHEAD(5) windowExpr = WindowExpr(signature, argList, filterExpr) )?
4440
4441 {
4442 if (windowExpr != null) {
4443 return windowExpr;
4444 } else {
4445 CallExpr callExpr = new CallExpr(signature, argList, filterExpr);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06004446 if (funcName.hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08004447 IExpressionAnnotation annotation = parseExpressionAnnotation(funcName.hintToken);
4448 if (annotation != null) {
4449 callExpr.addHint(annotation);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004450 }
4451 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004452 FunctionMapUtil.normalizedListInputFunctions(callExpr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004453 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004454 return callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004455 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004456 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004457}
4458
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004459WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr)
4460 throws ParseException:
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004461{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004462 WindowExpression windowExpr = null;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004463 Boolean fromLast = null, ignoreNulls = null;
4464}
4465{
4466 (
4467 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
4468 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
4469 <FROM> <IDENTIFIER>
4470 {
4471 if (isToken(FIRST)) {
4472 fromLast = false;
4473 } else if (isToken(LAST)) {
4474 fromLast = true;
4475 } else {
4476 throw createUnexpectedTokenError();
4477 }
4478 }
4479 )?
4480 (
4481 // ( RESPECT | IGNORE ) NULLS OVER
4482 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
4483 <IDENTIFIER>
4484 {
4485 if (isToken(RESPECT)) {
4486 ignoreNulls = false;
4487 } else if (isToken(IGNORE)) {
4488 ignoreNulls = true;
4489 } else {
4490 throw createUnexpectedTokenError();
4491 }
4492 }
4493 <IDENTIFIER>
4494 {
4495 if (!isToken(NULLS)) {
4496 throw createUnexpectedTokenError();
4497 }
4498 }
4499 )?
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004500 <OVER> windowExpr = OverClause(signature, argList, aggFilterExpr, token, fromLast, ignoreNulls)
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004501 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004502 return windowExpr;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004503 }
4504}
4505
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004506WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr,
4507 Token startToken, Boolean fromLast, Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004508{
4509 Expression partitionExpr = null;
4510 List<Expression> partitionExprs = new ArrayList<Expression>();
4511 OrderbyClause orderByClause = null;
4512 List<Expression> orderbyList = null;
4513 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004514 List<OrderbyClause.NullOrderModifier> orderbyNullModifierList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004515 WindowExpression.FrameMode frameMode = null;
4516 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
4517 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
4518 Expression frameStartExpr = null, frameEndExpr = null;
4519 WindowExpression.FrameExclusionKind frameExclusionKind = null;
4520 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
4521 VariableExpr windowVar = null;
4522 List<Pair<Expression, Identifier>> windowFieldList = null;
4523}
4524{
4525 (
4526 windowVarWithFieldList = VariableWithFieldMap() <AS>
4527 {
4528 windowVar = windowVarWithFieldList.first;
4529 windowFieldList = windowVarWithFieldList.second;
4530 }
4531 )?
4532 <LEFTPAREN>
4533 (
4534 <IDENTIFIER> { expectToken(PARTITION); } <BY>
4535 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
4536 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
4537 )?
4538 (
4539 orderByClause = OrderbyClause()
4540 {
4541 orderbyList = orderByClause.getOrderbyList();
4542 orderbyModifierList = orderByClause.getModifierList();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004543 orderbyNullModifierList = orderByClause.getNullModifierList();
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004544 }
4545 (
4546 frameMode = WindowFrameMode()
4547 (
4548 frameStart = WindowFrameBoundary() |
4549 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
4550 )
4551 ( frameExclusionKind = WindowFrameExclusion() )?
4552 {
4553 frameStartKind = frameStart.first;
4554 frameStartExpr = frameStart.second;
4555 if (frameEnd == null) {
4556 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4557 } else {
4558 frameEndKind = frameEnd.first;
4559 frameEndExpr = frameEnd.second;
4560 }
4561 if (frameExclusionKind == null) {
4562 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
4563 }
4564 }
4565 )?
4566 )?
4567 <RIGHTPAREN>
4568 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004569 WindowExpression winExpr = new WindowExpression(signature, argList, aggFilterExpr, partitionExprs, orderbyList,
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004570 orderbyModifierList, orderbyNullModifierList, frameMode, frameStartKind, frameStartExpr, frameEndKind,
4571 frameEndExpr, frameExclusionKind, windowVar, windowFieldList, ignoreNulls, fromLast);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004572 return addSourceLocation(winExpr, startToken);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004573 }
4574}
4575
4576WindowExpression.FrameMode WindowFrameMode() throws ParseException:
4577{
4578}
4579{
4580 <IDENTIFIER>
4581 {
4582 if (isToken(RANGE)) {
4583 return WindowExpression.FrameMode.RANGE;
4584 } else if (isToken(ROWS)) {
4585 return WindowExpression.FrameMode.ROWS;
4586 } else if (isToken(GROUPS)) {
4587 return WindowExpression.FrameMode.GROUPS;
4588 } else {
4589 throw createUnexpectedTokenError();
4590 }
4591 }
4592}
4593
4594Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
4595{
4596 boolean current = false;
4597 Expression expr = null;
4598}
4599{
4600 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004601 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004602 | expr = Expression()
4603 )
4604 <IDENTIFIER>
4605 {
4606 WindowExpression.FrameBoundaryKind kind;
4607 if (current && isToken(ROW)) {
4608 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4609 } else if (!current && isToken(PRECEDING)) {
4610 kind = expr == null
4611 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
4612 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
4613 } else if (!current && isToken(FOLLOWING)) {
4614 kind = expr == null
4615 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
4616 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
4617 } else {
4618 throw createUnexpectedTokenError();
4619 }
4620 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
4621 }
4622}
4623
4624WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
4625{
4626 boolean current = false, no = false;
4627}
4628{
4629 <IDENTIFIER>
4630 {
4631 expectToken(EXCLUDE);
4632 }
4633 (
4634 <GROUP>
4635 {
4636 return WindowExpression.FrameExclusionKind.GROUP;
4637 }
4638 |
4639 (
4640 <IDENTIFIER>
4641 {
4642 if (isToken(TIES)) {
4643 return WindowExpression.FrameExclusionKind.TIES;
4644 } else if (isToken(CURRENT)) {
4645 current = true;
4646 } else if (isToken(NO)) {
4647 no = true;
4648 } else {
4649 throw createUnexpectedTokenError();
4650 }
4651 }
4652 <IDENTIFIER>
4653 {
4654 if (current && isToken(ROW)) {
4655 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
4656 } else if (no && isToken(OTHERS)) {
4657 return WindowExpression.FrameExclusionKind.NO_OTHERS;
4658 } else {
4659 throw createUnexpectedTokenError();
4660 }
4661 }
4662 )
4663 )
4664}
4665
Yingyi Bu391f09e2015-10-29 13:49:39 -07004666Expression ParenthesizedExpression() throws ParseException:
4667{
Dmitry Lychagine9630592021-08-30 15:29:30 -07004668 Token startToken = null;
4669 Expression expr = null, expr2 = null;
4670 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004671}
4672{
4673 (
4674 LOOKAHEAD(2)
Dmitry Lychagine9630592021-08-30 15:29:30 -07004675 <LEFTPAREN> { startToken = token; } expr = Expression()
4676 (
4677 <COMMA> expr2 = Expression()
4678 {
4679 if (exprList == null) {
4680 exprList = new ArrayList<Expression>();
4681 exprList.add(expr);
4682 }
4683 exprList.add(expr2);
4684 }
4685 )*
4686 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004687 |
4688 expr = Subquery()
4689 )
4690 {
Dmitry Lychagine9630592021-08-30 15:29:30 -07004691 if (exprList != null) {
4692 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
4693 return addSourceLocation(listExpr, startToken);
4694 } else {
4695 return expr;
4696 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004697 }
4698}
4699
Yingyi Buc8c067c2016-07-25 23:37:19 -07004700Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004701{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004702 Token startToken = null;
4703 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004704 List<Expression> whenExprs = new ArrayList<Expression>();
4705 List<Expression> thenExprs = new ArrayList<Expression>();
4706 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004707 Expression whenExpr = null;
4708 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004709}
4710{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004711 <CASE> { startToken = token; }
4712 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07004713 (
4714 <WHEN> whenExpr = Expression()
4715 {
4716 whenExprs.add(whenExpr);
4717 }
4718 <THEN> thenExpr = Expression()
4719 {
4720 thenExprs.add(thenExpr);
4721 }
4722 )*
4723 (<ELSE> elseExpr = Expression() )?
4724 <END>
4725 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004726 if (conditionExpr == null) {
4727 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004728 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004729 }
4730 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004731 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07004732 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004733}
4734
Yingyi Buab817482016-08-19 21:29:31 -07004735SelectExpression SelectExpression(boolean subquery) throws ParseException:
4736{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004737 List<LetClause> letClauses = new ArrayList<LetClause>();
4738 SelectSetOperation selectSetOperation;
4739 OrderbyClause orderbyClause = null;
4740 LimitClause limitClause = null;
4741 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07004742}
4743{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004744 ( letClauses = LetClause() )?
4745 selectSetOperation = SelectSetOperation()
4746 (orderbyClause = OrderbyClause() {})?
4747 (limitClause = LimitClause() {})?
4748 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004749 SelectExpression selectExpr =
4750 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
4751 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
4752 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004753 }
4754}
4755
Yingyi Buab817482016-08-19 21:29:31 -07004756SelectSetOperation SelectSetOperation() throws ParseException:
4757{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004758 SetOperationInput setOperationInputLeft;
4759 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
4760}
4761{
4762 {
4763 SelectBlock selectBlockLeft = null;
4764 SelectExpression subqueryLeft = null;
4765 Expression expr = null;
4766 }
4767 selectBlockLeft = SelectBlock()
4768 {
4769 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
4770 }
4771 (
4772 {
4773 SetOpType opType = SetOpType.UNION;
4774 boolean setSemantics = true;
4775 SelectBlock selectBlockRight = null;
4776 SelectExpression subqueryRight = null;
4777 }
4778 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
4779 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
4780 {
4781 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
4782 }
4783 )*
4784 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004785 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
4786 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
4787 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004788 }
4789}
4790
Yingyi Buab817482016-08-19 21:29:31 -07004791SelectExpression Subquery() throws ParseException:
4792{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004793 SelectExpression selectExpr = null;
4794}
4795{
4796 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
4797 {
4798 return selectExpr;
4799 }
4800}
4801
Yingyi Buab817482016-08-19 21:29:31 -07004802SelectBlock SelectBlock() throws ParseException:
4803{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004804 SelectClause selectClause = null;
4805 FromClause fromClause = null;
4806 List<LetClause> fromLetClauses = null;
4807 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004808 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004809 GroupbyClause groupbyClause = null;
4810 List<LetClause> gbyLetClauses = null;
4811 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004812 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004813 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004814}
4815{
4816 (
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004817 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004818 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004819 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004820 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004821 fromClause = FromClause()
4822 (
4823 fromLetClauses = LetClause()
4824 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004825 (whereClause = WhereClause())?
4826 (
4827 groupbyClause = GroupbyClause()
4828 (
4829 gbyLetClauses = LetClause()
4830 )?
4831 (havingClause = HavingClause())?
4832 )?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004833 )
4834 |
4835 (
4836 fromLetClauses = LetClause()
4837 {
4838 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
4839 SourceLocation sourceLoc = getSourceLocation(token);
4840 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
4841 missingExpr.setSourceLocation(sourceLoc);
4842 List<Expression> list = new ArrayList<Expression>(1);
4843 list.add(missingExpr);
4844 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
4845 listExpr.setSourceLocation(sourceLoc);
4846 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
4847 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
4848 fromVar.setSourceLocation(sourceLoc);
4849 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
4850 fromClause = new FromClause(fromTerms);
4851 }
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004852 (whereClause = WhereClause())?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004853 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004854 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004855 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004856 |
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004857 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004858 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004859 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004860 fromLetClauses = LetClause()
4861 )?
4862 (whereClause = WhereClause())?
4863 (
4864 groupbyClause = GroupbyClause()
4865 (
4866 gbyLetClauses = LetClause()
4867 )?
4868 (havingClause = HavingClause())?
4869 )?
4870 selectClause = SelectClause()
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004871 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004872 )
4873 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004874 if (fromLetClauses != null) {
4875 fromLetWhereClauses.addAll(fromLetClauses);
4876 }
4877 if (whereClause != null) {
4878 fromLetWhereClauses.add(whereClause);
4879 }
4880 if (gbyLetClauses != null) {
4881 gbyLetHavingClauses.addAll(gbyLetClauses);
4882 }
4883 if (havingClause != null) {
4884 gbyLetHavingClauses.add(havingClause);
4885 }
4886 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
4887 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004888 selectBlock.setSourceLocation(startSrcLoc);
4889 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004890 }
4891}
4892
Yingyi Buab817482016-08-19 21:29:31 -07004893SelectClause SelectClause() throws ParseException:
4894{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004895 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004896 SelectRegular selectRegular = null;
4897 SelectElement selectElement = null;
ggalvizo82499442022-10-24 12:46:15 -07004898 List<List<String>> fieldExclusions = new ArrayList<List<String>>();
4899 List<String> nestedField = new ArrayList<String>();
4900 String identifier;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004901 boolean distinct = false;
4902}
4903{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004904 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004905 (
ggalvizo82499442022-10-24 12:46:15 -07004906 (
4907 selectRegular = SelectRegular()
4908 ( LOOKAHEAD({laIdentifier(EXCLUDE)}) <IDENTIFIER>
4909 identifier = Identifier() { nestedField.add(identifier); }
4910 ( <DOT> identifier = Identifier() { nestedField.add(identifier); } )*
4911 {
4912 fieldExclusions.add(nestedField);
4913 nestedField = new ArrayList<String>();
4914 }
4915 ( LOOKAHEAD(1) // Force <COMMA> to be recognized for a nested field in our EXCLUDE list.
4916 <COMMA> identifier = Identifier() { nestedField.add(identifier); }
4917 ( <DOT> identifier = Identifier() { nestedField.add(identifier); } )*
4918 {
4919 fieldExclusions.add(nestedField);
4920 nestedField = new ArrayList<String>();
4921 }
4922 )*
4923 )?
4924 )
Michael Blowd6cf6412016-06-30 02:44:35 -04004925 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07004926 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07004927 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004928 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004929 SourceLocation sourceLoc = getSourceLocation(startToken);
4930 if (selectRegular == null && selectElement == null){
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004931 Projection projection = new Projection(Projection.Kind.STAR, null, null);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004932 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004933 List<Projection> projections = new ArrayList<Projection>();
4934 projections.add(projection);
4935 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004936 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004937 }
ggalvizo82499442022-10-24 12:46:15 -07004938 SelectClause selectClause = new SelectClause(selectElement, selectRegular, fieldExclusions, distinct);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004939 selectClause.setSourceLocation(sourceLoc);
4940 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004941 }
4942}
4943
Yingyi Buab817482016-08-19 21:29:31 -07004944SelectRegular SelectRegular() throws ParseException:
4945{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004946 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004947 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004948 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004949}
4950{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004951 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004952 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004953 projections.add(projection);
4954 startSrcLoc = projection.getSourceLocation();
4955 }
4956 ( LOOKAHEAD(2) <COMMA> projection = Projection()
4957 {
4958 projections.add(projection);
4959 }
4960 )*
4961 {
4962 SelectRegular selectRegular = new SelectRegular(projections);
4963 selectRegular.setSourceLocation(startSrcLoc);
4964 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004965 }
4966}
4967
Yingyi Buab817482016-08-19 21:29:31 -07004968SelectElement SelectElement() throws ParseException:
4969{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004970 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004971 Expression expr = null;
4972 String name = null;
4973}
4974{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004975 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004976 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004977 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004978 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004979 }
4980}
4981
Yingyi Buab817482016-08-19 21:29:31 -07004982Projection Projection() throws ParseException :
4983{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004984 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004985 Expression expr = null;
4986 Identifier identifier = null;
4987 String name = null;
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004988 Projection.Kind kind = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004989 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004990 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004991}
4992{
4993 (
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07004994 <MUL> { kind = Projection.Kind.STAR; startSrcLoc = getSourceLocation(token); }
4995 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { kind = Projection.Kind.VAR_STAR; }
ggalvizo82499442022-10-24 12:46:15 -07004996 | expr = Expression()
4997 ( // EXCLUDE is a soft-keyword-- we want to avoid mistaking EXCLUDE as an identifier here.
4998 LOOKAHEAD({ getToken(1).kind == AS || getToken(1).kind == QUOTED_STRING
4999 || (getToken(1).kind == IDENTIFIER && !laIdentifier(1, EXCLUDE)) })
5000 (<AS>)? name = Identifier()
5001 )?
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08005002 {
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07005003 kind = Projection.Kind.NAMED_EXPR;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08005004 if (name == null) {
5005 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
5006 if (generatedColumnIdentifier != null) {
5007 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
5008 }
5009 }
5010 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005011 )
5012 {
Dmitry Lychaginb4121e42021-09-23 15:55:09 -07005013 Projection projection = new Projection(kind, expr, name);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005014 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
5015 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005016 }
5017}
5018
5019FromClause FromClause() throws ParseException :
5020{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005021 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005022 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
5023 extendCurrentScope();
5024}
5025{
5026 {
5027 FromTerm fromTerm = null;
5028 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005029 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005030 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
5031 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005032 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005033 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005034 }
5035}
5036
5037FromTerm FromTerm() throws ParseException :
5038{
5039 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04005040 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005041 VariableExpr posVar = null;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005042 AbstractBinaryCorrelateClause correlateClause = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005043 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
5044}
5045{
Ali Alsulimanba91b1c2023-02-22 21:33:55 -08005046 leftExpr = Expression()
5047 {
5048 if (leftExpr.getKind() == Expression.Kind.VARIABLE_EXPRESSION) {
5049 Token hintToken = fetchHint(token, SqlppHint.SUBPATH_HINT);
5050 if (hintToken != null) {
5051 String subPath = hintToken.hintParams;
5052 ((VariableExpr) leftExpr).addHint(new ExternalSubpathAnnotation(subPath));
5053 }
5054 }
5055 } ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005056 (
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005057 (
5058 correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER)
5059 | ( <INNER> correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER) )
5060 | ( <LEFT> ( <OUTER> )? correlateClause = JoinOrUnnestClause(JoinType.LEFTOUTER, UnnestType.LEFTOUTER) )
5061 | ( <RIGHT> ( <OUTER> )? correlateClause = JoinClause(JoinType.RIGHTOUTER) )
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005062 | ( <CROSS> correlateClause = CrossJoinClause() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005063 )
5064 {
5065 correlateClauses.add(correlateClause);
5066 }
5067 )*
5068 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005069 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07005070 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07005071 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005072 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
5073 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
5074 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005075 }
5076}
5077
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005078AbstractBinaryCorrelateClause JoinOrUnnestClause(JoinType joinType, UnnestType unnestType) throws ParseException :
5079{
5080 AbstractBinaryCorrelateClause correlateClause = null;
5081}
5082{
5083 ( correlateClause = JoinClause(joinType) | correlateClause = UnnestClause(unnestType) )
5084 {
5085 return correlateClause;
5086 }
5087}
5088
Yingyi Bu391f09e2015-10-29 13:49:39 -07005089JoinClause JoinClause(JoinType joinType) throws ParseException :
5090{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005091 Token startToken = null;
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005092 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005093 Expression conditionExpr = null;
5094}
5095{
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005096 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput() <ON> conditionExpr = Expression()
5097 {
5098 JoinClause joinClause = new JoinClause(joinType, rightInput.first, rightInput.second, rightInput.third,
Dmitry Lychagin34b72ac2021-11-29 16:25:40 -08005099 conditionExpr, joinType == JoinType.INNER ? null : Literal.Type.MISSING);
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005100 return addSourceLocation(joinClause, startToken);
5101 }
5102}
5103
5104JoinClause CrossJoinClause() throws ParseException :
5105{
5106 Token startToken = null;
5107 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
5108 Expression conditionExpr = null;
5109}
5110{
5111 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput()
5112 {
5113 JoinClause joinClause = new JoinClause(JoinType.INNER, rightInput.first, rightInput.second, rightInput.third,
Dmitry Lychagin34b72ac2021-11-29 16:25:40 -08005114 new LiteralExpr(TrueLiteral.INSTANCE), null);
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005115 return addSourceLocation(joinClause, startToken);
5116 }
5117}
5118
5119Triple<Expression, VariableExpr, VariableExpr> JoinClauseRightInput() throws ParseException :
5120{
5121 Expression rightExpr = null;
5122 VariableExpr rightVar = null;
5123 VariableExpr posVar = null;
5124}
5125{
Ali Alsulimanba91b1c2023-02-22 21:33:55 -08005126 rightExpr = Expression()
5127 {
5128 if (rightExpr.getKind() == Expression.Kind.VARIABLE_EXPRESSION) {
5129 Token hintToken = fetchHint(token, SqlppHint.SUBPATH_HINT);
5130 if (hintToken != null) {
5131 String subPath = hintToken.hintParams;
5132 ((VariableExpr) rightExpr).addHint(new ExternalSubpathAnnotation(subPath));
5133 }
5134 }
5135 } ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005136 {
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005137 if (rightVar == null) {
5138 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07005139 }
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005140 return new Triple<Expression, VariableExpr, VariableExpr>(rightExpr, rightVar, posVar);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005141 }
5142}
5143
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005144UnnestClause UnnestClause(UnnestType unnestType) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07005145{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005146 Token startToken = null;
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08005147 Expression rightExpr = null;
5148 VariableExpr rightVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005149 VariableExpr posVar = null;
5150}
5151{
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08005152 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005153 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005154 if (rightVar == null) {
5155 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07005156 }
Dmitry Lychagin34b72ac2021-11-29 16:25:40 -08005157 UnnestClause unnestClause = new UnnestClause(unnestType, rightExpr, rightVar, posVar,
5158 unnestType == UnnestType.INNER ? null : Literal.Type.MISSING);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005159 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005160 }
5161}
5162
Yingyi Bu391f09e2015-10-29 13:49:39 -07005163List<LetClause> LetClause() throws ParseException:
5164{
5165 List<LetClause> letList = new ArrayList<LetClause>();
5166 LetClause letClause;
5167}
5168{
5169 (
5170 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
5171 |
5172 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
5173 )
5174 {
5175 return letList;
5176 }
5177}
5178
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005179WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07005180{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005181 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005182 Expression whereExpr;
5183}
5184{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005185 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005186 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005187 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005188 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005189 }
5190}
5191
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005192OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07005193{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005194 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005195 OrderbyClause oc = new OrderbyClause();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005196 Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> orderbyExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005197 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005198 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005199 List<OrderbyClause.NullOrderModifier> nullModifierList = new ArrayList<OrderbyClause.NullOrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07005200}
5201{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005202 <ORDER>
5203 {
5204 startToken = token;
5205 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
5206 if (hintToken != null) {
5207 switch (hintToken.hint) {
5208 case INMEMORY_HINT:
5209 String[] splits = hintToken.hintParams.split("\\s+");
5210 int numFrames = Integer.parseInt(splits[0]);
5211 int numTuples = Integer.parseInt(splits[1]);
5212 oc.setNumFrames(numFrames);
5213 oc.setNumTuples(numTuples);
5214 break;
5215 case RANGE_HINT:
5216 try {
5217 Expression rangeExpr = parseExpression(hintToken.hintParams);
5218 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
5219 oc.setRangeMap(rangeMap);
5220 } catch (CompilationException e) {
5221 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
Ali Alsuliman80225e22018-10-15 14:17:07 -07005222 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005223 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005224 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005225 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005226 }
5227 <BY> orderbyExpr = OrderByExpression()
5228 {
5229 orderbyList.add(orderbyExpr.first);
5230 modifierList.add(orderbyExpr.second);
5231 nullModifierList.add(orderbyExpr.third);
5232 }
5233 (
5234 LOOKAHEAD(2) <COMMA> orderbyExpr = OrderByExpression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005235 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005236 orderbyList.add(orderbyExpr.first);
5237 modifierList.add(orderbyExpr.second);
5238 nullModifierList.add(orderbyExpr.third);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005239 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005240 )*
5241 {
5242 oc.setOrderbyList(orderbyList);
5243 oc.setModifierList(modifierList);
5244 oc.setNullModifierList(nullModifierList);
5245 return addSourceLocation(oc, startToken);
5246 }
5247}
Yingyi Bu391f09e2015-10-29 13:49:39 -07005248
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005249Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> OrderByExpression()
5250 throws ParseException:
5251{
5252 Expression orderbyExpr = null;
5253 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
5254 OrderbyClause.NullOrderModifier nullModif = null;
5255}
5256{
5257 orderbyExpr = Expression()
5258 (
5259 <ASC> { modif = OrderbyClause.OrderModifier.ASC; }
5260 |
5261 <DESC> { modif = OrderbyClause.OrderModifier.DESC; }
5262 )?
5263 (
5264 LOOKAHEAD({ laIdentifier(NULLS) }) <IDENTIFIER> { expectToken(NULLS); } <IDENTIFIER>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005265 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005266 if (isToken(FIRST)) {
5267 nullModif = OrderbyClause.NullOrderModifier.FIRST;
5268 } else if (isToken(LAST)) {
5269 nullModif = OrderbyClause.NullOrderModifier.LAST;
5270 } else {
5271 throw createUnexpectedTokenError();
5272 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005273 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07005274 )?
5275 {
5276 return new Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier>(orderbyExpr, modif,
5277 nullModif);
5278 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005279}
5280
5281GroupbyClause GroupbyClause()throws ParseException :
5282{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005283 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005284 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005285 List<List<GbyVariableExpressionPair>> gbyList = null;
5286 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08005287 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07005288 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08005289 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005290}
5291{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005292 {
5293 Scope newScope = extendCurrentScopeNoPush(true);
5294 // extendCurrentScope(true);
5295 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005296 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005297 {
5298 startToken = token;
5299 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
5300 if (hintToken != null) {
5301 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005302 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005303 }
5304 <BY> groupingElementList = GroupingElementList()
5305 (
5306 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08005307 {
5308 groupVar = groupVarWithFieldList.first;
5309 groupFieldList = groupVarWithFieldList.second;
5310 }
Yingyi Buacc12a92016-03-26 17:25:05 -07005311 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005312 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005313 if (groupingSetsParser == null) {
5314 groupingSetsParser = new SqlppGroupingSetsParser();
5315 }
5316 SourceLocation sourceLoc = getSourceLocation(startToken);
5317 try {
5318 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
5319 } catch (CompilationException e) {
5320 throw new SqlppParseException(sourceLoc, e.getMessage());
5321 }
5322 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005323 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07005324 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07005325 gbc.setGroupVar(groupVar);
5326 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005327 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005328 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005329 }
5330}
5331
Dmitry Lychaginac98f482020-03-31 12:26:40 -07005332List<GroupingElement> GroupingElementList() throws ParseException:
5333{
5334 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
5335 GroupingElement groupingElement = null;
5336}
5337{
5338 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
5339 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
5340 {
5341 return groupingElementList;
5342 }
5343}
5344
5345GroupingElement GroupingElement() throws ParseException:
5346{
5347 GroupingElement groupingElement = null;
5348 List<GroupingSet> groupingSets = null;
5349 List<GroupingElement> groupingElements = null;
5350}
5351{
5352 (
5353 LOOKAHEAD(2)
5354 <LEFTPAREN> <RIGHTPAREN>
5355 {
5356 groupingElement = GroupingSet.EMPTY;
5357 }
5358 |
5359 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
5360 <IDENTIFIER> { expectToken(ROLLUP); }
5361 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
5362 {
5363 groupingElement = new RollupCube(groupingSets, false);
5364 }
5365 |
5366 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
5367 <IDENTIFIER> { expectToken(CUBE); }
5368 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
5369 {
5370 groupingElement = new RollupCube(groupingSets, true);
5371 }
5372 |
5373 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
5374 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
5375 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
5376 {
5377 groupingElement = new GroupingSets(groupingElements);
5378 }
5379 |
5380 groupingElement = OrdinaryGroupingSet()
5381 )
5382 {
5383 return groupingElement;
5384 }
5385}
5386
5387GroupingSet OrdinaryGroupingSet() throws ParseException:
5388{
5389 GbyVariableExpressionPair gbyExprPair = null;
5390 List<GbyVariableExpressionPair> items = null;
5391}
5392{
5393 (
5394 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
5395 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
5396 )
5397 {
5398 return new GroupingSet(items);
5399 }
5400}
5401
5402List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
5403{
5404 GroupingSet groupingSet = null;
5405 List<GroupingSet> items = new ArrayList<GroupingSet>();
5406}
5407{
5408 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
5409 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
5410 {
5411 return items;
5412 }
5413}
5414
5415List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
5416{
5417 GbyVariableExpressionPair gbyExprPair = null;
5418 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
5419}
5420{
5421 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
5422 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
5423 {
5424 return items;
5425 }
5426}
5427
5428GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
5429{
5430 Expression expr = null;
5431 VariableExpr var = null;
5432}
5433{
5434 expr = Expression() ( (<AS>)? var = Variable() )?
5435 {
5436 return new GbyVariableExpressionPair(var, expr);
5437 }
5438}
5439
Yingyi Bu391f09e2015-10-29 13:49:39 -07005440HavingClause HavingClause() throws ParseException:
5441{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005442 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005443 Expression filterExpr = null;
5444}
5445{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005446 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005447 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005448 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005449 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005450 }
5451}
5452
5453LimitClause LimitClause() throws ParseException:
5454{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005455 Token startToken = null;
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005456 Expression limitExpr = null, offsetExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005457}
5458{
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005459 (
5460 (
5461 <LIMIT> { startToken = token; pushForbiddenScope(getCurrentScope()); } limitExpr = Expression()
5462 ( <OFFSET> offsetExpr = Expression() )?
5463 { popForbiddenScope(); }
5464 )
5465 |
5466 (
5467 <OFFSET> { startToken = token; pushForbiddenScope(getCurrentScope()); } offsetExpr = Expression()
5468 { popForbiddenScope(); }
5469 )
5470 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005471 {
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005472 LimitClause lc = new LimitClause(limitExpr, offsetExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005473 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005474 }
5475}
5476
5477QuantifiedExpression QuantifiedExpression()throws ParseException:
5478{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005479 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005480 QuantifiedExpression qc = new QuantifiedExpression();
5481 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
5482 Expression satisfiesExpr;
5483 VariableExpr var;
5484 Expression inExpr;
5485 QuantifiedPair pair;
5486}
5487{
5488 {
5489 createNewScope();
5490 }
5491
ggalvizo21e52822021-07-27 10:26:31 -10005492 ( LOOKAHEAD(2)
5493 (<ANY>|<SOME>)<AND><EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME_AND_EVERY); }
5494 | (<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); }
5495 | <EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005496 var = Variable() <IN> inExpr = Expression()
5497 {
5498 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005499 quantifiedList.add(pair);
5500 }
5501 (
5502 <COMMA> var = Variable() <IN> inExpr = Expression()
5503 {
5504 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005505 quantifiedList.add(pair);
5506 }
5507 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07005508 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005509 {
5510 qc.setSatisfiesExpr(satisfiesExpr);
5511 qc.setQuantifiedList(quantifiedList);
5512 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005513 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005514 }
5515}
5516
5517LetClause LetElement() throws ParseException:
5518{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005519 VariableExpr varExp;
5520 Expression beExp;
5521 extendCurrentScope();
5522}
5523{
5524 varExp = Variable() <EQ> beExp = Expression()
5525 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005526 LetClause lc = new LetClause(varExp, beExp);
5527 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005528 return lc;
5529 }
5530}
5531
5532LetClause WithElement() throws ParseException:
5533{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005534 VariableExpr varExp;
5535 Expression beExp;
5536 extendCurrentScope();
5537}
5538{
5539 varExp = Variable() <AS> beExp = Expression()
5540 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005541 LetClause lc = new LetClause(varExp, beExp);
5542 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005543 return lc;
5544 }
5545}
5546
5547TOKEN_MGR_DECLS:
5548{
5549 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07005550 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005551 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005552
5553 public void pushState() {
5554 lexerStateStack.push( curLexState );
5555 }
5556
5557 public void popState(String token) {
5558 if (lexerStateStack.size() > 0) {
5559 SwitchTo( lexerStateStack.pop() );
5560 } else {
5561 int errorLine = input_stream.getEndLine();
5562 int errorColumn = input_stream.getEndColumn();
5563 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
5564 + "\" but state stack is empty.";
5565 throw new TokenMgrError(msg, -1);
5566 }
5567 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005568
5569 void CommonTokenAction(Token token) {
5570 Token hintToken = token.specialToken;
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07005571 while (hintToken != null) { // make this a while loop
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005572 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
5573 String text = hintToken.image.substring(1).trim();
5574 boolean hintFound = hintToken.parseHint(text);
5575 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
Vijay Sarathy243ef2a2022-08-23 11:56:56 -07005576 hintToken = hintToken.specialToken;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005577 }
5578 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005579}
5580
5581<DEFAULT,IN_DBL_BRACE>
5582TOKEN [IGNORE_CASE]:
5583{
Ian Maxon38fe9402020-01-29 19:27:40 -08005584 <ADAPTER: "adapter">
5585 | <ALL : "all">
Dmitry Lychaginf3eec282022-06-10 12:44:25 -07005586 | <ANALYZE: "analyze">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005587 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07005588 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005589 | <APPLY : "apply">
5590 | <AS : "as">
5591 | <ASC : "asc">
5592 | <AT : "at">
5593 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005594 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005595 | <BTREE : "btree">
5596 | <BY : "by">
5597 | <CASE : "case">
Ali Alsuliman69ce7d82021-11-01 15:00:16 -07005598 | <CAST : "cast">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005599 | <CLOSED : "closed">
5600 | <CREATE : "create">
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005601 | <CROSS : "cross">
Dmitry Lychagin0ebc4202021-01-25 13:12:41 -08005602 | <COMPACTION : "compaction"> // no longer used
Yingyi Bu391f09e2015-10-29 13:49:39 -07005603 | <COMPACT : "compact">
5604 | <CONNECT : "connect">
5605 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07005606 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07005607 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07005608 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005609 | <DECLARE : "declare">
5610 | <DEFINITION : "definition">
5611 | <DELETE : "delete">
5612 | <DESC : "desc">
5613 | <DISCONNECT : "disconnect">
5614 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005615 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005616 | <DROP : "drop">
5617 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07005618 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005619 | <ELSE : "else">
5620 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07005621 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005622 | <EVERY : "every">
5623 | <EXCEPT : "except">
5624 | <EXISTS : "exists">
5625 | <EXTERNAL : "external">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005626 | <FALSE : "false">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005627 | <FEED : "feed">
5628 | <FILTER : "filter">
5629 | <FLATTEN : "flatten">
5630 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005631 | <FROM : "from">
5632 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08005633 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005634 | <FUNCTION : "function">
5635 | <GROUP : "group">
5636 | <HAVING : "having">
5637 | <HINTS : "hints">
5638 | <IF : "if">
5639 | <INTO : "into">
5640 | <IN : "in">
5641 | <INDEX : "index">
5642 | <INGESTION : "ingestion">
5643 | <INNER : "inner">
5644 | <INSERT : "insert">
5645 | <INTERNAL : "internal">
5646 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07005647 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005648 | <JOIN : "join">
5649 | <KEYWORD : "keyword">
5650 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07005651 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005652 | <LEFT : "left">
5653 | <LETTING : "letting">
5654 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005655 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005656 | <LIMIT : "limit">
5657 | <LOAD : "load">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005658 | <MISSING : "missing">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005659 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005660 | <NODEGROUP : "nodegroup">
5661 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07005662 | <NOT : "not">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005663 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005664 | <OFFSET : "offset">
5665 | <ON : "on">
5666 | <OPEN : "open">
5667 | <OR : "or">
5668 | <ORDER : "order">
5669 | <OUTER : "outer">
5670 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07005671 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005672 | <PATH : "path">
5673 | <POLICY : "policy">
5674 | <PRESORTED : "pre-sorted">
5675 | <PRIMARY : "primary">
5676 | <RAW : "raw">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005677 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005678 | <RETURNING : "returning">
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005679 | <RIGHT : "right">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005680 | <RTREE : "rtree">
5681 | <RUN : "run">
5682 | <SATISFIES : "satisfies">
5683 | <SECONDARY : "secondary">
5684 | <SELECT : "select">
5685 | <SET : "set">
5686 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08005687 | <START : "start">
5688 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08005689 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03005690 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07005691 | <THEN : "then">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005692 | <TO : "to">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005693 | <TRUE : "true">
5694 | <TYPE : "type">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005695 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08005696 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005697 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07005698 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005699 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07005700 | <USE : "use">
5701 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005702 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08005703 | <VALUED : "valued">
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07005704 | <VIEW : "view">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005705 | <WHEN : "when">
5706 | <WHERE : "where">
5707 | <WITH : "with">
5708 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005709}
5710
5711<DEFAULT,IN_DBL_BRACE>
5712TOKEN :
5713{
5714 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07005715 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005716 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005717 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005718 | <MUL : "*">
5719 | <PLUS : "+">
5720
5721 | <LEFTPAREN : "(">
5722 | <RIGHTPAREN : ")">
5723 | <LEFTBRACKET : "[">
5724 | <RIGHTBRACKET : "]">
5725
5726 | <ATT : "@">
5727 | <COLON : ":">
5728 | <COMMA : ",">
5729 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005730 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005731 | <QUES : "?">
5732 | <SEMICOLON : ";">
5733 | <SHARP : "#">
5734
5735 | <LT : "<">
5736 | <GT : ">">
5737 | <LE : "<=">
5738 | <GE : ">=">
5739 | <EQ : "=">
5740 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07005741 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005742 | <SIMILAR : "~=">
5743}
5744
5745<DEFAULT,IN_DBL_BRACE>
5746TOKEN :
5747{
5748 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
5749}
5750
5751<DEFAULT>
5752TOKEN :
5753{
5754 <RIGHTBRACE : "}"> { popState("}"); }
5755}
5756
5757<DEFAULT,IN_DBL_BRACE>
5758TOKEN :
5759{
5760 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
5761}
5762
5763<IN_DBL_BRACE>
5764TOKEN :
5765{
5766 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
5767}
5768
5769<DEFAULT,IN_DBL_BRACE>
5770TOKEN :
5771{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005772 <#DIGIT : ["0" - "9"]>
5773}
5774
5775<DEFAULT,IN_DBL_BRACE>
5776TOKEN:
5777{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005778 <INTEGER_LITERAL : <DIGITS> >
5779 | <DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07005780 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
5781 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005782 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005783 | <FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
Yingyi Bue4d919e2016-10-30 10:47:03 -07005784 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
5785 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005786 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005787 | <#DIGITS : (<DIGIT>)+ >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005788}
5789
5790<DEFAULT,IN_DBL_BRACE>
5791TOKEN :
5792{
5793 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005794 | <#IDENTIFIER_START_SPECIALCHAR : ["_"]>
5795 | <#IDENTIFIER_REST_SPECIALCHAR : ["$"]>
5796 | <#IDENTIFIER_START : <LETTER> | <IDENTIFIER_START_SPECIALCHAR> >
5797 | <#IDENTIFIER_REST : <LETTER> | <DIGIT> | <IDENTIFIER_START_SPECIALCHAR> | <IDENTIFIER_REST_SPECIALCHAR> >
5798 | <IDENTIFIER : <IDENTIFIER_START> (<IDENTIFIER_REST>)* >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005799}
5800
5801<DEFAULT,IN_DBL_BRACE>
5802TOKEN :
5803{
5804 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07005805 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005806 <EscapeQuot>
5807 | <EscapeBslash>
5808 | <EscapeSlash>
5809 | <EscapeBspace>
5810 | <EscapeFormf>
5811 | <EscapeNl>
5812 | <EscapeCr>
5813 | <EscapeTab>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305814 | <EscapeBTickWithBslash>
5815 | <EscapeBtickWithBtick>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005816 | ~["`","\\"])* "`">
Dmitry Lychagin984bc522021-10-13 16:22:25 -07005817 | <STRING_LITERAL : ( ("E")? "\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005818 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005819 | <EscapeBslash>
5820 | <EscapeSlash>
5821 | <EscapeBspace>
5822 | <EscapeFormf>
5823 | <EscapeNl>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305824 | <EscapeBTickWithBslash>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005825 | <EscapeCr>
5826 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005827 | ~["\"","\\"])* "\"")
Dmitry Lychagin984bc522021-10-13 16:22:25 -07005828 | ( ("E")? "\'" (
Yingyi Bu6d57e492016-06-06 21:24:42 -07005829 <EscapeApos>
5830 | <EscapeBslash>
5831 | <EscapeSlash>
5832 | <EscapeBspace>
5833 | <EscapeFormf>
5834 | <EscapeNl>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305835 | <EscapeBTickWithBslash>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005836 | <EscapeCr>
5837 | <EscapeTab>
5838 | ~["\'","\\"])* "\'")>
Ritik Rajdf16f3b2023-03-27 19:56:56 +05305839 | < #EscapeBTickWithBslash: "\\`" >
5840 | < #EscapeBtickWithBtick: "``" >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005841 | < #EscapeQuot: "\\\"" >
5842 | < #EscapeApos: "\\\'" >
5843 | < #EscapeBslash: "\\\\" >
5844 | < #EscapeSlash: "\\/" >
5845 | < #EscapeBspace: "\\b" >
5846 | < #EscapeFormf: "\\f" >
5847 | < #EscapeNl: "\\n" >
5848 | < #EscapeCr: "\\r" >
5849 | < #EscapeTab: "\\t" >
5850}
5851
5852<DEFAULT,IN_DBL_BRACE>
5853TOKEN :
5854{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005855 <DOLLAR_INTEGER_LITERAL : "$" <INTEGER_LITERAL> >
5856 | <DOLLAR_IDENTIFIER : "$" <IDENTIFIER> >
5857 | <DOLLAR_QUOTED_STRING: "$" <QUOTED_STRING> >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005858}
5859
5860<DEFAULT,IN_DBL_BRACE>
5861SKIP:
5862{
5863 " "
5864 | "\t"
5865 | "\r"
5866 | "\n"
5867}
5868
5869<DEFAULT,IN_DBL_BRACE>
5870SKIP:
5871{
5872 <"//" (~["\n"])* "\n">
5873}
5874
5875<DEFAULT,IN_DBL_BRACE>
5876SKIP:
5877{
5878 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5879}
5880
5881<DEFAULT,IN_DBL_BRACE>
5882SKIP:
5883{
Yingyi Bu93846a72016-09-13 16:30:39 -07005884 <"--" (~["\n"])* "\n">
5885}
5886
5887
5888<DEFAULT,IN_DBL_BRACE>
5889SKIP:
5890{
5891 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5892}
5893
5894<DEFAULT,IN_DBL_BRACE>
5895SKIP:
5896{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005897 <"/*"> { pushState(); } : INSIDE_COMMENT
5898}
5899
5900<INSIDE_COMMENT>
5901SPECIAL_TOKEN:
5902{
5903 <"+"(" ")*(~["*"])*>
5904}
5905
5906<INSIDE_COMMENT>
5907SKIP:
5908{
5909 <"/*"> { pushState(); }
5910}
5911
5912<INSIDE_COMMENT>
5913SKIP:
5914{
5915 <"*/"> { popState("*/"); }
5916 | <~[]>
5917}