blob: de9ea7a63ce5f8fc9b4685b46ddc1c6ae0dbf2a0 [file] [log] [blame]
Michael Blow82464fb2017-03-28 18:48:13 -04001//
2// Licensed to the Apache Software Foundation (ASF) under one
3// or more contributor license agreements. See the NOTICE file
4// distributed with this work for additional information
5// regarding copyright ownership. The ASF licenses this file
6// to you under the Apache License, Version 2.0 (the
7// "License"); you may not use this file except in compliance
8// with the License. You may obtain a copy of the License at
9//
10// http://www.apache.org/licenses/LICENSE-2.0
11//
12// Unless required by applicable law or agreed to in writing,
13// software distributed under the License is distributed on an
14// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15// KIND, either express or implied. See the License for the
16// specific language governing permissions and limitations
17// under the License.
18//
Yingyi Bu391f09e2015-10-29 13:49:39 -070019options {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -070020 COMMON_TOKEN_ACTION = true;
21 STATIC = false;
22 TOKEN_EXTENDS = "org.apache.asterix.lang.sqlpp.parser.SqlppToken";
Yingyi Bu391f09e2015-10-29 13:49:39 -070023}
24
Yingyi Bu391f09e2015-10-29 13:49:39 -070025PARSER_BEGIN(SQLPPParser)
26
27package org.apache.asterix.lang.sqlpp.parser;
28
29// For SQL++ ParserTokenManager
Till Westmanne9b2adf2016-10-15 12:39:01 -070030import java.util.ArrayDeque;
Yingyi Bu391f09e2015-10-29 13:49:39 -070031
32import java.io.BufferedReader;
33import java.io.File;
34import java.io.FileInputStream;
35import java.io.FileNotFoundException;
36import java.io.IOException;
37import java.io.InputStreamReader;
38import java.io.Reader;
39import java.io.StringReader;
40import java.util.ArrayList;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070041import java.util.Arrays;
42import java.util.Collection;
Yingyi Budaa549c2016-06-28 22:30:52 -070043import java.util.Collections;
Yingyi Bu391f09e2015-10-29 13:49:39 -070044import java.util.HashMap;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070045import java.util.HashSet;
Yingyi Bu391f09e2015-10-29 13:49:39 -070046import java.util.Iterator;
47import java.util.LinkedHashMap;
48import java.util.List;
49import java.util.Map;
50
51import org.apache.asterix.common.annotations.AutoDataGen;
52import org.apache.asterix.common.annotations.DateBetweenYearsDataGen;
53import org.apache.asterix.common.annotations.DatetimeAddRandHoursDataGen;
54import org.apache.asterix.common.annotations.DatetimeBetweenYearsDataGen;
55import org.apache.asterix.common.annotations.FieldIntervalDataGen;
56import org.apache.asterix.common.annotations.FieldValFileDataGen;
57import org.apache.asterix.common.annotations.FieldValFileSameIndexDataGen;
58import org.apache.asterix.common.annotations.IRecordFieldDataGen;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080059import org.apache.asterix.common.annotations.IndexedNLJoinExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070060import org.apache.asterix.common.annotations.InsertRandIntDataGen;
61import org.apache.asterix.common.annotations.ListDataGen;
62import org.apache.asterix.common.annotations.ListValFileDataGen;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -060063import org.apache.asterix.common.annotations.RangeAnnotation;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -080064import org.apache.asterix.common.annotations.SecondaryIndexSearchPreferenceAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -070065import org.apache.asterix.common.annotations.SkipSecondaryIndexSearchExpressionAnnotation;
66import org.apache.asterix.common.annotations.TypeDataGen;
67import org.apache.asterix.common.annotations.UndeclaredFieldsDataGen;
68import org.apache.asterix.common.config.DatasetConfig.DatasetType;
69import org.apache.asterix.common.config.DatasetConfig.IndexType;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -070070import org.apache.asterix.common.exceptions.AsterixException;
Taewoo Kime65e6ca2017-01-14 17:53:28 -080071import org.apache.asterix.common.exceptions.CompilationException;
Dmitry Lychagin85142c02018-04-05 17:27:36 -070072import org.apache.asterix.common.exceptions.ErrorCode;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -070073import org.apache.asterix.common.exceptions.WarningCollector;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -080074import org.apache.asterix.common.functions.FunctionConstants;
Yingyi Bu391f09e2015-10-29 13:49:39 -070075import org.apache.asterix.common.functions.FunctionSignature;
Dmitry Lychagin54c06012019-11-13 15:54:20 -080076import org.apache.asterix.common.metadata.DataverseName;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -070077import org.apache.asterix.common.metadata.DatasetFullyQualifiedName;
Dmitry Lychagin276adf92019-01-15 13:16:40 -080078import org.apache.asterix.lang.common.base.AbstractClause;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070079import org.apache.asterix.lang.common.base.AbstractLangExpression;
80import org.apache.asterix.lang.common.base.AbstractStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -070081import org.apache.asterix.lang.common.base.Expression;
82import org.apache.asterix.lang.common.base.Literal;
83import org.apache.asterix.lang.common.base.IParser;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -070084import org.apache.asterix.lang.common.base.ILangExpression;
Ali Alsuliman80225e22018-10-15 14:17:07 -070085import org.apache.asterix.lang.common.base.IParserFactory;
Yingyi Bu391f09e2015-10-29 13:49:39 -070086import org.apache.asterix.lang.common.base.Statement;
87import org.apache.asterix.lang.common.clause.GroupbyClause;
88import org.apache.asterix.lang.common.clause.LetClause;
89import org.apache.asterix.lang.common.clause.LimitClause;
90import org.apache.asterix.lang.common.clause.OrderbyClause;
91import org.apache.asterix.lang.common.clause.UpdateClause;
92import org.apache.asterix.lang.common.clause.WhereClause;
93import org.apache.asterix.lang.common.context.RootScopeFactory;
94import org.apache.asterix.lang.common.context.Scope;
95import org.apache.asterix.lang.common.expression.AbstractAccessor;
96import org.apache.asterix.lang.common.expression.CallExpr;
97import org.apache.asterix.lang.common.expression.FieldAccessor;
98import org.apache.asterix.lang.common.expression.FieldBinding;
99import org.apache.asterix.lang.common.expression.GbyVariableExpressionPair;
100import org.apache.asterix.lang.common.expression.IfExpr;
101import org.apache.asterix.lang.common.expression.IndexAccessor;
Dmitry Lychagin8ba59442017-06-16 14:19:45 -0700102import org.apache.asterix.lang.common.expression.IndexedTypeExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700103import org.apache.asterix.lang.common.expression.ListConstructor;
Hussain Towailebc5b5deb2018-12-15 18:48:48 +0300104import org.apache.asterix.lang.common.expression.ListSliceExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700105import org.apache.asterix.lang.common.expression.LiteralExpr;
106import org.apache.asterix.lang.common.expression.OperatorExpr;
107import org.apache.asterix.lang.common.expression.OrderedListTypeDefinition;
108import org.apache.asterix.lang.common.expression.QuantifiedExpression;
109import org.apache.asterix.lang.common.expression.RecordConstructor;
110import org.apache.asterix.lang.common.expression.RecordTypeDefinition;
111import org.apache.asterix.lang.common.expression.TypeExpression;
112import org.apache.asterix.lang.common.expression.TypeReferenceExpression;
113import org.apache.asterix.lang.common.expression.UnaryExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700114import org.apache.asterix.lang.common.expression.UnorderedListTypeDefinition;
115import org.apache.asterix.lang.common.expression.VariableExpr;
116import org.apache.asterix.lang.common.literal.DoubleLiteral;
117import org.apache.asterix.lang.common.literal.FalseLiteral;
118import org.apache.asterix.lang.common.literal.FloatLiteral;
119import org.apache.asterix.lang.common.literal.LongIntegerLiteral;
Yingyi Bu535d86b2016-05-23 16:44:25 -0700120import org.apache.asterix.lang.common.literal.MissingLiteral;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700121import org.apache.asterix.lang.common.literal.NullLiteral;
122import org.apache.asterix.lang.common.literal.StringLiteral;
123import org.apache.asterix.lang.common.literal.TrueLiteral;
124import org.apache.asterix.lang.common.parser.ScopeChecker;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -0700125import org.apache.asterix.lang.common.statement.AdapterDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700126import org.apache.asterix.lang.common.statement.CompactStatement;
127import org.apache.asterix.lang.common.statement.ConnectFeedStatement;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -0800128import org.apache.asterix.lang.common.statement.StartFeedStatement;
129import org.apache.asterix.lang.common.statement.StopFeedStatement;
Ian Maxon38fe9402020-01-29 19:27:40 -0800130import org.apache.asterix.lang.common.statement.CreateAdapterStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700131import org.apache.asterix.lang.common.statement.CreateDataverseStatement;
132import org.apache.asterix.lang.common.statement.CreateFeedPolicyStatement;
133import org.apache.asterix.lang.common.statement.CreateFeedStatement;
134import org.apache.asterix.lang.common.statement.CreateFunctionStatement;
135import org.apache.asterix.lang.common.statement.CreateIndexStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800136import org.apache.asterix.lang.common.statement.CreateSynonymStatement;
Rui Guoe6986dd2020-11-25 19:50:06 -0800137import org.apache.asterix.lang.common.statement.CreateFullTextFilterStatement;
138import org.apache.asterix.lang.common.statement.CreateFullTextConfigStatement;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700139import org.apache.asterix.lang.common.statement.CreateViewStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700140import org.apache.asterix.lang.common.statement.DatasetDecl;
141import org.apache.asterix.lang.common.statement.DataverseDecl;
142import org.apache.asterix.lang.common.statement.DataverseDropStatement;
143import org.apache.asterix.lang.common.statement.DeleteStatement;
144import org.apache.asterix.lang.common.statement.DisconnectFeedStatement;
Yingyi Buab817482016-08-19 21:29:31 -0700145import org.apache.asterix.lang.common.statement.DropDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700146import org.apache.asterix.lang.common.statement.ExternalDetailsDecl;
147import org.apache.asterix.lang.common.statement.FeedDropStatement;
Abdullah Alamoudi5dc73ed2016-07-28 05:03:13 +0300148import org.apache.asterix.lang.common.statement.FeedPolicyDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700149import org.apache.asterix.lang.common.statement.FunctionDecl;
150import org.apache.asterix.lang.common.statement.FunctionDropStatement;
151import org.apache.asterix.lang.common.statement.IndexDropStatement;
Rui Guoe6986dd2020-11-25 19:50:06 -0800152import org.apache.asterix.lang.common.statement.FullTextFilterDropStatement;
153import org.apache.asterix.lang.common.statement.FullTextConfigDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700154import org.apache.asterix.lang.common.statement.InsertStatement;
155import org.apache.asterix.lang.common.statement.InternalDetailsDecl;
156import org.apache.asterix.lang.common.statement.LoadStatement;
157import org.apache.asterix.lang.common.statement.NodeGroupDropStatement;
158import org.apache.asterix.lang.common.statement.NodegroupDecl;
159import org.apache.asterix.lang.common.statement.Query;
160import org.apache.asterix.lang.common.statement.RefreshExternalDatasetStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700161import org.apache.asterix.lang.common.statement.SetStatement;
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800162import org.apache.asterix.lang.common.statement.SynonymDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700163import org.apache.asterix.lang.common.statement.TypeDecl;
164import org.apache.asterix.lang.common.statement.TypeDropStatement;
165import org.apache.asterix.lang.common.statement.UpdateStatement;
Yingyi Bucb5bf332017-01-02 22:19:50 -0800166import org.apache.asterix.lang.common.statement.UpsertStatement;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700167import org.apache.asterix.lang.common.statement.ViewDecl;
168import org.apache.asterix.lang.common.statement.ViewDropStatement;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700169import org.apache.asterix.lang.common.statement.WriteStatement;
170import org.apache.asterix.lang.common.struct.Identifier;
Dmitry Lychaginef1719e2017-12-15 08:33:07 -0800171import org.apache.asterix.lang.common.struct.OperatorType;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700172import org.apache.asterix.lang.common.struct.QuantifiedPair;
173import org.apache.asterix.lang.common.struct.VarIdentifier;
Ali Alsuliman80225e22018-10-15 14:17:07 -0700174import org.apache.asterix.lang.common.util.RangeMapBuilder;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700175import org.apache.asterix.lang.sqlpp.clause.AbstractBinaryCorrelateClause;
176import org.apache.asterix.lang.sqlpp.clause.FromClause;
177import org.apache.asterix.lang.sqlpp.clause.FromTerm;
178import org.apache.asterix.lang.sqlpp.clause.HavingClause;
179import org.apache.asterix.lang.sqlpp.clause.JoinClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700180import org.apache.asterix.lang.sqlpp.clause.Projection;
181import org.apache.asterix.lang.sqlpp.clause.SelectBlock;
182import org.apache.asterix.lang.sqlpp.clause.SelectClause;
183import org.apache.asterix.lang.sqlpp.clause.SelectElement;
184import org.apache.asterix.lang.sqlpp.clause.SelectRegular;
185import org.apache.asterix.lang.sqlpp.clause.SelectSetOperation;
186import org.apache.asterix.lang.sqlpp.clause.UnnestClause;
Xikui Wangf6741682018-02-22 19:17:17 -0800187import org.apache.asterix.lang.common.clause.WhereClause;
Yingyi Buc8c067c2016-07-25 23:37:19 -0700188import org.apache.asterix.lang.sqlpp.expression.CaseExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700189import org.apache.asterix.lang.sqlpp.expression.SelectExpression;
Dmitry Lychaginfdedf622018-10-30 18:12:40 -0700190import org.apache.asterix.lang.sqlpp.expression.WindowExpression;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700191import org.apache.asterix.lang.sqlpp.optype.JoinType;
192import org.apache.asterix.lang.sqlpp.optype.SetOpType;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -0700193import org.apache.asterix.lang.sqlpp.optype.UnnestType;
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700194import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser;
195import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingElement;
196import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSet;
197import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.GroupingSets;
198import org.apache.asterix.lang.sqlpp.parser.SqlppGroupingSetsParser.RollupCube;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700199import org.apache.asterix.lang.sqlpp.parser.SqlppHint;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700200import org.apache.asterix.lang.sqlpp.struct.SetOperationInput;
201import org.apache.asterix.lang.sqlpp.struct.SetOperationRight;
Yingyi Bu9e3f9be2016-07-01 10:07:37 -0700202import org.apache.asterix.lang.sqlpp.util.ExpressionToVariableUtil;
Xikui Wang96fd4022017-04-10 14:23:31 -0700203import org.apache.asterix.lang.sqlpp.util.FunctionMapUtil;
Yingyi Buacc12a92016-03-26 17:25:05 -0700204import org.apache.asterix.lang.sqlpp.util.SqlppVariableUtil;
Ian Maxon38fe9402020-01-29 19:27:40 -0800205import org.apache.asterix.external.dataset.adapter.AdapterIdentifier;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -0800206import org.apache.asterix.om.functions.BuiltinFunctions;
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700207import org.apache.asterix.om.types.BuiltinType;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700208import org.apache.commons.lang3.ArrayUtils;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700209import org.apache.commons.lang3.StringUtils;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700210import org.apache.hyracks.algebricks.common.utils.Pair;
211import org.apache.hyracks.algebricks.common.utils.Triple;
Xikui Wang5a61b2a2018-01-02 14:14:03 -0800212import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
Shiva2ea73232019-10-09 18:04:05 -0700213import org.apache.hyracks.algebricks.core.algebra.expressions.BroadcastExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700214import org.apache.hyracks.algebricks.core.algebra.expressions.IExpressionAnnotation;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700215import org.apache.hyracks.algebricks.core.algebra.functions.FunctionIdentifier;
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800216import org.apache.hyracks.api.exceptions.IWarningCollector;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700217import org.apache.hyracks.api.exceptions.SourceLocation;
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700218import org.apache.hyracks.api.exceptions.Warning;
Dmitry Lychagin70bf47a2020-10-22 13:25:57 -0700219import org.apache.hyracks.dataflow.common.data.partition.range.RangeMap;
Ali Alsuliman587b7902019-01-21 14:33:50 -0800220import org.apache.hyracks.util.LogRedactionUtil;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700221import org.apache.hyracks.util.StringUtil;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700222
Yingyi Bucaea8f02015-11-16 15:12:15 -0800223class SQLPPParser extends ScopeChecker implements IParser {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700224
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800225 // tokens parsed as identifiers
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700226 private static final String CUBE = "CUBE";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800227 private static final String CURRENT = "CURRENT";
Dmitry Lychagin33c77f92021-07-21 15:59:04 -0700228 private static final String DEFAULT = "DEFAULT";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800229 private static final String EXCLUDE = "EXCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700230 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800231 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700232 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800233 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700234 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700235 private static final String LAST = "LAST";
Glenn67fd1f32021-02-25 16:04:49 -0800236 private static final String META = "META";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800237 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700238 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800239 private static final String OTHERS = "OTHERS";
240 private static final String PARTITION = "PARTITION";
241 private static final String PRECEDING = "PRECEDING";
242 private static final String RANGE = "RANGE";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700243 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700244 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800245 private static final String ROW = "ROW";
246 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700247 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800248 private static final String TIES = "TIES";
249 private static final String UNBOUNDED = "UNBOUNDED";
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700250 private static final String REPLACE = "REPLACE";
Ian Maxon38fe9402020-01-29 19:27:40 -0800251 private static final String RETURNS = "RETURNS";
Rui Guoe6986dd2020-11-25 19:50:06 -0800252 private static final String CONFIG = "CONFIG";
253
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800254
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700255 private static final String INT_TYPE_NAME = "int";
Dmitry Lychagin541652082020-11-09 14:04:53 -0800256 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 -0700257
Till Westmann7199a562016-09-17 16:07:32 -0700258 // error configuration
259 protected static final boolean REPORT_EXPECTED_TOKENS = false;
260
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700261 private int externalVarCounter;
262
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800263 private DataverseName defaultDataverse;
264
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700265 private SqlppGroupingSetsParser groupingSetsParser;
266
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700267 private final WarningCollector warningCollector = new WarningCollector();
268
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700269 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
270
Yingyi Bu391f09e2015-10-29 13:49:39 -0700271 private static class IndexParams {
272 public IndexType type;
273 public int gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800274 public String fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700275
Rui Guoe6986dd2020-11-25 19:50:06 -0800276 public IndexParams(IndexType type, int gramLength, String fullTextConfig) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700277 this.type = type;
278 this.gramLength = gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800279 this.fullTextConfig = fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700280 }
281 };
282
283 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800284 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700285 public String library;
286 public String function;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -0600287 public Token hintToken;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700288 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700289 }
290
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700291 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
292 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
293 switch (hintToken.hint) {
294 case VAL_FILE_HINT:
295 File[] valFiles = new File[splits.length];
296 for (int k=0; k<splits.length; k++) {
297 valFiles[k] = new File(splits[k]);
298 }
299 return new FieldValFileDataGen(valFiles);
300 case VAL_FILE_SAME_INDEX_HINT:
301 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
302 case LIST_VAL_FILE_HINT:
303 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
304 case LIST_HINT:
305 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
306 case INTERVAL_HINT:
307 FieldIntervalDataGen.ValueType vt;
308 switch (splits[0]) {
309 case "int":
310 vt = FieldIntervalDataGen.ValueType.INT;
311 break;
312 case "long":
313 vt = FieldIntervalDataGen.ValueType.LONG;
314 break;
315 case "float":
316 vt = FieldIntervalDataGen.ValueType.FLOAT;
317 break;
318 case "double":
319 vt = FieldIntervalDataGen.ValueType.DOUBLE;
320 break;
321 default:
322 throw new SqlppParseException(getSourceLocation(hintToken),
323 "Unknown type for interval data gen: " + splits[0]);
324 }
325 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
326 case INSERT_RAND_INT_HINT:
327 return new InsertRandIntDataGen(splits[0], splits[1]);
328 case DATE_BETWEEN_YEARS_HINT:
329 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
330 case DATETIME_BETWEEN_YEARS_HINT:
331 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
332 case DATETIME_ADD_RAND_HOURS_HINT:
333 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
334 case AUTO_HINT:
335 return new AutoDataGen(splits[0]);
336 default:
337 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700338 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700339 }
340
Till Westmann7199a562016-09-17 16:07:32 -0700341 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700342 this(new StringReader(s));
343 super.setInput(s);
344 }
345
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800346 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700347 File file = new File(args[0]);
348 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
349 SQLPPParser parser = new SQLPPParser(fis);
350 List<Statement> st = parser.parse();
351 //st.accept(new SQLPPPrintVisitor(), 0);
352 }
353
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800354 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800355 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700356 return parseImpl(new ParseFunction<List<Statement>>() {
357 @Override
358 public List<Statement> parse() throws ParseException {
359 return SQLPPParser.this.Statement();
360 }
361 });
362 }
363
Dmitry Lychagin62f0beb2020-09-08 11:44:19 -0700364 @Override
365 public Expression parseExpression() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700366 return parseImpl(new ParseFunction<Expression>() {
367 @Override
368 public Expression parse() throws ParseException {
369 return SQLPPParser.this.Expression();
370 }
371 });
372 }
373
374 private static Expression parseExpression(String text) throws CompilationException {
375 return new SQLPPParser(text).parseExpression();
376 }
377
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800378 @Override
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700379 public List<String> parseMultipartIdentifier() throws CompilationException {
380 return parseImpl(new ParseFunction<List<String>>() {
381 @Override
382 public List<String> parse() throws ParseException {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700383 return SQLPPParser.this.MultipartIdentifier().first;
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700384 }
385 });
386 }
387
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800388 private List<String> parseParenthesizedIdentifierList() throws CompilationException {
389 return parseImpl(new ParseFunction<List<String>>() {
390 @Override
391 public List<String> parse() throws ParseException {
392 return SQLPPParser.this.ParenthesizedIdentifierList();
393 }
394 });
395 }
396
397 private static List<String> parseParenthesizedIdentifierList(String text) throws CompilationException {
398 return new SQLPPParser(text).parseParenthesizedIdentifierList();
399 }
400
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700401 @Override
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700402 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames, boolean isStored)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800403 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800404 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800405 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800406 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800407 DataverseName dataverse = defaultDataverse;
408 defaultDataverse = signature.getDataverseName();
409 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800410 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
411 for (String paramName : paramNames) {
412 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
413 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800414 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
415 removeCurrentScope();
416 defaultDataverse = dataverse;
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700417 return new FunctionDecl(signature, paramVars, functionBodyExpr, isStored);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800418 }
419 });
420 }
421
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700422 @Override
423 public ViewDecl parseViewBody(DatasetFullyQualifiedName viewName) throws CompilationException {
424 return parseImpl(new ParseFunction<ViewDecl>() {
425 @Override
426 public ViewDecl parse() throws ParseException {
427 DataverseName dataverse = defaultDataverse;
428 defaultDataverse = viewName.getDataverseName();
429 createNewScope();
430 Expression viewBodyExpr = SQLPPParser.this.ViewBody();
431 removeCurrentScope();
432 defaultDataverse = dataverse;
433 return new ViewDecl(viewName, viewBodyExpr);
434 }
435 });
436 }
437
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700438 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700439 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700440 hintCollector.clear();
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800441 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700442 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700443 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700444 } catch (SqlppParseException e) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700445 if (e.getCause() instanceof AlgebricksException) {
446 AlgebricksException cause = (AlgebricksException) e.getCause();
447 if (cause.getError().isPresent() && cause.getError().get() instanceof ErrorCode) {
448 throw new CompilationException((ErrorCode) cause.getError().get(), e.getSourceLocation(),
449 cause.getParams());
450 }
451 }
452 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(),
453 LogRedactionUtil.userData(getMessage(e)));
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700454 } catch (ParseException e) {
455 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700456 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700457 // 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 -0700458 // 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 -0700459 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700460 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700461 } finally {
462 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700463 }
464 }
Till Westmann7199a562016-09-17 16:07:32 -0700465
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700466 @FunctionalInterface
467 private interface ParseFunction<T> {
468 T parse() throws ParseException;
469 }
470
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700471 @Override
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800472 public void getWarnings(IWarningCollector outWarningCollector) {
473 warningCollector.getWarnings(outWarningCollector);
474 }
475
476 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700477 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
478 warningCollector.getWarnings(outWarnings, maxWarnings);
479 }
480
481 @Override
482 public long getTotalWarningsCount() {
483 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700484 }
485
Till Westmann7199a562016-09-17 16:07:32 -0700486 protected String getMessage(ParseException pe) {
487 Token currentToken = pe.currentToken;
488 if (currentToken == null) {
489 return pe.getMessage();
490 }
491 int[][] expectedTokenSequences = pe.expectedTokenSequences;
492 String[] tokenImage = pe.tokenImage;
493 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
494 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
495 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
496 Token tok = currentToken.next;
497 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700498 StringBuilder message = new StringBuilder(128);
499 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700500 for (int i = 0; i < maxSize; i++) {
501 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700502 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700503 }
504 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700505 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700506 break;
507 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700508 final String fixedTokenImage = tokenImage[tok.kind];
509 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700510 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700511 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700512 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700513 tok = tok.next;
514 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700515 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700516 if (REPORT_EXPECTED_TOKENS) {
517 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700518 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700519 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700520 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700521 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700522 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700523 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700524 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700525 }
526
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700527 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700528 return
529 token == null ? null :
530 token.sourceLocation != null ? token.sourceLocation :
531 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700532 }
533
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700534 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
535 expr.setSourceLocation(getSourceLocation(token));
536 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700537 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800538
539 private boolean isToken(String image) {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700540 return isToken(token, image);
541 }
542
543 private static boolean isToken(Token token, String image) {
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800544 return token.image.equalsIgnoreCase(image);
545 }
546
547 private void expectToken(String image) throws SqlppParseException {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700548 expectToken(token, image);
549 }
550
551 private static void expectToken(Token token, String image) throws SqlppParseException {
552 if (!isToken(token, image)) {
553 throw createUnexpectedTokenError(token);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800554 }
555 }
556
557 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700558 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700559 }
560
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700561 private static SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700562 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800563 }
564
565 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700566 return createUnexpectedTokenError(token, expected);
567 }
568
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700569 private static SqlppParseException createUnexpectedTokenError(Token t, String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700570 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800571 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700572 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800573 }
574
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700575 private boolean laToken(int idx, int kind) {
576 Token t = getToken(idx);
577 return t.kind == kind;
578 }
579
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800580 private boolean laToken(int idx, int kind, String image) {
581 Token t = getToken(idx);
582 return t.kind == kind && t.image.equalsIgnoreCase(image);
583 }
584
585 private boolean laIdentifier(int idx, String image) {
586 return laToken(idx, IDENTIFIER, image);
587 }
588
589 private boolean laIdentifier(String image) {
590 return laIdentifier(1, image);
591 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700592
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700593 private Token fetchHint(Token token, SqlppHint... expectedHints) {
594 Token hintToken = token.specialToken;
595 if (hintToken == null) {
596 return null;
597 }
598 SourceLocation sourceLoc = getSourceLocation(hintToken);
599 hintCollector.remove(sourceLoc);
600 if (hintToken.hint == null) {
601 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
602 return null;
603 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
604 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
605 return null;
606 } else {
607 return hintToken;
608 }
609 }
610
611 private void reportUnclaimedHints() {
612 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
613 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
614 }
615 }
616
617 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
618 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
619 }
620
621 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700622 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500623 warningCollector.warn(Warning.of(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
Ali Alsulimanaa118862019-09-10 20:55:45 -0700624 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700625 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700626
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800627 private IExpressionAnnotation parseExpressionAnnotation(Token hintToken) {
628 // placeholder for the annotation that should be returned if this hint's parameters cannot be parsed
629 IExpressionAnnotation onParseErrorReturn = null;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800630 try {
631 switch (hintToken.hint) {
632 case HASH_BROADCAST_JOIN_HINT:
633 return new BroadcastExpressionAnnotation(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
634 case INDEXED_NESTED_LOOP_JOIN_HINT:
635 if (hintToken.hintParams == null) {
636 return IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
637 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800638 // if parameter parsing fails then return hint annotation without parameters
639 onParseErrorReturn = IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800640 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
641 return IndexedNLJoinExpressionAnnotation.newInstance(indexNames);
642 }
643 case RANGE_HINT:
644 Expression rangeExpr = parseExpression(hintToken.hintParams);
645 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
646 return new RangeAnnotation(rangeMap);
647 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
648 if (hintToken.hintParams == null) {
649 return SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE_ANY_INDEX;
650 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800651 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800652 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
653 return SkipSecondaryIndexSearchExpressionAnnotation.newInstance(indexNames);
654 }
655 case USE_SECONDARY_INDEX_SEARCH_HINT:
656 if (hintToken.hintParams == null) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800657 throw new SqlppParseException(getSourceLocation(hintToken), "Expected index name(s)");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800658 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800659 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800660 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
661 return SecondaryIndexSearchPreferenceAnnotation.newInstance(indexNames);
662 }
663 default:
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800664 throw new SqlppParseException(getSourceLocation(hintToken), "Unexpected hint");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800665 }
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800666 } catch (SqlppParseException e) {
667 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500668 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800669 hintToken.hint.toString(), e.getMessage()));
670 }
671 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800672 } catch (CompilationException e) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800673 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500674 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800675 hintToken.hint.toString(), e.getMessage()));
676 }
677 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800678 }
679 }
680
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700681 private void ensureNoTypeDeclsInFunction(String fnName, List<Pair<VarIdentifier, TypeExpression>> paramList,
682 TypeExpression returnType, Token startToken) throws SqlppParseException {
683 for (Pair<VarIdentifier, TypeExpression> p : paramList) {
684 if (p.second != null) {
685 String paramName = SqlppVariableUtil.toUserDefinedName(p.first.getValue());
686 throw new SqlppParseException(getSourceLocation(startToken),
687 "Unexpected type declaration for parameter " + paramName + " in function " + fnName);
688 }
689 }
690 if (returnType != null) {
691 throw new SqlppParseException(getSourceLocation(startToken),
692 "Unexpected return type declaration for function " + fnName);
693 }
694 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -0700695
696 private void ensureIntegerLiteral(LiteralExpr expr, String errorPrefix) throws SqlppParseException {
697 Literal lit = expr.getValue();
698 if (lit.getLiteralType() != Literal.Type.INTEGER && lit.getLiteralType() != Literal.Type.LONG) {
699 throw new SqlppParseException(expr.getSourceLocation(), errorPrefix + " should be an INTEGER");
700 }
701 }
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700702
703 private DataverseName createDataverseName(List<String> parts, int fromIndex, int toIndex, Token startToken)
704 throws SqlppParseException {
705 try {
706 return DataverseName.create(parts, fromIndex, toIndex);
707 } catch (AsterixException e) {
708 SqlppParseException pe = new SqlppParseException(getSourceLocation(startToken), e.getMessage());
709 pe.initCause(e);
710 throw pe;
711 }
712 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700713}
714
715PARSER_END(SQLPPParser)
716
Yingyi Bu391f09e2015-10-29 13:49:39 -0700717List<Statement> Statement() throws ParseException:
718{
719 scopeStack.push(RootScopeFactory.createRootScope(this));
720 List<Statement> decls = new ArrayList<Statement>();
721 Statement stmt = null;
722}
723{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300724 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700725 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300726 {
727 decls.add(stmt);
728 }
729 )?
730 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700731 )*
732 <EOF>
733 {
734 return decls;
735 }
736}
737
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700738Statement ExplainStatement() throws ParseException:
739{
740 Statement stmt = null;
741 Token explainToken = null;
742}
743{
744 ( <EXPLAIN> { explainToken = token; } )?
745 stmt = SingleStatement()
746 {
747 if (explainToken != null) {
748 if (stmt.getKind() == Statement.Kind.QUERY) {
749 ((Query)stmt).setExplain(true);
750 } else {
751 throw new SqlppParseException(getSourceLocation(explainToken),
752 "EXPLAIN is not supported for this kind of statement");
753 }
754 }
755 return stmt;
756 }
757}
758
Yingyi Bu391f09e2015-10-29 13:49:39 -0700759Statement SingleStatement() throws ParseException:
760{
761 Statement stmt = null;
762}
763{
764 (
765 stmt = DataverseDeclaration()
766 | stmt = FunctionDeclaration()
767 | stmt = CreateStatement()
768 | stmt = LoadStatement()
769 | stmt = DropStatement()
770 | stmt = WriteStatement()
771 | stmt = SetStatement()
772 | stmt = InsertStatement()
773 | stmt = DeleteStatement()
774 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800775 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700776 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700777 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700778 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700779 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700780 )
781 {
782 return stmt;
783 }
784}
785
786DataverseDecl DataverseDeclaration() throws ParseException:
787{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700788 Token startToken = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700789 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700790}
791{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700792 <USE> { startToken = token; } dvName = DataverseName()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700793 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700794 defaultDataverse = dvName;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800795 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700796 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700797 }
798}
799
800Statement CreateStatement() throws ParseException:
801{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700802 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700803 Statement stmt = null;
804}
805{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700806 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700807 (
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700808 stmt = CreateOrReplaceStatement(startToken)
809 | stmt = CreateTypeStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800810 | stmt = CreateNodegroupStatement(startToken)
811 | stmt = CreateDatasetStatement(startToken)
812 | stmt = CreateIndexStatement(startToken)
813 | stmt = CreateDataverseStatement(startToken)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700814 | stmt = CreateFunctionStatement(startToken, false)
Ian Maxon38fe9402020-01-29 19:27:40 -0800815 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800816 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800817 | stmt = CreateFeedStatement(startToken)
818 | stmt = CreateFeedPolicyStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -0800819 | stmt = CreateFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700820 | stmt = CreateViewStatement(startToken, false)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700821 )
822 {
823 return stmt;
824 }
825}
826
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700827Statement CreateOrReplaceStatement(Token startStmtToken) throws ParseException:
828{
829 Statement stmt = null;
830 Token replaceToken = null;
831}
832{
833 <OR> <IDENTIFIER> { replaceToken = token; }
834 (
835 stmt = CreateFunctionStatement(startStmtToken, true)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700836 | stmt = CreateViewStatement(startStmtToken, true)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700837 )
838 {
839 // check expected token here to make the grammar extension plugin happy
840 expectToken(replaceToken, REPLACE);
841 return stmt;
842 }
843}
844
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800845TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
846{
847 TypeDecl stmt = null;
848}
849{
850 <TYPE> stmt = TypeSpecification(startStmtToken)
851 {
852 return stmt;
853 }
854}
855
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700856TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700857{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800858 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700859 boolean ifNotExists = false;
860 TypeExpression typeExpr = null;
861}
862{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800863 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700864 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800865 {
866 boolean dgen = false;
867 long numValues = -1;
868 String filename = null;
869 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
870 if (hintToken != null) {
871 String hintParams = hintToken.hintParams;
872 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
873 if (splits == null || splits.length != 2) {
874 throw new SqlppParseException(getSourceLocation(hintToken),
875 "Expecting /*+ dgen <filename> <numberOfItems> */");
876 }
877 dgen = true;
878 filename = splits[0];
879 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700880 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800881 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
882 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
883 return addSourceLocation(stmt, startStmtToken);
884 }
885}
886
887NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
888{
889 NodegroupDecl stmt = null;
890}
891{
892 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
893 {
894 return stmt;
895 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700896}
897
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700898NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700899{
900 String name = null;
901 String tmp = null;
902 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800903 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700904}
905{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800906 name = Identifier() ifNotExists = IfNotExists()
907 <ON> tmp = Identifier()
908 {
909 ncNames.add(new Identifier(tmp));
910 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700911 ( <COMMA> tmp = Identifier()
912 {
913 ncNames.add(new Identifier(tmp));
914 }
915 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800916 {
917 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
918 return addSourceLocation(stmt, startStmtToken);
919 }
920}
921
922void Dataset() throws ParseException:
923{
924}
925{
926 (<DATASET>|<COLLECTION>)
927}
928
929DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
930{
931 DatasetDecl stmt = null;
932}
933{
934 (
935 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
936 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
937 )
938 {
939 return stmt;
940 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700941}
942
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700943DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700944{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800945 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700946 boolean ifNotExists = false;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700947 TypeExpression typeExpr = null;
948 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800949 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700950 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700951 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700952 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800953 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -0800954 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700955}
956{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800957 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700958 typeExpr = DatasetTypeSpecification()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700959 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800960 { String name; }
961 <WITH>
962 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700963 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800964 if (!name.equalsIgnoreCase("meta")){
965 throw new SqlppParseException(getSourceLocation(startStmtToken),
966 "We can only support one additional associated field called \"meta\".");
967 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700968 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700969 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800970 )?
971 ifNotExists = IfNotExists()
972 primaryKeyFields = PrimaryKey()
973 (<AUTOGENERATED> { autogenerated = true; } )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800974 ( <HINTS> hints = Properties() )?
975 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
976 ( <WITH> withRecord = RecordConstructor() )?
977 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800978 try {
Xikui Wangca7927f2020-09-03 11:27:57 -0700979 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
980 filterField == null? null : filterField.first, filterField == null? null : filterField.second);
Murtadha Hubail353e95f2020-08-24 22:18:04 +0300981 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr, hints,
982 DatasetType.INTERNAL, idd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800983 return addSourceLocation(stmt, startStmtToken);
984 } catch (CompilationException e) {
985 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
986 }
987 }
988}
989
990DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
991{
992 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700993 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800994 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800995 String adapterName = null;
996 Map<String,String> properties = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800997 Map<String,String> hints = new HashMap<String,String>();
998 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800999 RecordConstructor withRecord = null;
1000}
1001{
1002 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001003 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001004 ifNotExists = IfNotExists()
1005 <USING> adapterName = AdapterName() properties = Configuration()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001006 ( <HINTS> hints = Properties() )?
1007 ( <WITH> withRecord = RecordConstructor() )?
1008 {
1009 ExternalDetailsDecl edd = new ExternalDetailsDecl();
1010 edd.setAdapter(adapterName);
1011 edd.setProperties(properties);
1012 try {
Murtadha Hubail353e95f2020-08-24 22:18:04 +03001013 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null, hints, DatasetType.EXTERNAL,
1014 edd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001015 return addSourceLocation(stmt, startStmtToken);
1016 } catch (CompilationException e) {
1017 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1018 }
1019 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001020}
1021
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001022TypeExpression DatasetTypeSpecification() throws ParseException:
1023{
1024 TypeExpression typeExpr = null;
1025}
1026{
1027 (
1028 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
1029 | typeExpr = DatasetReferenceTypeSpecification()
1030 )
1031 {
1032 return typeExpr;
1033 }
1034}
1035
1036TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
1037{
1038 TypeExpression typeExpr = null;
1039}
1040{
1041 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
1042 {
1043 return typeExpr;
1044 }
1045}
1046
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001047RecordTypeDefinition DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001048{
1049 RecordTypeDefinition recordTypeDef = null;
1050 RecordTypeDefinition.RecordKind recordKind = null;
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001051 Token startToken = null, recordKindToken = null;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001052}
1053{
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001054 <LEFTPAREN> { startToken = token; } recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001055 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
1056 {
1057 if (recordKind == null) {
1058 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
1059 } else if (!allowRecordKindModifier) {
1060 throw createUnexpectedTokenError(recordKindToken);
1061 }
1062 recordTypeDef.setRecordKind(recordKind);
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001063 return addSourceLocation(recordTypeDef, startToken);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001064 }
1065}
1066
1067RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
1068{
1069 RecordTypeDefinition recType = new RecordTypeDefinition();
1070}
1071{
1072 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
1073 {
1074 return recType;
1075 }
1076}
1077
1078void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
1079{
1080 String fieldName;
1081 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001082 boolean nullable = true, missable = true;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001083}
1084{
1085 fieldName = Identifier()
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001086 type = TypeReference() ( <NOT> <UNKNOWN> { nullable = false; missable = false; } )?
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001087 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001088 recType.addField(fieldName, type, nullable, missable);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001089 }
1090}
1091
Yingyi Bu391f09e2015-10-29 13:49:39 -07001092RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
1093{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001094 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001095 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001096 String datasetName = null;
1097}
1098{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001099 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
1100 {
1101 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
1102 stmt.setDataverseName(nameComponents.first);
1103 stmt.setDatasetName(nameComponents.second);
1104 return addSourceLocation(stmt, startToken);
1105 }
1106}
1107
1108CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
1109{
1110 CreateIndexStatement stmt = null;
1111}
1112{
1113 (
1114 <INDEX> stmt = IndexSpecification(startStmtToken)
1115 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
1116 )
1117 {
1118 return stmt;
1119 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001120}
1121
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001122CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001123{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001124 Pair<DataverseName,Identifier> nameComponents = null;
Glenn67fd1f32021-02-25 16:04:49 -08001125 String indexName = null;
1126 IndexParams indexParams = null;
1127 CreateIndexStatement.IndexedElement indexedElement = null;
1128 List<CreateIndexStatement.IndexedElement> indexedElementList = new ArrayList<CreateIndexStatement.IndexedElement>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001129 boolean enforced = false;
Glenn67fd1f32021-02-25 16:04:49 -08001130 boolean ifNotExists = false;
1131 boolean hasUnnest = false;
1132 String fullTextConfigName = null;
1133 Token startElementToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001134}
1135{
Ali Alsuliman8351d252017-09-24 00:43:15 -07001136 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001137 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -07001138 <ON> nameComponents = QualifiedName()
Glenn67fd1f32021-02-25 16:04:49 -08001139 <LEFTPAREN> { startElementToken = token; }
1140 indexedElement = IndexedElement(startElementToken) {
1141 indexedElementList.add(indexedElement);
1142 hasUnnest |= indexedElement.hasUnnest();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001143 }
Glenn67fd1f32021-02-25 16:04:49 -08001144 (<COMMA> { startElementToken = token; }
1145 indexedElement = IndexedElement(startElementToken) {
1146 indexedElementList.add(indexedElement);
1147 hasUnnest |= indexedElement.hasUnnest();
1148 }
1149 )*
1150 <RIGHTPAREN>
1151 ( <TYPE> indexParams = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman8351d252017-09-24 00:43:15 -07001152 )
1153 {
Glenn67fd1f32021-02-25 16:04:49 -08001154 IndexType indexType;
1155 int gramLength;
1156 if (indexParams != null) {
1157 indexType = indexParams.type;
1158 gramLength = indexParams.gramLength;
1159 fullTextConfigName = indexParams.fullTextConfig;
1160 } else {
1161 indexType = hasUnnest ? IndexType.ARRAY : IndexType.BTREE;
1162 gramLength = -1;
1163 fullTextConfigName = null;
Ali Alsuliman8351d252017-09-24 00:43:15 -07001164 }
Glenn67fd1f32021-02-25 16:04:49 -08001165 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
1166 new Identifier(indexName), indexType, indexedElementList, enforced, gramLength, fullTextConfigName, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001167 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001168 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001169}
1170
Glenn67fd1f32021-02-25 16:04:49 -08001171CreateIndexStatement.IndexedElement IndexedElement(Token startElementToken) throws ParseException:
1172{
1173 Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1174 Pair<List<String>, IndexedTypeExpression> elementSimple = null;
1175 int elementSimpleSource = 0;
1176}
1177{
1178 (
1179 element = IndexedElementUnnestSelect()
1180 | (
1181 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1182 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1183 <DOT> elementSimple = IndexedField()
1184 { elementSimpleSource = 1; }
1185 )
1186 | elementSimple = IndexedField()
1187 | <LEFTPAREN> ( element = IndexedElementUnnestSelect() | elementSimple = IndexedField() ) <RIGHTPAREN>
1188 )
1189 {
1190 int source;
1191 List<List<String>> unnestList;
1192 List<Pair<List<String>, IndexedTypeExpression>> projectList;
1193 if (elementSimple != null) {
1194 source = elementSimpleSource;
1195 unnestList = null;
1196 projectList = Collections.singletonList(elementSimple);
1197 } else {
1198 source = element.first;
1199 unnestList = element.second;
1200 projectList = element.third;
1201 }
1202 CreateIndexStatement.IndexedElement ie = new CreateIndexStatement.IndexedElement(source, unnestList, projectList);
1203 ie.setSourceLocation(getSourceLocation(startElementToken));
1204 return ie;
1205 }
1206}
1207
1208Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelect()
1209 throws ParseException:
1210{
1211 int source = 0;
1212 Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1213}
1214{
1215 <UNNEST>
1216 (
1217 (
1218 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1219 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1220 <DOT> element = IndexedElementUnnestSelectBody() { source = 1; }
1221 ) | element = IndexedElementUnnestSelectBody()
1222 )
1223 {
1224 return new Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(
1225 source, element.first, element.second
1226 );
1227 }
1228}
1229
1230Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelectBody()
1231 throws ParseException:
1232{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001233 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001234 IndexedTypeExpression type = null;
1235 List<List<String>> unnestList = new ArrayList();
Glenn67fd1f32021-02-25 16:04:49 -08001236 List<Pair<List<String>, IndexedTypeExpression>> projectList = new ArrayList();
1237}
1238{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001239 path = MultipartIdentifier() { unnestList.add(path.first); }
1240 ( <UNNEST> path = MultipartIdentifier() { unnestList.add(path.first); })*
Glenn67fd1f32021-02-25 16:04:49 -08001241 (
Glenn58329202021-04-09 12:03:46 -07001242 ( <COLON> type = IndexedTypeExpr(false)
Glenn67fd1f32021-02-25 16:04:49 -08001243 {
1244 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, type));
1245 }
1246 ) |
1247 (
Glenn58329202021-04-09 12:03:46 -07001248 <SELECT> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1249 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001250 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001251 }
1252 ( <COMMA> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1253 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001254 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001255 }
1256 )*
Glenn67fd1f32021-02-25 16:04:49 -08001257 )
1258 )?
1259 {
1260 if (projectList.isEmpty()) {
1261 // To support the case (<UNNEST> IDENTIFIER)* IDENTIFIER w/o any type specification.
1262 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, null));
1263 }
1264
1265 return new Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(unnestList, projectList);
1266 }
1267}
1268
1269Pair<List<String>, IndexedTypeExpression> IndexedField() throws ParseException:
1270{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001271 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001272 IndexedTypeExpression type = null;
1273}
1274{
Glenn58329202021-04-09 12:03:46 -07001275 path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(true) )?
Glenn67fd1f32021-02-25 16:04:49 -08001276 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001277 return new Pair<List<String>, IndexedTypeExpression>(path.first, type);
Glenn67fd1f32021-02-25 16:04:49 -08001278 }
1279}
1280
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001281CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
1282{
Glenn67fd1f32021-02-25 16:04:49 -08001283 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001284 String indexName = null;
1285 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001286}
1287{
1288 (indexName = Identifier())? ifNotExists = IfNotExists()
1289 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
1290 {
1291 if (indexName == null) {
1292 indexName = "primary_idx_" + nameComponents.second;
1293 }
Glenn67fd1f32021-02-25 16:04:49 -08001294 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
1295 new Identifier(indexName), IndexType.BTREE, Collections.emptyList(), false, -1, null, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001296 return addSourceLocation(stmt, startStmtToken);
1297 }
1298}
1299
Yingyi Bu391f09e2015-10-29 13:49:39 -07001300String FilterField() throws ParseException :
1301{
1302 String filterField = null;
1303}
1304{
1305 filterField = Identifier()
1306 {
1307 return filterField;
1308 }
1309}
1310
1311IndexParams IndexType() throws ParseException:
1312{
1313 IndexType type = null;
1314 int gramLength = 0;
Rui Guoe6986dd2020-11-25 19:50:06 -08001315 String fullTextConfig = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001316}
1317{
1318 (<BTREE>
1319 {
1320 type = IndexType.BTREE;
1321 }
1322 | <RTREE>
1323 {
1324 type = IndexType.RTREE;
1325 }
1326 | <KEYWORD>
1327 {
1328 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1329 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001330 | <FULLTEXT>
Taewoo Kimc49405a2017-01-04 00:30:43 -08001331 {
1332 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1333 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001334 // For now we don't allow inverted index creation using a full-text config in another data verse.
1335 // We may want to support corss-dataverse full-text config access later
1336 // If so, replace the Identifier() with QualifiedName() to get the dataverse name
1337 ( <USING> Identifier()
1338 {
1339 fullTextConfig = token.image;
1340 }
1341 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001342 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1343 {
1344 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1345 gramLength = Integer.valueOf(token.image);
1346 }
1347 <RIGHTPAREN>)
1348 {
Rui Guoe6986dd2020-11-25 19:50:06 -08001349 return new IndexParams(type, gramLength, fullTextConfig);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001350 }
1351}
1352
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001353CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1354{
1355 CreateDataverseStatement stmt = null;
1356}
1357{
1358 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1359 {
1360 return stmt;
1361 }
1362}
1363
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001364CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001365{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001366 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001367 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001368}
1369{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001370 dvName = DataverseName() ifNotExists = IfNotExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001371 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001372 CreateDataverseStatement stmt = new CreateDataverseStatement(dvName, null, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001373 return addSourceLocation(stmt, startStmtToken);
1374 }
1375}
1376
Ian Maxon38fe9402020-01-29 19:27:40 -08001377CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1378{
1379 CreateAdapterStatement stmt = null;
1380}
1381{
1382 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1383 {
1384 return stmt;
1385 }
1386}
1387
1388CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1389{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001390 Pair<DataverseName,Identifier> adapterName = null;
1391 Pair<DataverseName,Identifier> libraryName = null;
1392 List<String> externalIdentifier = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001393 boolean ifNotExists = false;
1394}
1395{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001396 adapterName = QualifiedName()
1397 ifNotExists = IfNotExists()
1398 <AS> externalIdentifier = FunctionExternalIdentifier()
1399 <AT> libraryName = QualifiedName()
1400 {
1401 CreateAdapterStatement stmt = new CreateAdapterStatement(adapterName.first, adapterName.second.getValue(),
1402 libraryName.first, libraryName.second.getValue(), externalIdentifier, ifNotExists);
1403 return addSourceLocation(stmt, startStmtToken);
1404 }
1405}
Ian Maxon38fe9402020-01-29 19:27:40 -08001406
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001407CreateViewStatement CreateViewStatement(Token startStmtToken, boolean orReplace) throws ParseException:
1408{
1409 CreateViewStatement stmt = null;
1410}
1411{
1412 <VIEW> stmt = ViewSpecification(startStmtToken, orReplace)
1413 {
1414 return stmt;
1415 }
1416}
1417
1418CreateViewStatement ViewSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
1419{
1420 Pair<DataverseName, Identifier> nameComponents = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001421 TypeExpression typeExpr = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001422 boolean ifNotExists = false;
1423 Token beginPos = null, endPos = null;
1424 Expression viewBodyExpr = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001425 Boolean defaultNull = null;
1426 RecordConstructor withRecord = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001427 DataverseName currentDataverse = defaultDataverse;
1428}
1429{
1430 nameComponents = QualifiedName()
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001431 (
1432 (
1433 typeExpr = DatasetTypeSpecification()
1434 ifNotExists = IfNotExists()
1435 <IDENTIFIER> { expectToken(DEFAULT); } <NULL> { defaultNull = true; }
1436 ( <WITH> withRecord = RecordConstructor() )?
1437 )
1438 |
1439 ( ifNotExists = IfNotExists() )
1440 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001441 {
1442 if (orReplace && ifNotExists) {
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001443 throw new SqlppParseException(getSourceLocation(startStmtToken), "Unexpected IF NOT EXISTS");
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001444 }
1445 }
1446 <AS>
1447 {
1448 beginPos = token;
1449 createNewScope();
1450 if (nameComponents.first != null) {
1451 defaultDataverse = nameComponents.first;
1452 }
1453 }
1454 viewBodyExpr = ViewBody()
1455 {
1456 endPos = token;
1457 String viewBody = extractFragment(beginPos.beginLine, beginPos.beginColumn + 1, endPos.endLine,
1458 endPos.endColumn + 1);
1459 removeCurrentScope();
1460 defaultDataverse = currentDataverse;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001461 try {
1462 CreateViewStatement stmt = new CreateViewStatement(nameComponents.first, nameComponents.second.getValue(),
1463 typeExpr, viewBody, viewBodyExpr, defaultNull, withRecord, orReplace, ifNotExists);
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001464 return addSourceLocation(stmt, startStmtToken);
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001465 } catch (CompilationException e) {
1466 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1467 }
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001468 }
1469}
1470
1471Expression ViewBody() throws ParseException:
1472{
1473 Expression viewBodyExpr = null;
1474}
1475{
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001476 (
1477 ( viewBodyExpr = VariableRef() ( viewBodyExpr = FieldAccessor(viewBodyExpr) )* )
1478 | viewBodyExpr = SelectExpression(true)
1479 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001480 {
1481 return viewBodyExpr;
1482 }
1483}
1484
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001485CreateFunctionStatement CreateFunctionStatement(Token startStmtToken, boolean orReplace) throws ParseException:
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001486{
1487 CreateFunctionStatement stmt = null;
1488}
1489{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001490 <FUNCTION> stmt = FunctionSpecification(startStmtToken, orReplace)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001491 {
1492 return stmt;
1493 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001494}
1495
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001496CreateFunctionStatement FunctionSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001497{
Ian Maxon38fe9402020-01-29 19:27:40 -08001498 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001499 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001500 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001501 List<Pair<VarIdentifier,TypeExpression>> params = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001502 int arity = 0;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001503 TypeExpression returnType = null;
1504 Token beginPos = null, endPos = null;
1505 Expression functionBodyExpr = null;
1506 Pair<DataverseName,Identifier> libraryName = null;
1507 List<String> externalIdentifier = null;
1508 RecordConstructor withOptions = null;
1509 boolean ifNotExists = false;
1510 CreateFunctionStatement stmt = null;
1511 DataverseName currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001512}
1513{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001514 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001515 {
1516 defaultDataverse = fctName.dataverse;
1517 }
Dmitry Lychagin541652082020-11-09 14:04:53 -08001518 paramsWithArity = FunctionParameters()
1519 {
1520 arity = paramsWithArity.first;
1521 params = paramsWithArity.second;
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001522 signature = new FunctionSignature(fctName.dataverse, fctName.function, arity);
Dmitry Lychagin541652082020-11-09 14:04:53 -08001523 }
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001524 ifNotExists = IfNotExists()
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001525 {
1526 if (orReplace && ifNotExists) {
1527 throw new SqlppParseException(getSourceLocation(token), "Unexpected IF NOT EXISTS");
1528 }
1529 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001530 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001531 (
1532 (
1533 <LEFTBRACE>
1534 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001535 beginPos = token;
1536 createNewScope();
Ian Maxon38fe9402020-01-29 19:27:40 -08001537 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001538 functionBodyExpr = FunctionBody()
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001539 <RIGHTBRACE>
1540 {
1541 endPos = token;
1542 String functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine,
1543 endPos.beginColumn);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001544 getCurrentScope().addFunctionDescriptor(signature, false);
1545 removeCurrentScope();
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001546 ensureNoTypeDeclsInFunction(fctName.function, params, returnType, startStmtToken);
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001547 stmt = new CreateFunctionStatement(signature, params, functionBody, functionBodyExpr, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001548 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001549 )
1550 |
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001551 (
1552 <AS> externalIdentifier = FunctionExternalIdentifier()
1553 <AT> libraryName = QualifiedName()
1554 (<WITH> withOptions = RecordConstructor())?
1555 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001556 try {
1557 stmt = new CreateFunctionStatement(signature, params, returnType, libraryName.first,
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001558 libraryName.second.getValue(), externalIdentifier, withOptions, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001559 } catch (AlgebricksException e) {
1560 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1561 }
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001562 }
1563 )
Ian Maxon38fe9402020-01-29 19:27:40 -08001564 )
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001565 {
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001566 defaultDataverse = currentDataverse;
1567 return addSourceLocation(stmt, startStmtToken);
1568 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001569}
1570
Dmitry Lychagin541652082020-11-09 14:04:53 -08001571Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameters() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001572{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001573 Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> params = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001574}
1575{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001576 <LEFTPAREN> (params = FunctionParameterList())? <RIGHTPAREN>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001577 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001578 if (params == null) {
1579 params = new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(
1580 0, Collections.<Pair<VarIdentifier, TypeExpression>>emptyList()
1581 );
1582 }
1583 return params;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001584 }
1585}
1586
Dmitry Lychagin541652082020-11-09 14:04:53 -08001587Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameterList() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001588{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001589 List<Pair<VarIdentifier, TypeExpression>> paramList = null;
1590 Pair<VarIdentifier, TypeExpression> param = null;
1591 int arity = 0;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001592}
1593{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001594 (
1595 (
1596 <DOT> <DOT> <DOT>
1597 {
1598 param = new Pair<VarIdentifier, TypeExpression>(
1599 SqlppVariableUtil.toInternalVariableIdentifier(UDF_VARARGS_PARAM_NAME), null
1600 );
1601 paramList = Collections.<Pair<VarIdentifier, TypeExpression>>singletonList(param);
1602 arity = FunctionIdentifier.VARARGS;
1603 }
1604 )
1605 |
1606 (
1607 param = FunctionParameter()
1608 {
1609 paramList = new ArrayList<Pair<VarIdentifier, TypeExpression>>();
1610 paramList.add(param);
1611 }
1612 ( <COMMA> param = FunctionParameter() { paramList.add(param); } )*
1613 {
1614 arity = paramList.size();
1615 }
1616 )
1617 )
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001618 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001619 return new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(arity, paramList);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001620 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001621}
1622
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001623Pair<VarIdentifier,TypeExpression> FunctionParameter() throws ParseException:
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001624{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001625 String name = null;
1626 TypeExpression type = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001627}
1628{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001629 name = VariableIdentifier()
1630 ( LOOKAHEAD(3) (<COLON>)? type = TypeExpr(false) )?
1631 {
1632 return new Pair<VarIdentifier,TypeExpression>(new VarIdentifier(name), type);
1633 }
1634}
1635
1636TypeExpression FunctionReturnType() throws ParseException:
1637{
1638 TypeExpression returnType = null;
1639}
1640{
1641 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = TypeExpr(false) )?
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001642 {
1643 return returnType;
1644 }
1645}
1646
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001647Expression FunctionBody() throws ParseException:
1648{
1649 Expression functionBodyExpr = null;
1650}
1651{
1652 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1653 {
1654 return functionBodyExpr;
1655 }
1656}
1657
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001658List<String> FunctionExternalIdentifier() throws ParseException:
1659{
1660 String ident = null;
1661 List<String> identList = new ArrayList(2);
1662}
1663{
1664 ident = StringLiteral() { identList.add(ident.trim()); }
1665 ( <COMMA> ident = StringLiteral() { identList.add(ident.trim()); } )*
1666 {
1667 return identList;
1668 }
1669}
1670
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001671CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1672{
1673 CreateFeedStatement stmt = null;
1674}
1675{
1676 <FEED> stmt = FeedSpecification(startStmtToken)
1677 {
1678 return stmt;
1679 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001680}
1681
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001682CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001683{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001684 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001685 boolean ifNotExists = false;
1686 String adapterName = null;
1687 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001688 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001689 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001690 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001691}
1692{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001693 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001694 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001695 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001696 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001697 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001698 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001699 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001700 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001701 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001702 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001703}
1704
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001705CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1706{
1707 CreateFeedPolicyStatement stmt = null;
1708}
1709{
1710 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1711 {
1712 return stmt;
1713 }
1714}
1715
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001716CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001717{
Michael Blowd6cf6412016-06-30 02:44:35 -04001718 String policyName = null;
1719 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001720 String sourcePolicyFile = null;
1721 String definition = null;
1722 boolean ifNotExists = false;
1723 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001724 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001725}
1726{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001727 policyName = Identifier() ifNotExists = IfNotExists()
1728 <FROM>
1729 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001730 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001731 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001732 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001733 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1734 {
1735 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1736 }
1737 )
1738 {
1739 return addSourceLocation(stmt, startStmtToken);
1740 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001741}
1742
Rui Guoe6986dd2020-11-25 19:50:06 -08001743Statement CreateFullTextStatement(Token startStmtToken) throws ParseException:
1744{
1745 Statement stmt = null;
1746}
1747{
1748 (
1749 <FULLTEXT>
1750 (
1751 <FILTER> stmt = CreateFullTextFilterSpec(startStmtToken)
1752 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = CreateFullTextConfigSpec(startStmtToken))
1753 )
1754 )
1755 {
1756 return stmt;
1757 }
1758}
1759
1760CreateFullTextFilterStatement CreateFullTextFilterSpec(Token startStmtToken) throws ParseException:
1761{
1762 CreateFullTextFilterStatement stmt = null;
1763 Pair<DataverseName,Identifier> nameComponents = null;
1764 boolean ifNotExists = false;
1765 RecordConstructor expr = null;
1766}
1767{
1768 (
1769 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1770 <AS>
1771 expr = RecordConstructor()
1772 )
1773 {
1774 try {
1775 stmt = new CreateFullTextFilterStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
1776 return addSourceLocation(stmt, startStmtToken);
1777 } catch (CompilationException e) {
1778 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1779 }
1780 }
1781}
1782
1783CreateFullTextConfigStatement CreateFullTextConfigSpec(Token startStmtToken) throws ParseException:
1784{
1785 CreateFullTextConfigStatement stmt = null;
1786 Pair<DataverseName,Identifier> nameComponents = null;
1787 boolean ifNotExists = false;
1788 RecordConstructor expr = null;
1789}
1790{
1791 (
1792 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1793 <AS>
1794 expr = RecordConstructor()
1795 )
1796 {
1797 try {
1798 stmt = new CreateFullTextConfigStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
1799 return addSourceLocation(stmt, startStmtToken);
1800 } catch (CompilationException e) {
1801 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1802 }
1803 }
1804}
1805
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001806CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
1807{
1808 CreateSynonymStatement stmt = null;
1809}
1810{
1811 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
1812 {
1813 return stmt;
1814 }
1815}
1816
1817CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
1818{
1819 Pair<DataverseName,Identifier> nameComponents = null;
1820 Pair<DataverseName,Identifier> objectNameComponents = null;
1821 boolean ifNotExists = false;
1822}
1823{
1824 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1825 <FOR> objectNameComponents = QualifiedName()
1826 {
1827 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
1828 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
1829 return addSourceLocation(stmt, startStmtToken);
1830 }
1831}
1832
Yingyi Bu391f09e2015-10-29 13:49:39 -07001833boolean IfNotExists() throws ParseException:
1834{
1835}
1836{
Yingyi Budaa549c2016-06-28 22:30:52 -07001837 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001838 {
1839 return true;
1840 }
1841 )?
1842 {
1843 return false;
1844 }
1845}
1846
Xikui Wang9d63f622017-05-18 17:50:44 -07001847void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001848{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001849 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07001850 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001851}
1852{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001853 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001854 {
1855 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1856 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
1857 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07001858 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001859 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07001860 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001861 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1862 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07001863 }
1864 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07001865}
1866
1867String GetPolicy() throws ParseException:
1868{
1869 String policy = null;
1870}
1871{
1872 <USING> <POLICY> policy = Identifier()
1873 {
1874 return policy;
1875 }
1876
1877}
1878
1879FunctionSignature FunctionSignature() throws ParseException:
1880{
1881 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001882 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> params = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001883 int arity = 0;
1884}
1885{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001886 fctName = FunctionName()
1887 (
Dmitry Lychagin541652082020-11-09 14:04:53 -08001888 LOOKAHEAD(2) params = FunctionParameters() { arity = params.first; }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001889 | ( <LEFTPAREN> arity = FunctionArity() <RIGHTPAREN> )
1890 | ( <ATT> arity = FunctionArity() ) // back-compat
1891 )
1892 {
1893 return new FunctionSignature(fctName.dataverse, fctName.function, arity);
1894 }
1895}
Yingyi Bu391f09e2015-10-29 13:49:39 -07001896
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001897int FunctionArity() throws ParseException:
1898{
1899 int arity;
1900}
1901{
1902 <INTEGER_LITERAL>
1903 {
1904 try {
1905 arity = Integer.parseInt(token.image);
1906 } catch (NumberFormatException e) {
1907 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001908 }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001909 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
1910 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + arity);
1911 }
1912 return arity;
1913 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001914}
1915
Yingyi Buc9bfe252016-03-01 00:02:40 -08001916Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001917{
Yingyi Buc9bfe252016-03-01 00:02:40 -08001918 Pair<Integer, List<String>> tmp = null;
1919 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001920 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
1921}
1922{
1923 <PRIMARY> <KEY> tmp = NestedField()
1924 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001925 keyFieldSourceIndicators.add(tmp.first);
1926 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001927 }
1928 ( <COMMA> tmp = NestedField()
1929 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001930 keyFieldSourceIndicators.add(tmp.first);
1931 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001932 }
1933 )*
1934 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001935 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001936 }
1937}
1938
1939Statement DropStatement() throws ParseException:
1940{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001941 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001942 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001943}
1944{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001945 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001946 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001947 stmt = DropDatasetStatement(startToken)
1948 | stmt = DropIndexStatement(startToken)
1949 | stmt = DropNodeGroupStatement(startToken)
1950 | stmt = DropTypeStatement(startToken)
1951 | stmt = DropDataverseStatement(startToken)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001952 | stmt = DropAdapterStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001953 | stmt = DropFunctionStatement(startToken)
1954 | stmt = DropFeedStatement(startToken)
1955 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001956 | stmt = DropSynonymStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -08001957 | stmt = DropFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001958 | stmt = DropViewStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07001959 )
1960 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001961 return stmt;
1962 }
1963}
1964
1965DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
1966{
1967 DropDatasetStatement stmt = null;
1968}
1969{
1970 Dataset() stmt = DropDatasetSpecification(startStmtToken)
1971 {
1972 return stmt;
1973 }
1974}
1975
1976DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
1977{
1978 Pair<DataverseName,Identifier> pairId = null;
1979 boolean ifExists = false;
1980}
1981{
1982 pairId = QualifiedName() ifExists = IfExists()
1983 {
1984 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
1985 return addSourceLocation(stmt, startStmtToken);
1986 }
1987}
1988
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001989ViewDropStatement DropViewStatement(Token startStmtToken) throws ParseException:
1990{
1991 ViewDropStatement stmt = null;
1992}
1993{
1994 <VIEW> stmt = DropViewSpecification(startStmtToken)
1995 {
1996 return stmt;
1997 }
1998}
1999
2000ViewDropStatement DropViewSpecification(Token startStmtToken) throws ParseException:
2001{
2002 Pair<DataverseName,Identifier> pairId = null;
2003 boolean ifExists = false;
2004}
2005{
2006 pairId = QualifiedName() ifExists = IfExists()
2007 {
2008 ViewDropStatement stmt = new ViewDropStatement(pairId.first, pairId.second, ifExists);
2009 return addSourceLocation(stmt, startStmtToken);
2010 }
2011}
2012
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002013IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
2014{
2015 IndexDropStatement stmt = null;
2016}
2017{
2018 <INDEX> stmt = DropIndexSpecification(startStmtToken)
2019 {
2020 return stmt;
2021 }
2022}
2023
2024IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
2025{
2026 Triple<DataverseName,Identifier,Identifier> tripleId = null;
2027 boolean ifExists = false;
2028}
2029{
2030 tripleId = DoubleQualifiedName() ifExists = IfExists()
2031 {
2032 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
2033 return addSourceLocation(stmt, startStmtToken);
2034 }
2035}
2036
Rui Guoe6986dd2020-11-25 19:50:06 -08002037Statement DropFullTextStatement(Token startStmtToken) throws ParseException:
2038{
2039 Statement stmt = null;
2040}
2041{
2042 <FULLTEXT>
2043 (
2044 <FILTER> stmt = DropFullTextFilterSpec(startStmtToken)
2045 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = DropFullTextConfigSpec(startStmtToken))
2046 )
2047 {
2048 return stmt;
2049 }
2050}
2051
2052FullTextFilterDropStatement DropFullTextFilterSpec(Token startStmtToken) throws ParseException:
2053{
2054 FullTextFilterDropStatement stmt = null;
2055 Pair<DataverseName,Identifier> nameComponents = null;
2056 boolean ifExists = false;
2057}
2058{
2059 nameComponents = QualifiedName() ifExists = IfExists()
2060 {
2061 stmt = new FullTextFilterDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2062 return addSourceLocation(stmt, startStmtToken);
2063 }
2064}
2065
2066FullTextConfigDropStatement DropFullTextConfigSpec(Token startStmtToken) throws ParseException:
2067{
2068 FullTextConfigDropStatement stmt = null;
2069 Pair<DataverseName,Identifier> nameComponents = null;
2070 boolean ifExists = false;
2071}
2072{
2073 nameComponents = QualifiedName() ifExists = IfExists()
2074 {
2075 stmt = new FullTextConfigDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2076 return addSourceLocation(stmt, startStmtToken);
2077 }
2078}
2079
2080
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002081NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
2082{
2083 NodeGroupDropStatement stmt = null;
2084}
2085{
2086 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
2087 {
2088 return stmt;
2089 }
2090}
2091
2092NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
2093{
2094 String id = null;
2095 boolean ifExists = false;
2096}
2097{
2098 id = Identifier() ifExists = IfExists()
2099 {
2100 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
2101 return addSourceLocation(stmt, startStmtToken);
2102 }
2103}
2104
2105TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
2106{
2107 TypeDropStatement stmt = null;
2108}
2109{
2110 <TYPE> stmt = DropTypeSpecification(startStmtToken)
2111 {
2112 return stmt;
2113 }
2114}
2115
2116TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
2117{
2118 Pair<DataverseName,Identifier> pairId = null;
2119 boolean ifExists = false;
2120}
2121{
2122 pairId = TypeName() ifExists = IfExists()
2123 {
2124 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
2125 return addSourceLocation(stmt, startStmtToken);
2126 }
2127}
2128
2129DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
2130{
2131 DataverseDropStatement stmt = null;
2132}
2133{
2134 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
2135 {
2136 return stmt;
2137 }
2138}
2139
2140DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
2141{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002142 DataverseName dvName = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002143 boolean ifExists = false;
2144}
2145{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002146 dvName = DataverseName() ifExists = IfExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002147 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002148 DataverseDropStatement stmt = new DataverseDropStatement(dvName, ifExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002149 return addSourceLocation(stmt, startStmtToken);
2150 }
2151}
2152
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002153AdapterDropStatement DropAdapterStatement(Token startStmtToken) throws ParseException:
2154{
2155 AdapterDropStatement stmt = null;
2156}
2157{
2158 <ADAPTER> stmt = DropAdapterSpecification(startStmtToken)
2159 {
2160 return stmt;
2161 }
2162}
2163
2164AdapterDropStatement DropAdapterSpecification(Token startStmtToken) throws ParseException:
2165{
2166 Pair<DataverseName,Identifier> adapterName = null;
2167 boolean ifExists = false;
2168}
2169{
2170 adapterName = QualifiedName() ifExists = IfExists()
2171 {
2172 AdapterDropStatement stmt = new AdapterDropStatement(adapterName.first, adapterName.second.getValue(), ifExists);
2173 return addSourceLocation(stmt, startStmtToken);
2174 }
2175}
2176
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002177FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
2178{
2179 FunctionDropStatement stmt = null;
2180}
2181{
2182 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
2183 {
2184 return stmt;
2185 }
2186}
2187
2188FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
2189{
2190 FunctionSignature funcSig = null;
2191 boolean ifExists = false;
2192}
2193{
2194 funcSig = FunctionSignature() ifExists = IfExists()
2195 {
2196 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
2197 return addSourceLocation(stmt, startStmtToken);
2198 }
2199}
2200
2201FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
2202{
2203 FeedDropStatement stmt = null;
2204}
2205{
2206 <FEED> stmt = DropFeedSpecification(startStmtToken)
2207 {
2208 return stmt;
2209 }
2210}
2211
2212FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
2213{
2214 Pair<DataverseName,Identifier> pairId = null;
2215 boolean ifExists = false;
2216}
2217{
2218 pairId = QualifiedName() ifExists = IfExists()
2219 {
2220 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
2221 return addSourceLocation(stmt, startStmtToken);
2222 }
2223}
2224
2225FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
2226{
2227 FeedPolicyDropStatement stmt = null;
2228}
2229{
2230 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
2231 {
2232 return stmt;
2233 }
2234}
2235
2236FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
2237{
2238 Pair<DataverseName,Identifier> pairId = null;
2239 boolean ifExists = false;
2240}
2241{
2242 pairId = QualifiedName() ifExists = IfExists()
2243 {
2244 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
2245 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002246 }
2247}
2248
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002249SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
2250{
2251 SynonymDropStatement stmt = null;
2252}
2253{
2254 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
2255 {
2256 return stmt;
2257 }
2258}
2259
2260SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
2261{
2262 Pair<DataverseName,Identifier> pairId = null;
2263 boolean ifExists = false;
2264}
2265{
2266 pairId = QualifiedName() ifExists = IfExists()
2267 {
2268 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
2269 return addSourceLocation(stmt, startStmtToken);
2270 }
2271}
2272
Yingyi Bu391f09e2015-10-29 13:49:39 -07002273boolean IfExists() throws ParseException :
2274{
2275}
2276{
2277 ( LOOKAHEAD(1) <IF> <EXISTS>
2278 {
2279 return true;
2280 }
2281 )?
2282 {
2283 return false;
2284 }
2285}
2286
2287InsertStatement InsertStatement() throws ParseException:
2288{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002289 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002290 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002291 VariableExpr var = null;
2292 Query query = null;
2293 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002294}
2295{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002296 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002297 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002298 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002299 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002300 if (var == null && returnExpression != null) {
2301 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2302 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002303 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08002304 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002305 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2306 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002307 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002308 }
2309}
2310
2311UpsertStatement UpsertStatement() throws ParseException:
2312{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002313 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002314 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002315 VariableExpr var = null;
2316 Query query = null;
2317 Expression returnExpression = null;
2318}
2319{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002320 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002321 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002322 ( <RETURNING> returnExpression = Expression())?
2323 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002324 if (var == null && returnExpression != null) {
2325 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2326 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002327 }
2328 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002329 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2330 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002331 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002332 }
2333}
2334
2335DeleteStatement DeleteStatement() throws ParseException:
2336{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002337 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002338 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002339 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002340 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002341}
2342{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002343 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002344 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002345 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07002346 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002347 if (var == null) {
2348 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2349 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002350 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002351 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07002352 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002353 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002354 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002355}
2356
2357UpdateStatement UpdateStatement() throws ParseException:
2358{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002359 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002360 VariableExpr vars;
2361 Expression target;
2362 Expression condition;
2363 UpdateClause uc;
2364 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
2365}
2366{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002367 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002368 <WHERE> condition = Expression()
2369 <LEFTPAREN> (uc = UpdateClause()
2370 {
2371 ucs.add(uc);
2372 }
2373 (<COMMA> uc = UpdateClause()
2374 {
2375 ucs.add(uc);
2376 }
2377 )*) <RIGHTPAREN>
2378 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002379 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002380 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002381 }
2382}
2383
2384UpdateClause UpdateClause() throws ParseException:
2385{
2386 Expression target = null;
2387 Expression value = null ;
2388 InsertStatement is = null;
2389 DeleteStatement ds = null;
2390 UpdateStatement us = null;
2391 Expression condition = null;
2392 UpdateClause ifbranch = null;
2393 UpdateClause elsebranch = null;
2394}
2395{
2396 (<SET> target = Expression() <EQ> value = Expression()
2397 | is = InsertStatement()
2398 | ds = DeleteStatement()
2399 | us = UpdateStatement()
2400 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
2401 <THEN> ifbranch = UpdateClause()
2402 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
2403 {
2404 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
2405 }
2406 )
2407}
2408
2409Statement SetStatement() throws ParseException:
2410{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002411 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002412 String pn = null;
2413 String pv = null;
2414}
2415{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002416 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002417 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002418 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002419 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002420 }
2421}
2422
2423Statement WriteStatement() throws ParseException:
2424{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002425 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002426 String nodeName = null;
2427 String fileName = null;
2428 Query query;
2429 String writerClass = null;
2430 Pair<Identifier,Identifier> nameComponents = null;
2431}
2432{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002433 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07002434 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002435 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002436 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002437 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002438 }
2439}
2440
2441LoadStatement LoadStatement() throws ParseException:
2442{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002443 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002444 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002445 Identifier datasetName = null;
2446 boolean alreadySorted = false;
2447 String adapterName;
2448 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002449 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002450}
2451{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002452 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002453 {
2454 dataverseName = nameComponents.first;
2455 datasetName = nameComponents.second;
2456 }
2457 <USING> adapterName = AdapterName() properties = Configuration()
2458 (<PRESORTED>
2459 {
2460 alreadySorted = true;
2461 }
2462 )?
2463 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002464 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
2465 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002466 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002467 }
2468}
2469
2470
2471String AdapterName() throws ParseException :
2472{
2473 String adapterName = null;
2474}
2475{
2476 adapterName = Identifier()
2477 {
2478 return adapterName;
2479 }
2480}
2481
2482Statement CompactStatement() throws ParseException:
2483{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002484 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002485 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002486}
2487{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002488 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002489 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002490 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002491 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002492 }
2493}
2494
Yingyi Buab817482016-08-19 21:29:31 -07002495Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002496{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002497 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002498 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002499}
2500{
2501 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002502 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
2503 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
2504 | <START> { startToken = token; } stmt = StartStatement(startToken)
2505 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07002506 )
2507 {
2508 return stmt;
2509 }
2510}
2511
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002512Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002513{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002514 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002515 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002516}
2517{
2518 <FEED> feedNameComponents = QualifiedName()
2519 {
2520 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002521 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002522 }
2523}
2524
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002525AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002526{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002527 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002528 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002529}
2530{
2531 <FEED> feedNameComponents = QualifiedName()
2532 {
2533 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002534 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002535 }
2536}
2537
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002538AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002539{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002540 Pair<DataverseName,Identifier> feedNameComponents = null;
2541 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002542
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002543 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002544}
2545{
2546 (
2547 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002548 {
2549 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
2550 }
2551 )
Yingyi Buab817482016-08-19 21:29:31 -07002552 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002553 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002554 }
2555}
2556
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002557AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002558{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002559 Pair<DataverseName,Identifier> feedNameComponents = null;
2560 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002561
2562 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002563 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002564 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002565 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002566 String whereClauseBody = null;
2567 WhereClause whereClause = null;
2568 Token beginPos = null;
2569 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002570}
2571{
2572 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002573 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002574 (ApplyFunction(appliedFunctions))?
2575 (policy = GetPolicy())?
2576 (
2577 <WHERE>
2578 {
2579 beginPos = token;
2580 whereClause = new WhereClause();
2581 Expression whereExpr;
2582 }
2583 whereExpr = Expression()
2584 {
2585 whereClause.setWhereExpr(whereExpr);
2586 }
2587 )?
2588 {
2589 if (whereClause != null) {
2590 endPos = token;
2591 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2592 }
2593 }
Yingyi Buab817482016-08-19 21:29:31 -07002594 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002595 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002596 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002597 }
2598 )
2599 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002600 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002601 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002602}
2603
2604Map<String,String> Configuration() throws ParseException :
2605{
2606 Map<String,String> configuration = new LinkedHashMap<String,String>();
2607 Pair<String, String> keyValuePair = null;
2608}
2609{
2610 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2611 {
2612 configuration.put(keyValuePair.first, keyValuePair.second);
2613 }
2614 ( <COMMA> keyValuePair = KeyValuePair()
2615 {
2616 configuration.put(keyValuePair.first, keyValuePair.second);
2617 }
2618 )* )? <RIGHTPAREN>
2619 {
2620 return configuration;
2621 }
2622}
2623
2624Pair<String, String> KeyValuePair() throws ParseException:
2625{
2626 String key;
2627 String value;
2628}
2629{
Ali Alsulimane2986012020-05-14 18:14:22 -07002630 <LEFTPAREN> key = ConstantString()
2631 <EQ> ( value = ConstantString() | (<TRUE> | <FALSE>) {value = token.image.toLowerCase();} )
2632 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002633 {
2634 return new Pair<String, String>(key, value);
2635 }
2636}
2637
2638Map<String,String> Properties() throws ParseException:
2639{
2640 Map<String,String> properties = new HashMap<String,String>();
2641 Pair<String, String> property;
2642}
2643{
2644 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
2645 {
2646 properties.put(property.first, property.second);
2647 }
2648 ( <COMMA> property = Property()
2649 {
2650 properties.put(property.first, property.second);
2651 }
2652 )* <RIGHTPAREN> )?
2653 {
2654 return properties;
2655 }
2656}
2657
2658Pair<String, String> Property() throws ParseException:
2659{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002660 String key = null;
2661 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002662}
2663{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002664 (key = Identifier() | key = StringLiteral())
2665 <EQ>
2666 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002667 {
2668 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002669 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002670 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002671 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002672 }
2673 }
2674 )
2675 {
2676 return new Pair<String, String>(key.toUpperCase(), value);
2677 }
2678}
2679
Glenn58329202021-04-09 12:03:46 -07002680IndexedTypeExpression IndexedTypeExpr(boolean allowQues) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002681{
2682 TypeExpression typeExpr = null;
Glenn58329202021-04-09 12:03:46 -07002683 boolean isUnknownable = !allowQues;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002684}
2685{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002686 typeExpr = TypeExpr(false)
Glenn58329202021-04-09 12:03:46 -07002687 ( <QUES>
2688 {
2689 if (!allowQues) {
2690 throw new SqlppParseException(getSourceLocation(token), "'?' quantifier is illegal for the type expression.");
2691 }
2692 isUnknownable = true;
2693 }
2694 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002695 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002696 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002697 }
2698}
2699
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002700TypeExpression TypeExpr(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002701{
2702 TypeExpression typeExpr = null;
2703}
2704{
2705 (
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002706 typeExpr = TypeReference()
2707 | typeExpr = OrderedListTypeDef(allowRecordTypeDef)
2708 | typeExpr = UnorderedListTypeDef(allowRecordTypeDef)
2709 | typeExpr = RecordTypeDef() {
2710 if (!allowRecordTypeDef) {
2711 throw new SqlppParseException(typeExpr.getSourceLocation(), "Unexpected record type declaration");
2712 }
2713 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002714 )
2715 {
2716 return typeExpr;
2717 }
2718}
2719
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002720RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
2721{
2722 RecordTypeDefinition.RecordKind recordKind = null;
2723}
2724{
2725 (
2726 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
2727 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
2728 )
2729 {
2730 return recordKind;
2731 }
2732}
2733
Yingyi Bu391f09e2015-10-29 13:49:39 -07002734RecordTypeDefinition RecordTypeDef() throws ParseException:
2735{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002736 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002737 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002738 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002739}
2740{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002741 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002742 <LEFTBRACE>
2743 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002744 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002745 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
2746 if (hintToken != null) {
2747 String hintParams = hintToken.hintParams;
2748 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
2749 if (splits == null || splits.length != 4) {
2750 throw new SqlppParseException(getSourceLocation(hintToken),
2751 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002752 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002753 if (!splits[0].equals("int")) {
2754 throw new SqlppParseException(getSourceLocation(hintToken),
2755 "The only supported type for gen-fields is int.");
2756 }
2757 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
2758 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
2759 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002760 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002761 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002762 (
2763 RecordField(recType)
2764 ( <COMMA> RecordField(recType) )*
2765 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002766 <RIGHTBRACE>
2767 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002768 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002769 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002770 }
2771}
2772
2773void RecordField(RecordTypeDefinition recType) throws ParseException:
2774{
2775 String fieldName;
2776 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002777 boolean nullable = false, missable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002778}
2779{
2780 fieldName = Identifier()
2781 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002782 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
2783 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
2784 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
2785 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002786 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002787 <COLON> type = TypeExpr(true) ( <QUES> { nullable = true; missable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002788 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002789 recType.addField(fieldName, type, nullable, missable, rfdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002790 }
2791}
2792
2793TypeReferenceExpression TypeReference() throws ParseException:
2794{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002795 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002796}
2797{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002798 id = QualifiedName()
2799 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07002800 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
2801 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002802 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002803
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002804 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002805 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002806 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002807}
2808
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002809OrderedListTypeDefinition OrderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002810{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002811 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002812 TypeExpression type = null;
2813}
2814{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002815 <LEFTBRACKET> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002816 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002817 <RIGHTBRACKET>
2818 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002819 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002820 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002821 }
2822}
2823
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002824UnorderedListTypeDefinition UnorderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002825{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002826 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002827 TypeExpression type = null;
2828}
2829{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002830 <LEFTDBLBRACE> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002831 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002832 <RIGHTDBLBRACE>
2833 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002834 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002835 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002836 }
2837}
2838
2839FunctionName FunctionName() throws ParseException:
2840{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002841 Triple<List<String>, Token, Token> prefix = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002842 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002843}
2844{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002845 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
2846 // that copy must be kept in sync with this code
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002847 prefix = MultipartIdentifierWithHints(
2848 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.RANGE_HINT,
2849 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
2850 )
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002851 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002852 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002853 Token startToken = prefix.second;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002854 FunctionName result = new FunctionName();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002855 result.sourceLoc = getSourceLocation(startToken);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002856 result.hintToken = prefix.third;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002857 List<String> list = prefix.first;
2858 int ln = list.size();
2859 String last = list.get(ln - 1);
2860 if (suffix == null) {
2861 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
2862 // no library name
2863 result.function = last;
2864 } else {
2865 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
2866 // suffix = func_name
2867 result.library = last;
2868 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002869 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002870 if (ln > 1) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002871 result.dataverse = createDataverseName(list, 0, ln - 1, startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002872 } else {
2873 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002874 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002875
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002876 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
2877 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002878 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002879 return result;
2880 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002881}
2882
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002883Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002884{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002885 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002886}
2887{
2888 name = QualifiedName()
2889 {
2890 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002891 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002892 }
2893 return name;
2894 }
2895}
2896
2897String Identifier() throws ParseException:
2898{
2899 String lit = null;
2900}
2901{
2902 (<IDENTIFIER>
2903 {
2904 return token.image;
2905 }
2906 | lit = QuotedString()
2907 {
2908 return lit;
2909 }
2910 )
2911}
2912
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002913List<String> ParenthesizedIdentifierList() throws ParseException:
2914{
2915 List<String> list = new ArrayList<String>();
2916 String ident = null;
2917}
2918{
2919 <LEFTPAREN>
2920 ident = Identifier() { list.add(ident); }
2921 ( <COMMA> ident = Identifier() { list.add(ident); } )*
2922 <RIGHTPAREN>
2923 {
2924 return list;
2925 }
2926}
2927
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002928Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002929{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002930 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002931 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002932}
2933{
2934 fieldList = NestedField()
Glenn58329202021-04-09 12:03:46 -07002935 ( <COLON> fieldType = IndexedTypeExpr(true) )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002936 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002937 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
2938 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002939 }
2940}
2941
Yingyi Buc9bfe252016-03-01 00:02:40 -08002942Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002943{
2944 List<String> exprList = new ArrayList<String>();
2945 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002946 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002947 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002948}
2949{
2950 lit = Identifier()
2951 {
Yingyi Bub9169b62016-02-26 21:21:49 -08002952 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002953 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08002954 }
2955 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08002956 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08002957 <LEFTPAREN><RIGHTPAREN>
2958 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002959 if(!lit.equalsIgnoreCase("meta")){
2960 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08002961 }
2962 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002963 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08002964 }
2965 )?
2966 {
2967 if(!meetParens){
2968 exprList.add(lit);
2969 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002970 }
2971 (<DOT>
2972 lit = Identifier()
2973 {
2974 exprList.add(lit);
2975 }
2976 )*
2977 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08002978 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002979 }
2980}
2981
Yingyi Bu6d57e492016-06-06 21:24:42 -07002982String ConstantString() throws ParseException:
2983{
2984 String value = null;
2985}
2986{
2987 (value = QuotedString() | value = StringLiteral())
2988 {
2989 return value;
2990 }
2991}
2992
Yingyi Bu391f09e2015-10-29 13:49:39 -07002993String QuotedString() throws ParseException:
2994{
2995}
2996{
2997 <QUOTED_STRING>
2998 {
2999 return removeQuotesAndEscapes(token.image);
3000 }
3001}
3002
Yingyi Bu391f09e2015-10-29 13:49:39 -07003003String StringLiteral() throws ParseException:
3004{
3005}
3006{
3007 <STRING_LITERAL>
3008 {
3009 return removeQuotesAndEscapes(token.image);
3010 }
3011}
3012
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003013Triple<List<String>, Token, Token> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003014{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003015 Triple<List<String>, Token, Token> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003016}
3017{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003018 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003019 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003020 return result;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003021 }
3022}
3023
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003024Triple<List<String>, Token, Token> MultipartIdentifierWithHints(SqlppHint... expectedHints)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003025 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003026{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003027 List<String> list = new ArrayList<String>();
3028 SourceLocation sourceLoc = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003029 Token startToken, hintToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003030 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003031}
3032{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003033 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003034 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003035 list.add(item);
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003036 startToken = token;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003037 if (expectedHints != null && expectedHints.length > 0) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003038 hintToken = fetchHint(token, expectedHints);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003039 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003040 }
3041 (<DOT> item = Identifier() { list.add(item); } )*
3042 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003043 return new Triple<List<String>, Token, Token>(list, startToken, hintToken);
3044 }
3045}
3046
3047DataverseName DataverseName() throws ParseException:
3048{
3049 Triple<List<String>, Token, Token> ident = null;
3050}
3051{
3052 ident = MultipartIdentifier()
3053 {
3054 List<String> list = ident.first;
3055 Token startToken = ident.second;
3056 return createDataverseName(list, 0, list.size(), startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003057 }
3058}
3059
3060Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
3061{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003062 Triple<List<String>, Token, Token> ident = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003063}
3064{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003065 ident = MultipartIdentifier()
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003066 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003067 List<String> list = ident.first;
3068 Token startToken = ident.second;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003069 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003070 DataverseName id1 = len > 1 ? createDataverseName(list, 0, len - 1, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003071 Identifier id2 = new Identifier(list.get(len - 1));
3072 return new Pair<DataverseName,Identifier>(id1, id2);
3073 }
3074}
3075
3076Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
3077{
3078 List<String> list = new ArrayList<String>();
3079 String item = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003080 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003081}
3082{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003083 item = Identifier() { list.add(item); startToken = token; }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003084 (<DOT> item = Identifier() { list.add(item); } )+
3085 {
3086 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003087 DataverseName id1 = len > 2 ? createDataverseName(list, 0, len - 2, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003088 Identifier id2 = new Identifier(list.get(len - 2));
3089 Identifier id3 = new Identifier(list.get(len - 1));
3090 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003091 }
3092}
3093
3094FunctionDecl FunctionDeclaration() throws ParseException:
3095{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003096 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003097 String functionName;
Dmitry Lychagin541652082020-11-09 14:04:53 -08003098 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003099 Expression funcBody;
3100 createNewScope();
3101}
3102{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003103 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07003104 functionName = Identifier()
Dmitry Lychagin541652082020-11-09 14:04:53 -08003105 paramsWithArity = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07003106 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003107 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07003108 <RIGHTBRACE>
3109 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08003110 int arity = paramsWithArity.first;
3111 List<Pair<VarIdentifier,TypeExpression>> paramList = paramsWithArity.second;
3112 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, arity);
Xikui Wang3de700a2018-03-15 16:32:55 -07003113 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003114 ensureNoTypeDeclsInFunction(functionName, paramList, null, startToken);
3115 List<VarIdentifier> params = new ArrayList<VarIdentifier>(paramList.size());
3116 for (Pair<VarIdentifier,TypeExpression> p: paramList) {
3117 params.add(p.getFirst());
Ian Maxon38fe9402020-01-29 19:27:40 -08003118 }
Dmitry Lychagin9ba74872021-04-05 14:38:20 -07003119 FunctionDecl stmt = new FunctionDecl(signature, params, funcBody, false);
Xikui Wang3de700a2018-03-15 16:32:55 -07003120 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003121 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07003122 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003123}
3124
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003125Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003126{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003127 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003128 Expression expr;
3129}
3130{
3131 (
3132 expr = Expression()
3133 |
3134 expr = SelectExpression(false)
3135 )
3136 {
3137 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003138 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003139 return query;
3140 }
3141}
3142
3143
Yingyi Bu391f09e2015-10-29 13:49:39 -07003144Expression Expression():
3145{
3146 Expression expr = null;
3147 Expression exprP = null;
3148}
3149{
3150(
3151 LOOKAHEAD(2)
3152 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003153 | expr = QuantifiedExpression()
3154)
3155 {
3156 return (exprP==null) ? expr : exprP;
3157 }
3158}
3159
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003160Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003161{
3162 OperatorExpr op = null;
3163 Expression operand = null;
3164}
3165{
3166 operand = AndExpr()
3167 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003168 <OR>
3169 {
3170 if (op == null) {
3171 op = new OperatorExpr();
3172 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07003173 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003174 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003175 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003176 try{
3177 op.addOperator(token.image.toLowerCase());
3178 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003179 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003180 }
Xikui Wang3de700a2018-03-15 16:32:55 -07003181 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003182
Xikui Wang3de700a2018-03-15 16:32:55 -07003183 operand = AndExpr()
3184 {
3185 op.addOperand(operand);
3186 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003187
3188 )*
3189
3190 {
3191 return op==null? operand: op;
3192 }
3193}
3194
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003195Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003196{
3197 OperatorExpr op = null;
3198 Expression operand = null;
3199}
3200{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003201 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003202 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003203 <AND>
3204 {
3205 if (op == null) {
3206 op = new OperatorExpr();
3207 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003208 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003209 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003210 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003211 try{
3212 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003213 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003214 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003215 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003216 }
3217
Yingyi Bu196db5d2016-07-15 19:07:20 -07003218 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003219 {
3220 op.addOperand(operand);
3221 }
3222
3223 )*
3224
3225 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003226 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003227 }
3228}
3229
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003230Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07003231{
3232 Expression inputExpr;
3233 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003234 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07003235}
3236{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003237 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07003238 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003239 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003240 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003241 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003242 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003243 } else {
3244 return inputExpr;
3245 }
3246 }
3247}
Yingyi Bu391f09e2015-10-29 13:49:39 -07003248
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003249Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003250{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003251 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003252 OperatorExpr op = null;
3253 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003254 IExpressionAnnotation annotation = null;
3255}
3256{
Yingyi Bu6c638342016-09-02 17:54:34 -07003257 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003258
3259 (
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003260 LOOKAHEAD(2)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN>)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003261 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003262 Token hintToken = fetchHint(token,
3263 SqlppHint.HASH_BROADCAST_JOIN_HINT, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
3264 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
3265 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003266 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003267 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003268 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003269 String operator = token.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07003270 if (operator.equals("<>")){
3271 operator = "!=";
3272 }
3273 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003274 operator = "not_" + operator;
3275 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003276 if (op == null) {
3277 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07003278 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003279 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003280 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003281 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003282 try{
3283 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003284 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003285 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003286 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003287 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003288
Yingyi Bu6c638342016-09-02 17:54:34 -07003289 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07003290 {
Shiva2ea73232019-10-09 18:04:05 -07003291 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07003292 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003293 )?
3294
3295 {
3296 if (annotation != null) {
3297 op.addHint(annotation);
3298 }
3299 return op==null? operand: op;
3300 }
3301}
3302
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003303Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07003304{
3305 boolean not = false;
3306 OperatorExpr op = null;
3307 Expression operand = null;
3308 IExpressionAnnotation annotation = null;
3309}
3310{
3311 operand = IsExpr()
3312 (
3313 LOOKAHEAD(2)
3314 (<NOT> { not = true; })? <BETWEEN>
3315 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003316 Token hintToken = fetchHint(token,
3317 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT,
3318 SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
3319 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003320 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003321 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu6c638342016-09-02 17:54:34 -07003322 }
3323 String operator = token.image.toLowerCase();
3324 if(not){
3325 operator = "not_" + operator;
3326 }
3327 if (op == null) {
3328 op = new OperatorExpr();
3329 op.addOperand(operand);
3330 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003331 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07003332 }
3333 try{
3334 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003335 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003336 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07003337 }
3338 }
3339
3340 operand = IsExpr()
3341 {
3342 op.addOperand(operand);
3343 }
3344
3345 <AND>
3346 operand = IsExpr()
3347 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08003348 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07003349 op.addOperand(operand);
3350 }
3351 )?
3352
3353 {
3354 if (annotation != null) {
3355 op.addHint(annotation);
3356 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003357 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07003358 }
3359}
3360
Yingyi Budaa549c2016-06-28 22:30:52 -07003361Expression IsExpr() throws ParseException:
3362{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003363 Token notToken = null;
3364 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003365 Expression operand = null;
3366 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003367 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003368}
3369{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003370 operand = LikeExpr()
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003371 ( <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003372 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003373 (
3374 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
3375 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08003376 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07003377 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003378 )
Yingyi Budaa549c2016-06-28 22:30:52 -07003379 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003380 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07003381 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003382 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003383 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003384 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07003385 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003386 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07003387 }
3388 }
3389 )?
3390 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003391 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07003392 }
3393}
3394
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003395Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003396{
3397 boolean not = false;
3398 OperatorExpr op = null;
3399 Expression operand = null;
3400}
3401{
3402 operand = ConcatExpr()
3403 (
3404 LOOKAHEAD(2)
3405 (<NOT> { not = true; })? <LIKE>
3406 {
3407 op = new OperatorExpr();
3408 op.addOperand(operand);
3409 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003410 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003411
3412 String operator = token.image.toLowerCase();
3413 if (not) {
3414 operator = "not_" + operator;
3415 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003416 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003417 op.addOperator(operator);
3418 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003419 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003420 }
3421 }
3422
3423 operand = ConcatExpr()
3424 {
3425 op.addOperand(operand);
3426 }
3427 )?
3428
3429 {
3430 return op == null ? operand : op;
3431 }
3432}
3433
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003434Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003435{
3436 OperatorExpr op = null;
3437 Expression operand = null;
3438}
3439{
3440 operand = AddExpr()
3441 (
3442 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003443 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003444 {
3445 if (op == null) {
3446 op = new OperatorExpr();
3447 op.addOperand(operand);
3448 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003449 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003450 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003451 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003452 }
3453 operand = AddExpr()
3454 {
3455 op.addOperand(operand);
3456 }
3457 )*
3458
3459 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003460 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003461 }
3462}
Yingyi Budaa549c2016-06-28 22:30:52 -07003463
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003464Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003465{
3466 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003467 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003468 Expression operand = null;
3469}
3470{
3471 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07003472 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003473 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003474 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003475 {
3476 if (op == null) {
3477 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003478 op.addOperand(operand);
3479 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003480 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003481 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003482 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003483 }
3484
3485 operand = MultExpr()
3486 {
3487 op.addOperand(operand);
3488 }
3489 )*
3490
3491 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003492 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003493 }
3494}
3495
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003496Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003497{
3498 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003499 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003500 Expression operand = null;
3501}
3502{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003503 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003504
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003505 ( (
3506 <MUL> { opType = OperatorType.MUL; } |
3507 <DIVIDE> { opType = OperatorType.DIVIDE; } |
3508 <DIV> { opType = OperatorType.DIV; } |
3509 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
3510 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003511 {
3512 if (op == null) {
3513 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003514 op.addOperand(operand);
3515 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003516 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003517 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003518 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003519 }
3520 operand = ExponentExpr()
3521 {
3522 op.addOperand(operand);
3523 }
3524 )*
3525
3526 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003527 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003528 }
3529}
3530
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003531Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003532{
3533 OperatorExpr op = null;
3534 Expression operand = null;
3535}
3536{
3537 operand = UnaryExpr()
3538 (<CARET>
3539 {
3540 if (op == null) {
3541 op = new OperatorExpr();
3542 op.addOperand(operand);
3543 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003544 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003545 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003546 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003547 }
3548 operand = UnaryExpr()
3549 {
3550 op.addOperand(operand);
3551 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003552 )?
3553 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003554 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003555 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003556}
3557
3558Expression UnaryExpr() throws ParseException:
3559{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003560 boolean not = false;
3561 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003562 Expression expr = null;
3563}
Yingyi Budaa549c2016-06-28 22:30:52 -07003564{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003565 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003566 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003567 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003568 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003569 exprType = "not_" + exprType;
3570 }
3571 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003572 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003573 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003574 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003575 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003576 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07003577 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003578 }
3579 )?
3580
3581 expr = ValueExpr()
3582 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003583 if (uexpr == null) {
3584 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003585 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003586 uexpr.setExpr(expr);
3587 return uexpr;
3588 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003589 }
3590}
3591
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003592Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003593{
3594 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003595 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003596}
3597{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003598 expr = PrimaryExpr()
3599 (
3600 accessor = FieldAccessor(accessor != null ? accessor : expr)
3601 |
3602 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003603 )*
3604 {
3605 return accessor == null ? expr : accessor;
3606 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003607}
3608
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003609FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003610{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003611 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003612 String ident = null;
3613}
3614{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003615 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003616 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003617 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003618 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003619 }
3620}
3621
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003622AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003623{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003624 Token startToken = null;
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003625 boolean star = false, slice = false;
3626 Expression expr1 = null, expr2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003627}
3628{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003629 <LEFTBRACKET> { startToken = token; }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003630 (
3631 <MUL> { star = true; }
3632 |
3633 ( expr1 = Expression() ( <COLON> { slice = true; } ( expr2 = Expression() )? )? )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003634 )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003635 <RIGHTBRACKET>
3636 {
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003637 if (expr1 != null && expr1.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3638 ensureIntegerLiteral( (LiteralExpr) expr1, "Index");
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003639 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003640 if (expr2 != null && expr2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3641 ensureIntegerLiteral( (LiteralExpr) expr2, "Index");
3642 }
3643 AbstractAccessor resultAccessor;
3644 if (slice) {
3645 resultAccessor = new ListSliceExpression(inputExpr, expr1, expr2);
3646 } else if (star) {
3647 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.STAR, null);
3648 } else {
3649 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.ELEMENT, expr1);
3650 }
3651 return addSourceLocation(resultAccessor, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003652 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003653}
3654
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003655Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003656{
3657 Expression expr = null;
3658}
3659{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003660 (
3661 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07003662 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003663 | expr = Literal()
3664 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003665 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003666 | expr = ListConstructor()
3667 | expr = RecordConstructor()
3668 | expr = ParenthesizedExpression()
3669 )
3670 {
3671 return expr;
3672 }
3673}
3674
3675Expression Literal() throws ParseException:
3676{
3677 LiteralExpr lit = new LiteralExpr();
3678 String str = null;
3679}
3680{
3681 ( str = StringLiteral()
3682 {
3683 lit.setValue(new StringLiteral(str));
3684 }
3685 | <INTEGER_LITERAL>
3686 {
Till Westmann68c6a992016-09-30 00:19:12 -07003687 try {
3688 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
3689 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003690 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003691 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003692 }
3693 | <FLOAT_LITERAL>
3694 {
Till Westmann68c6a992016-09-30 00:19:12 -07003695 try {
3696 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
3697 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003698 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003699 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003700 }
3701 | <DOUBLE_LITERAL>
3702 {
Till Westmann68c6a992016-09-30 00:19:12 -07003703 try {
3704 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
3705 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003706 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003707 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003708 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07003709 | <MISSING>
3710 {
3711 lit.setValue(MissingLiteral.INSTANCE);
3712 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003713 | <NULL>
3714 {
3715 lit.setValue(NullLiteral.INSTANCE);
3716 }
3717 | <TRUE>
3718 {
3719 lit.setValue(TrueLiteral.INSTANCE);
3720 }
3721 | <FALSE>
3722 {
3723 lit.setValue(FalseLiteral.INSTANCE);
3724 }
3725 )
3726 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003727 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003728 }
3729}
3730
Yingyi Bu391f09e2015-10-29 13:49:39 -07003731VariableExpr VariableRef() throws ParseException:
3732{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003733 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003734}
3735{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003736 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003737 {
3738 Identifier ident = lookupSymbol(id);
3739 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003740 throw new SqlppParseException(getSourceLocation(token),
3741 "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 -07003742 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003743 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003744 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003745 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003746 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003747 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07003748 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003749 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003750 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003751 }
3752}
3753
Yingyi Bu391f09e2015-10-29 13:49:39 -07003754VariableExpr Variable() throws ParseException:
3755{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003756 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003757}
3758{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003759 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003760 {
3761 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003762 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
3763 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07003764 varExp.setIsNewVar(false);
3765 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003766 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003767 }
3768}
3769
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003770String VariableIdentifier() throws ParseException:
3771{
3772 String id = null;
3773}
3774{
3775 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3776 {
3777 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
3778 }
3779}
3780
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003781Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
3782{
3783 VariableExpr var = null;
3784 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
3785}
3786{
3787 var = Variable()
3788 ( LOOKAHEAD(1)
3789 {
3790 VariableExpr fieldVarExpr = null;
3791 String fieldIdentifierStr = null;
3792 }
3793 <LEFTPAREN>
3794 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3795 {
3796 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3797 }
3798 (<COMMA>
3799 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3800 {
3801 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3802 }
3803 )*
3804 <RIGHTPAREN>
3805 )?
3806 {
3807 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
3808 }
3809}
3810
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003811VariableExpr ExternalVariableRef() throws ParseException:
3812{
3813 String name = null;
3814}
3815{
3816 (
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08003817 <DOLLAR_IDENTIFIER> { name = token.image.substring(1); }
3818 | <DOLLAR_INTEGER_LITERAL> { name = token.image.substring(1); }
3819 | <DOLLAR_QUOTED_STRING> { name = removeQuotesAndEscapes(token.image.substring(1)); }
3820 | <QUES> { name = String.valueOf(++externalVarCounter); }
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003821 )
3822 {
3823 String idName = SqlppVariableUtil.toExternalVariableName(name);
3824 VarIdentifier id = new VarIdentifier(idName);
3825 VariableExpr varExp = new VariableExpr(id);
3826 return addSourceLocation(varExp, token);
3827 }
3828}
3829
Yingyi Bu391f09e2015-10-29 13:49:39 -07003830Expression ListConstructor() throws ParseException:
3831{
3832 Expression expr = null;
3833}
3834{
3835 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003836 expr = OrderedListConstructor() |
3837 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003838 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003839 {
3840 return expr;
3841 }
3842}
3843
Yingyi Bu391f09e2015-10-29 13:49:39 -07003844ListConstructor OrderedListConstructor() throws ParseException:
3845{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003846 Token startToken = null;
3847 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003848}
3849{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003850 <LEFTBRACKET> { startToken = token; }
3851 exprList = ExpressionList()
3852 <RIGHTBRACKET>
3853 {
3854 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003855 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003856 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003857}
3858
3859ListConstructor UnorderedListConstructor() throws ParseException:
3860{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003861 Token startToken = null;
3862 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003863}
3864{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003865 <LEFTDBLBRACE> { startToken = token; }
3866 exprList = ExpressionList()
3867 <RIGHTDBLBRACE>
3868 {
3869 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003870 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003871 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003872}
3873
3874List<Expression> ExpressionList() throws ParseException:
3875{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003876 Expression expr = null;
3877 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003878}
3879{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003880 (
3881 expr = Expression()
3882 {
3883 exprList.add(expr);
3884 }
3885 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07003886 {
3887 exprList.add(expr);
3888 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003889 )*
3890 )?
3891 {
3892 return exprList;
3893 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003894}
3895
Yingyi Bu391f09e2015-10-29 13:49:39 -07003896RecordConstructor RecordConstructor() throws ParseException:
3897{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003898 Token startToken = null;
3899 FieldBinding fb = null;
3900 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003901}
3902{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003903 <LEFTBRACE> { startToken = token; }
3904 (
3905 fb = FieldBinding() { fbList.add(fb); }
3906 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
3907 )?
3908 <RIGHTBRACE>
3909 {
3910 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003911 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003912 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003913}
3914
3915FieldBinding FieldBinding() throws ParseException:
3916{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003917 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003918}
3919{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003920 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003921 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003922 if (right == null) {
3923 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
3924 if (generatedIdentifier == null) {
3925 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
3926 }
3927 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07003928 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
3929 generatedNameExpr.setSourceLocation(left.getSourceLocation());
3930 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003931 } else {
3932 return new FieldBinding(left, right);
3933 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003934 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003935}
3936
Yingyi Bu391f09e2015-10-29 13:49:39 -07003937Expression FunctionCallExpr() throws ParseException:
3938{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003939 List<Expression> argList = new ArrayList<Expression>();
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003940 Expression argExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003941 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07003942 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003943 boolean distinct = false;
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003944 Expression filterExpr = null;
3945 WindowExpression windowExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003946}
3947{
3948 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003949 <LEFTPAREN> (
3950 ( <DISTINCT> { distinct = true; } )?
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003951 ( argExpr = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003952 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003953 if (star) {
3954 if (funcName.function.equalsIgnoreCase(BuiltinFunctions.SCALAR_COUNT.getName())) {
3955 argExpr = new LiteralExpr(new LongIntegerLiteral(1L));
3956 } else {
3957 throw new SqlppParseException(getSourceLocation(token),
3958 "The parameter * can only be used in " + BuiltinFunctions.SCALAR_COUNT.getName() + "().");
Yingyi Buf4d09842016-08-26 00:03:52 -07003959 }
Yingyi Buf4d09842016-08-26 00:03:52 -07003960 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003961 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003962 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003963 (<COMMA> argExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003964 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003965 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003966 }
3967 )*)? <RIGHTPAREN>
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003968
3969 {
3970 String name = funcName.function;
3971 if (distinct) {
3972 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
3973 }
3974 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
3975 int arity = argList.size();
3976 FunctionSignature signature = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
3977 if (signature == null) {
3978 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
3979 }
3980 }
3981
3982 ( <FILTER> <LEFTPAREN> <WHERE> filterExpr = Expression() <RIGHTPAREN> )?
3983
3984 ( LOOKAHEAD(5) windowExpr = WindowExpr(signature, argList, filterExpr) )?
3985
3986 {
3987 if (windowExpr != null) {
3988 return windowExpr;
3989 } else {
3990 CallExpr callExpr = new CallExpr(signature, argList, filterExpr);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06003991 if (funcName.hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003992 IExpressionAnnotation annotation = parseExpressionAnnotation(funcName.hintToken);
3993 if (annotation != null) {
3994 callExpr.addHint(annotation);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003995 }
3996 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003997 FunctionMapUtil.normalizedListInputFunctions(callExpr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003998 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003999 return callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004000 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004001 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004002}
4003
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004004WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr)
4005 throws ParseException:
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004006{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004007 WindowExpression windowExpr = null;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004008 Boolean fromLast = null, ignoreNulls = null;
4009}
4010{
4011 (
4012 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
4013 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
4014 <FROM> <IDENTIFIER>
4015 {
4016 if (isToken(FIRST)) {
4017 fromLast = false;
4018 } else if (isToken(LAST)) {
4019 fromLast = true;
4020 } else {
4021 throw createUnexpectedTokenError();
4022 }
4023 }
4024 )?
4025 (
4026 // ( RESPECT | IGNORE ) NULLS OVER
4027 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
4028 <IDENTIFIER>
4029 {
4030 if (isToken(RESPECT)) {
4031 ignoreNulls = false;
4032 } else if (isToken(IGNORE)) {
4033 ignoreNulls = true;
4034 } else {
4035 throw createUnexpectedTokenError();
4036 }
4037 }
4038 <IDENTIFIER>
4039 {
4040 if (!isToken(NULLS)) {
4041 throw createUnexpectedTokenError();
4042 }
4043 }
4044 )?
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004045 <OVER> windowExpr = OverClause(signature, argList, aggFilterExpr, token, fromLast, ignoreNulls)
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004046 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004047 return windowExpr;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004048 }
4049}
4050
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004051WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr,
4052 Token startToken, Boolean fromLast, Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004053{
4054 Expression partitionExpr = null;
4055 List<Expression> partitionExprs = new ArrayList<Expression>();
4056 OrderbyClause orderByClause = null;
4057 List<Expression> orderbyList = null;
4058 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004059 List<OrderbyClause.NullOrderModifier> orderbyNullModifierList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004060 WindowExpression.FrameMode frameMode = null;
4061 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
4062 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
4063 Expression frameStartExpr = null, frameEndExpr = null;
4064 WindowExpression.FrameExclusionKind frameExclusionKind = null;
4065 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
4066 VariableExpr windowVar = null;
4067 List<Pair<Expression, Identifier>> windowFieldList = null;
4068}
4069{
4070 (
4071 windowVarWithFieldList = VariableWithFieldMap() <AS>
4072 {
4073 windowVar = windowVarWithFieldList.first;
4074 windowFieldList = windowVarWithFieldList.second;
4075 }
4076 )?
4077 <LEFTPAREN>
4078 (
4079 <IDENTIFIER> { expectToken(PARTITION); } <BY>
4080 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
4081 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
4082 )?
4083 (
4084 orderByClause = OrderbyClause()
4085 {
4086 orderbyList = orderByClause.getOrderbyList();
4087 orderbyModifierList = orderByClause.getModifierList();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004088 orderbyNullModifierList = orderByClause.getNullModifierList();
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004089 }
4090 (
4091 frameMode = WindowFrameMode()
4092 (
4093 frameStart = WindowFrameBoundary() |
4094 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
4095 )
4096 ( frameExclusionKind = WindowFrameExclusion() )?
4097 {
4098 frameStartKind = frameStart.first;
4099 frameStartExpr = frameStart.second;
4100 if (frameEnd == null) {
4101 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4102 } else {
4103 frameEndKind = frameEnd.first;
4104 frameEndExpr = frameEnd.second;
4105 }
4106 if (frameExclusionKind == null) {
4107 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
4108 }
4109 }
4110 )?
4111 )?
4112 <RIGHTPAREN>
4113 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004114 WindowExpression winExpr = new WindowExpression(signature, argList, aggFilterExpr, partitionExprs, orderbyList,
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004115 orderbyModifierList, orderbyNullModifierList, frameMode, frameStartKind, frameStartExpr, frameEndKind,
4116 frameEndExpr, frameExclusionKind, windowVar, windowFieldList, ignoreNulls, fromLast);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004117 return addSourceLocation(winExpr, startToken);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004118 }
4119}
4120
4121WindowExpression.FrameMode WindowFrameMode() throws ParseException:
4122{
4123}
4124{
4125 <IDENTIFIER>
4126 {
4127 if (isToken(RANGE)) {
4128 return WindowExpression.FrameMode.RANGE;
4129 } else if (isToken(ROWS)) {
4130 return WindowExpression.FrameMode.ROWS;
4131 } else if (isToken(GROUPS)) {
4132 return WindowExpression.FrameMode.GROUPS;
4133 } else {
4134 throw createUnexpectedTokenError();
4135 }
4136 }
4137}
4138
4139Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
4140{
4141 boolean current = false;
4142 Expression expr = null;
4143}
4144{
4145 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004146 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004147 | expr = Expression()
4148 )
4149 <IDENTIFIER>
4150 {
4151 WindowExpression.FrameBoundaryKind kind;
4152 if (current && isToken(ROW)) {
4153 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4154 } else if (!current && isToken(PRECEDING)) {
4155 kind = expr == null
4156 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
4157 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
4158 } else if (!current && isToken(FOLLOWING)) {
4159 kind = expr == null
4160 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
4161 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
4162 } else {
4163 throw createUnexpectedTokenError();
4164 }
4165 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
4166 }
4167}
4168
4169WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
4170{
4171 boolean current = false, no = false;
4172}
4173{
4174 <IDENTIFIER>
4175 {
4176 expectToken(EXCLUDE);
4177 }
4178 (
4179 <GROUP>
4180 {
4181 return WindowExpression.FrameExclusionKind.GROUP;
4182 }
4183 |
4184 (
4185 <IDENTIFIER>
4186 {
4187 if (isToken(TIES)) {
4188 return WindowExpression.FrameExclusionKind.TIES;
4189 } else if (isToken(CURRENT)) {
4190 current = true;
4191 } else if (isToken(NO)) {
4192 no = true;
4193 } else {
4194 throw createUnexpectedTokenError();
4195 }
4196 }
4197 <IDENTIFIER>
4198 {
4199 if (current && isToken(ROW)) {
4200 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
4201 } else if (no && isToken(OTHERS)) {
4202 return WindowExpression.FrameExclusionKind.NO_OTHERS;
4203 } else {
4204 throw createUnexpectedTokenError();
4205 }
4206 }
4207 )
4208 )
4209}
4210
Yingyi Bu391f09e2015-10-29 13:49:39 -07004211Expression ParenthesizedExpression() throws ParseException:
4212{
4213 Expression expr;
4214}
4215{
4216 (
4217 LOOKAHEAD(2)
4218 <LEFTPAREN> expr = Expression() <RIGHTPAREN>
4219 |
4220 expr = Subquery()
4221 )
4222 {
4223 return expr;
4224 }
4225}
4226
Yingyi Buc8c067c2016-07-25 23:37:19 -07004227Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004228{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004229 Token startToken = null;
4230 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004231 List<Expression> whenExprs = new ArrayList<Expression>();
4232 List<Expression> thenExprs = new ArrayList<Expression>();
4233 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004234 Expression whenExpr = null;
4235 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004236}
4237{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004238 <CASE> { startToken = token; }
4239 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07004240 (
4241 <WHEN> whenExpr = Expression()
4242 {
4243 whenExprs.add(whenExpr);
4244 }
4245 <THEN> thenExpr = Expression()
4246 {
4247 thenExprs.add(thenExpr);
4248 }
4249 )*
4250 (<ELSE> elseExpr = Expression() )?
4251 <END>
4252 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004253 if (conditionExpr == null) {
4254 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004255 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004256 }
4257 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004258 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07004259 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004260}
4261
Yingyi Buab817482016-08-19 21:29:31 -07004262SelectExpression SelectExpression(boolean subquery) throws ParseException:
4263{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004264 List<LetClause> letClauses = new ArrayList<LetClause>();
4265 SelectSetOperation selectSetOperation;
4266 OrderbyClause orderbyClause = null;
4267 LimitClause limitClause = null;
4268 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07004269}
4270{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004271 ( letClauses = LetClause() )?
4272 selectSetOperation = SelectSetOperation()
4273 (orderbyClause = OrderbyClause() {})?
4274 (limitClause = LimitClause() {})?
4275 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004276 SelectExpression selectExpr =
4277 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
4278 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
4279 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004280 }
4281}
4282
Yingyi Buab817482016-08-19 21:29:31 -07004283SelectSetOperation SelectSetOperation() throws ParseException:
4284{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004285 SetOperationInput setOperationInputLeft;
4286 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
4287}
4288{
4289 {
4290 SelectBlock selectBlockLeft = null;
4291 SelectExpression subqueryLeft = null;
4292 Expression expr = null;
4293 }
4294 selectBlockLeft = SelectBlock()
4295 {
4296 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
4297 }
4298 (
4299 {
4300 SetOpType opType = SetOpType.UNION;
4301 boolean setSemantics = true;
4302 SelectBlock selectBlockRight = null;
4303 SelectExpression subqueryRight = null;
4304 }
4305 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
4306 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
4307 {
4308 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
4309 }
4310 )*
4311 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004312 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
4313 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
4314 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004315 }
4316}
4317
Yingyi Buab817482016-08-19 21:29:31 -07004318SelectExpression Subquery() throws ParseException:
4319{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004320 SelectExpression selectExpr = null;
4321}
4322{
4323 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
4324 {
4325 return selectExpr;
4326 }
4327}
4328
Yingyi Buab817482016-08-19 21:29:31 -07004329SelectBlock SelectBlock() throws ParseException:
4330{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004331 SelectClause selectClause = null;
4332 FromClause fromClause = null;
4333 List<LetClause> fromLetClauses = null;
4334 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004335 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004336 GroupbyClause groupbyClause = null;
4337 List<LetClause> gbyLetClauses = null;
4338 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004339 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004340 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004341}
4342{
4343 (
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004344 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004345 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004346 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004347 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004348 fromClause = FromClause()
4349 (
4350 fromLetClauses = LetClause()
4351 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004352 (whereClause = WhereClause())?
4353 (
4354 groupbyClause = GroupbyClause()
4355 (
4356 gbyLetClauses = LetClause()
4357 )?
4358 (havingClause = HavingClause())?
4359 )?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004360 )
4361 |
4362 (
4363 fromLetClauses = LetClause()
4364 {
4365 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
4366 SourceLocation sourceLoc = getSourceLocation(token);
4367 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
4368 missingExpr.setSourceLocation(sourceLoc);
4369 List<Expression> list = new ArrayList<Expression>(1);
4370 list.add(missingExpr);
4371 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
4372 listExpr.setSourceLocation(sourceLoc);
4373 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
4374 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
4375 fromVar.setSourceLocation(sourceLoc);
4376 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
4377 fromClause = new FromClause(fromTerms);
4378 }
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004379 (whereClause = WhereClause())?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004380 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004381 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004382 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004383 |
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004384 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004385 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004386 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004387 fromLetClauses = LetClause()
4388 )?
4389 (whereClause = WhereClause())?
4390 (
4391 groupbyClause = GroupbyClause()
4392 (
4393 gbyLetClauses = LetClause()
4394 )?
4395 (havingClause = HavingClause())?
4396 )?
4397 selectClause = SelectClause()
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004398 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004399 )
4400 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004401 if (fromLetClauses != null) {
4402 fromLetWhereClauses.addAll(fromLetClauses);
4403 }
4404 if (whereClause != null) {
4405 fromLetWhereClauses.add(whereClause);
4406 }
4407 if (gbyLetClauses != null) {
4408 gbyLetHavingClauses.addAll(gbyLetClauses);
4409 }
4410 if (havingClause != null) {
4411 gbyLetHavingClauses.add(havingClause);
4412 }
4413 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
4414 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004415 selectBlock.setSourceLocation(startSrcLoc);
4416 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004417 }
4418}
4419
Yingyi Buab817482016-08-19 21:29:31 -07004420SelectClause SelectClause() throws ParseException:
4421{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004422 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004423 SelectRegular selectRegular = null;
4424 SelectElement selectElement = null;
4425 boolean distinct = false;
4426}
4427{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004428 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004429 (
4430 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04004431 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07004432 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07004433 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004434 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004435 SourceLocation sourceLoc = getSourceLocation(startToken);
4436 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07004437 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004438 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004439 List<Projection> projections = new ArrayList<Projection>();
4440 projections.add(projection);
4441 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004442 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004443 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004444 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
4445 selectClause.setSourceLocation(sourceLoc);
4446 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004447 }
4448}
4449
Yingyi Buab817482016-08-19 21:29:31 -07004450SelectRegular SelectRegular() throws ParseException:
4451{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004452 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004453 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004454 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004455}
4456{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004457 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004458 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004459 projections.add(projection);
4460 startSrcLoc = projection.getSourceLocation();
4461 }
4462 ( LOOKAHEAD(2) <COMMA> projection = Projection()
4463 {
4464 projections.add(projection);
4465 }
4466 )*
4467 {
4468 SelectRegular selectRegular = new SelectRegular(projections);
4469 selectRegular.setSourceLocation(startSrcLoc);
4470 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004471 }
4472}
4473
Yingyi Buab817482016-08-19 21:29:31 -07004474SelectElement SelectElement() throws ParseException:
4475{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004476 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004477 Expression expr = null;
4478 String name = null;
4479}
4480{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004481 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004482 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004483 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004484 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004485 }
4486}
4487
Yingyi Buab817482016-08-19 21:29:31 -07004488Projection Projection() throws ParseException :
4489{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004490 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004491 Expression expr = null;
4492 Identifier identifier = null;
4493 String name = null;
4494 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004495 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004496}
4497{
4498 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004499 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
4500 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004501 | expr = Expression() ((<AS>)? name = Identifier())?
4502 {
4503 if (name == null) {
4504 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
4505 if (generatedColumnIdentifier != null) {
4506 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
4507 }
4508 }
4509 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004510 )
4511 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004512 Projection projection = new Projection(expr, name, star, varStar);
4513 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
4514 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004515 }
4516}
4517
4518FromClause FromClause() throws ParseException :
4519{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004520 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004521 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
4522 extendCurrentScope();
4523}
4524{
4525 {
4526 FromTerm fromTerm = null;
4527 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004528 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004529 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
4530 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004531 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004532 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004533 }
4534}
4535
4536FromTerm FromTerm() throws ParseException :
4537{
4538 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004539 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004540 VariableExpr posVar = null;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004541 AbstractBinaryCorrelateClause correlateClause = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004542 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
4543}
4544{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004545 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004546 (
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004547 (
4548 correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER)
4549 | ( <INNER> correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER) )
4550 | ( <LEFT> ( <OUTER> )? correlateClause = JoinOrUnnestClause(JoinType.LEFTOUTER, UnnestType.LEFTOUTER) )
4551 | ( <RIGHT> ( <OUTER> )? correlateClause = JoinClause(JoinType.RIGHTOUTER) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004552 )
4553 {
4554 correlateClauses.add(correlateClause);
4555 }
4556 )*
4557 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004558 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07004559 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004560 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004561 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
4562 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
4563 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004564 }
4565}
4566
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004567AbstractBinaryCorrelateClause JoinOrUnnestClause(JoinType joinType, UnnestType unnestType) throws ParseException :
4568{
4569 AbstractBinaryCorrelateClause correlateClause = null;
4570}
4571{
4572 ( correlateClause = JoinClause(joinType) | correlateClause = UnnestClause(unnestType) )
4573 {
4574 return correlateClause;
4575 }
4576}
4577
Yingyi Bu391f09e2015-10-29 13:49:39 -07004578JoinClause JoinClause(JoinType joinType) throws ParseException :
4579{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004580 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004581 Expression rightExpr = null;
4582 VariableExpr rightVar = null;
4583 VariableExpr posVar = null;
4584 Expression conditionExpr = null;
4585}
4586{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004587 <JOIN> { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())? <ON> conditionExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004588 {
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004589 if (rightVar == null) {
4590 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004591 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004592 JoinClause joinClause = new JoinClause(joinType, rightExpr, rightVar, posVar, conditionExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004593 return addSourceLocation(joinClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004594 }
4595}
4596
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004597UnnestClause UnnestClause(UnnestType unnestType) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004598{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004599 Token startToken = null;
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08004600 Expression rightExpr = null;
4601 VariableExpr rightVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004602 VariableExpr posVar = null;
4603}
4604{
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08004605 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004606 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004607 if (rightVar == null) {
4608 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004609 }
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004610 UnnestClause unnestClause = new UnnestClause(unnestType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004611 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004612 }
4613}
4614
Yingyi Bu391f09e2015-10-29 13:49:39 -07004615List<LetClause> LetClause() throws ParseException:
4616{
4617 List<LetClause> letList = new ArrayList<LetClause>();
4618 LetClause letClause;
4619}
4620{
4621 (
4622 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
4623 |
4624 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
4625 )
4626 {
4627 return letList;
4628 }
4629}
4630
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004631WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004632{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004633 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004634 Expression whereExpr;
4635}
4636{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004637 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004638 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004639 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004640 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004641 }
4642}
4643
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004644OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004645{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004646 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004647 OrderbyClause oc = new OrderbyClause();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004648 Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> orderbyExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004649 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004650 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004651 List<OrderbyClause.NullOrderModifier> nullModifierList = new ArrayList<OrderbyClause.NullOrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004652}
4653{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004654 <ORDER>
4655 {
4656 startToken = token;
4657 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
4658 if (hintToken != null) {
4659 switch (hintToken.hint) {
4660 case INMEMORY_HINT:
4661 String[] splits = hintToken.hintParams.split("\\s+");
4662 int numFrames = Integer.parseInt(splits[0]);
4663 int numTuples = Integer.parseInt(splits[1]);
4664 oc.setNumFrames(numFrames);
4665 oc.setNumTuples(numTuples);
4666 break;
4667 case RANGE_HINT:
4668 try {
4669 Expression rangeExpr = parseExpression(hintToken.hintParams);
4670 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
4671 oc.setRangeMap(rangeMap);
4672 } catch (CompilationException e) {
4673 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
Ali Alsuliman80225e22018-10-15 14:17:07 -07004674 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004675 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004676 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004677 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004678 }
4679 <BY> orderbyExpr = OrderByExpression()
4680 {
4681 orderbyList.add(orderbyExpr.first);
4682 modifierList.add(orderbyExpr.second);
4683 nullModifierList.add(orderbyExpr.third);
4684 }
4685 (
4686 LOOKAHEAD(2) <COMMA> orderbyExpr = OrderByExpression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004687 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004688 orderbyList.add(orderbyExpr.first);
4689 modifierList.add(orderbyExpr.second);
4690 nullModifierList.add(orderbyExpr.third);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004691 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004692 )*
4693 {
4694 oc.setOrderbyList(orderbyList);
4695 oc.setModifierList(modifierList);
4696 oc.setNullModifierList(nullModifierList);
4697 return addSourceLocation(oc, startToken);
4698 }
4699}
Yingyi Bu391f09e2015-10-29 13:49:39 -07004700
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004701Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> OrderByExpression()
4702 throws ParseException:
4703{
4704 Expression orderbyExpr = null;
4705 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
4706 OrderbyClause.NullOrderModifier nullModif = null;
4707}
4708{
4709 orderbyExpr = Expression()
4710 (
4711 <ASC> { modif = OrderbyClause.OrderModifier.ASC; }
4712 |
4713 <DESC> { modif = OrderbyClause.OrderModifier.DESC; }
4714 )?
4715 (
4716 LOOKAHEAD({ laIdentifier(NULLS) }) <IDENTIFIER> { expectToken(NULLS); } <IDENTIFIER>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004717 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004718 if (isToken(FIRST)) {
4719 nullModif = OrderbyClause.NullOrderModifier.FIRST;
4720 } else if (isToken(LAST)) {
4721 nullModif = OrderbyClause.NullOrderModifier.LAST;
4722 } else {
4723 throw createUnexpectedTokenError();
4724 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004725 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004726 )?
4727 {
4728 return new Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier>(orderbyExpr, modif,
4729 nullModif);
4730 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004731}
4732
4733GroupbyClause GroupbyClause()throws ParseException :
4734{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004735 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004736 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004737 List<List<GbyVariableExpressionPair>> gbyList = null;
4738 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004739 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07004740 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004741 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004742}
4743{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004744 {
4745 Scope newScope = extendCurrentScopeNoPush(true);
4746 // extendCurrentScope(true);
4747 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004748 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004749 {
4750 startToken = token;
4751 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
4752 if (hintToken != null) {
4753 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004754 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004755 }
4756 <BY> groupingElementList = GroupingElementList()
4757 (
4758 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004759 {
4760 groupVar = groupVarWithFieldList.first;
4761 groupFieldList = groupVarWithFieldList.second;
4762 }
Yingyi Buacc12a92016-03-26 17:25:05 -07004763 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004764 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004765 if (groupingSetsParser == null) {
4766 groupingSetsParser = new SqlppGroupingSetsParser();
4767 }
4768 SourceLocation sourceLoc = getSourceLocation(startToken);
4769 try {
4770 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
4771 } catch (CompilationException e) {
4772 throw new SqlppParseException(sourceLoc, e.getMessage());
4773 }
4774 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004775 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07004776 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07004777 gbc.setGroupVar(groupVar);
4778 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004779 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004780 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004781 }
4782}
4783
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004784List<GroupingElement> GroupingElementList() throws ParseException:
4785{
4786 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
4787 GroupingElement groupingElement = null;
4788}
4789{
4790 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
4791 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
4792 {
4793 return groupingElementList;
4794 }
4795}
4796
4797GroupingElement GroupingElement() throws ParseException:
4798{
4799 GroupingElement groupingElement = null;
4800 List<GroupingSet> groupingSets = null;
4801 List<GroupingElement> groupingElements = null;
4802}
4803{
4804 (
4805 LOOKAHEAD(2)
4806 <LEFTPAREN> <RIGHTPAREN>
4807 {
4808 groupingElement = GroupingSet.EMPTY;
4809 }
4810 |
4811 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
4812 <IDENTIFIER> { expectToken(ROLLUP); }
4813 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4814 {
4815 groupingElement = new RollupCube(groupingSets, false);
4816 }
4817 |
4818 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
4819 <IDENTIFIER> { expectToken(CUBE); }
4820 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4821 {
4822 groupingElement = new RollupCube(groupingSets, true);
4823 }
4824 |
4825 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
4826 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
4827 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
4828 {
4829 groupingElement = new GroupingSets(groupingElements);
4830 }
4831 |
4832 groupingElement = OrdinaryGroupingSet()
4833 )
4834 {
4835 return groupingElement;
4836 }
4837}
4838
4839GroupingSet OrdinaryGroupingSet() throws ParseException:
4840{
4841 GbyVariableExpressionPair gbyExprPair = null;
4842 List<GbyVariableExpressionPair> items = null;
4843}
4844{
4845 (
4846 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
4847 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
4848 )
4849 {
4850 return new GroupingSet(items);
4851 }
4852}
4853
4854List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
4855{
4856 GroupingSet groupingSet = null;
4857 List<GroupingSet> items = new ArrayList<GroupingSet>();
4858}
4859{
4860 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
4861 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
4862 {
4863 return items;
4864 }
4865}
4866
4867List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
4868{
4869 GbyVariableExpressionPair gbyExprPair = null;
4870 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
4871}
4872{
4873 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
4874 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
4875 {
4876 return items;
4877 }
4878}
4879
4880GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
4881{
4882 Expression expr = null;
4883 VariableExpr var = null;
4884}
4885{
4886 expr = Expression() ( (<AS>)? var = Variable() )?
4887 {
4888 return new GbyVariableExpressionPair(var, expr);
4889 }
4890}
4891
Yingyi Bu391f09e2015-10-29 13:49:39 -07004892HavingClause HavingClause() throws ParseException:
4893{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004894 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004895 Expression filterExpr = null;
4896}
4897{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004898 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004899 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004900 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004901 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004902 }
4903}
4904
4905LimitClause LimitClause() throws ParseException:
4906{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004907 Token startToken = null;
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08004908 Expression limitExpr = null, offsetExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004909}
4910{
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08004911 (
4912 (
4913 <LIMIT> { startToken = token; pushForbiddenScope(getCurrentScope()); } limitExpr = Expression()
4914 ( <OFFSET> offsetExpr = Expression() )?
4915 { popForbiddenScope(); }
4916 )
4917 |
4918 (
4919 <OFFSET> { startToken = token; pushForbiddenScope(getCurrentScope()); } offsetExpr = Expression()
4920 { popForbiddenScope(); }
4921 )
4922 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004923 {
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08004924 LimitClause lc = new LimitClause(limitExpr, offsetExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004925 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004926 }
4927}
4928
4929QuantifiedExpression QuantifiedExpression()throws ParseException:
4930{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004931 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004932 QuantifiedExpression qc = new QuantifiedExpression();
4933 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
4934 Expression satisfiesExpr;
4935 VariableExpr var;
4936 Expression inExpr;
4937 QuantifiedPair pair;
4938}
4939{
4940 {
4941 createNewScope();
4942 }
4943
ggalvizo21e52822021-07-27 10:26:31 -10004944 ( LOOKAHEAD(2)
4945 (<ANY>|<SOME>)<AND><EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME_AND_EVERY); }
4946 | (<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); }
4947 | <EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004948 var = Variable() <IN> inExpr = Expression()
4949 {
4950 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004951 quantifiedList.add(pair);
4952 }
4953 (
4954 <COMMA> var = Variable() <IN> inExpr = Expression()
4955 {
4956 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004957 quantifiedList.add(pair);
4958 }
4959 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07004960 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004961 {
4962 qc.setSatisfiesExpr(satisfiesExpr);
4963 qc.setQuantifiedList(quantifiedList);
4964 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004965 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004966 }
4967}
4968
4969LetClause LetElement() throws ParseException:
4970{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004971 VariableExpr varExp;
4972 Expression beExp;
4973 extendCurrentScope();
4974}
4975{
4976 varExp = Variable() <EQ> beExp = Expression()
4977 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004978 LetClause lc = new LetClause(varExp, beExp);
4979 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004980 return lc;
4981 }
4982}
4983
4984LetClause WithElement() throws ParseException:
4985{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004986 VariableExpr varExp;
4987 Expression beExp;
4988 extendCurrentScope();
4989}
4990{
4991 varExp = Variable() <AS> beExp = Expression()
4992 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004993 LetClause lc = new LetClause(varExp, beExp);
4994 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07004995 return lc;
4996 }
4997}
4998
4999TOKEN_MGR_DECLS:
5000{
5001 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07005002 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005003 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005004
5005 public void pushState() {
5006 lexerStateStack.push( curLexState );
5007 }
5008
5009 public void popState(String token) {
5010 if (lexerStateStack.size() > 0) {
5011 SwitchTo( lexerStateStack.pop() );
5012 } else {
5013 int errorLine = input_stream.getEndLine();
5014 int errorColumn = input_stream.getEndColumn();
5015 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
5016 + "\" but state stack is empty.";
5017 throw new TokenMgrError(msg, -1);
5018 }
5019 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005020
5021 void CommonTokenAction(Token token) {
5022 Token hintToken = token.specialToken;
5023 if (hintToken != null) {
5024 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
5025 String text = hintToken.image.substring(1).trim();
5026 boolean hintFound = hintToken.parseHint(text);
5027 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
5028 }
5029 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005030}
5031
5032<DEFAULT,IN_DBL_BRACE>
5033TOKEN [IGNORE_CASE]:
5034{
Ian Maxon38fe9402020-01-29 19:27:40 -08005035 <ADAPTER: "adapter">
5036 | <ALL : "all">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005037 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07005038 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005039 | <APPLY : "apply">
5040 | <AS : "as">
5041 | <ASC : "asc">
5042 | <AT : "at">
5043 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005044 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005045 | <BTREE : "btree">
5046 | <BY : "by">
5047 | <CASE : "case">
5048 | <CLOSED : "closed">
5049 | <CREATE : "create">
Dmitry Lychagin0ebc4202021-01-25 13:12:41 -08005050 | <COMPACTION : "compaction"> // no longer used
Yingyi Bu391f09e2015-10-29 13:49:39 -07005051 | <COMPACT : "compact">
5052 | <CONNECT : "connect">
5053 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07005054 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07005055 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07005056 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005057 | <DECLARE : "declare">
5058 | <DEFINITION : "definition">
5059 | <DELETE : "delete">
5060 | <DESC : "desc">
5061 | <DISCONNECT : "disconnect">
5062 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005063 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005064 | <DROP : "drop">
5065 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07005066 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005067 | <ELSE : "else">
5068 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07005069 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005070 | <EVERY : "every">
5071 | <EXCEPT : "except">
5072 | <EXISTS : "exists">
5073 | <EXTERNAL : "external">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005074 | <FALSE : "false">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005075 | <FEED : "feed">
5076 | <FILTER : "filter">
5077 | <FLATTEN : "flatten">
5078 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005079 | <FROM : "from">
5080 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08005081 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005082 | <FUNCTION : "function">
5083 | <GROUP : "group">
5084 | <HAVING : "having">
5085 | <HINTS : "hints">
5086 | <IF : "if">
5087 | <INTO : "into">
5088 | <IN : "in">
5089 | <INDEX : "index">
5090 | <INGESTION : "ingestion">
5091 | <INNER : "inner">
5092 | <INSERT : "insert">
5093 | <INTERNAL : "internal">
5094 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07005095 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005096 | <JOIN : "join">
5097 | <KEYWORD : "keyword">
5098 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07005099 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005100 | <LEFT : "left">
5101 | <LETTING : "letting">
5102 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005103 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005104 | <LIMIT : "limit">
5105 | <LOAD : "load">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005106 | <MISSING : "missing">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005107 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005108 | <NODEGROUP : "nodegroup">
5109 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07005110 | <NOT : "not">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005111 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005112 | <OFFSET : "offset">
5113 | <ON : "on">
5114 | <OPEN : "open">
5115 | <OR : "or">
5116 | <ORDER : "order">
5117 | <OUTER : "outer">
5118 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07005119 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005120 | <PATH : "path">
5121 | <POLICY : "policy">
5122 | <PRESORTED : "pre-sorted">
5123 | <PRIMARY : "primary">
5124 | <RAW : "raw">
5125 | <REFRESH : "refresh">
5126 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005127 | <RETURNING : "returning">
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005128 | <RIGHT : "right">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005129 | <RTREE : "rtree">
5130 | <RUN : "run">
5131 | <SATISFIES : "satisfies">
5132 | <SECONDARY : "secondary">
5133 | <SELECT : "select">
5134 | <SET : "set">
5135 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08005136 | <START : "start">
5137 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08005138 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03005139 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07005140 | <THEN : "then">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005141 | <TO : "to">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005142 | <TRUE : "true">
5143 | <TYPE : "type">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005144 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08005145 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005146 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07005147 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005148 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07005149 | <USE : "use">
5150 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005151 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08005152 | <VALUED : "valued">
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07005153 | <VIEW : "view">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005154 | <WHEN : "when">
5155 | <WHERE : "where">
5156 | <WITH : "with">
5157 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005158}
5159
5160<DEFAULT,IN_DBL_BRACE>
5161TOKEN :
5162{
5163 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07005164 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005165 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005166 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005167 | <MUL : "*">
5168 | <PLUS : "+">
5169
5170 | <LEFTPAREN : "(">
5171 | <RIGHTPAREN : ")">
5172 | <LEFTBRACKET : "[">
5173 | <RIGHTBRACKET : "]">
5174
5175 | <ATT : "@">
5176 | <COLON : ":">
5177 | <COMMA : ",">
5178 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005179 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005180 | <QUES : "?">
5181 | <SEMICOLON : ";">
5182 | <SHARP : "#">
5183
5184 | <LT : "<">
5185 | <GT : ">">
5186 | <LE : "<=">
5187 | <GE : ">=">
5188 | <EQ : "=">
5189 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07005190 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005191 | <SIMILAR : "~=">
5192}
5193
5194<DEFAULT,IN_DBL_BRACE>
5195TOKEN :
5196{
5197 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
5198}
5199
5200<DEFAULT>
5201TOKEN :
5202{
5203 <RIGHTBRACE : "}"> { popState("}"); }
5204}
5205
5206<DEFAULT,IN_DBL_BRACE>
5207TOKEN :
5208{
5209 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
5210}
5211
5212<IN_DBL_BRACE>
5213TOKEN :
5214{
5215 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
5216}
5217
5218<DEFAULT,IN_DBL_BRACE>
5219TOKEN :
5220{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005221 <#DIGIT : ["0" - "9"]>
5222}
5223
5224<DEFAULT,IN_DBL_BRACE>
5225TOKEN:
5226{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005227 <INTEGER_LITERAL : <DIGITS> >
5228 | <DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07005229 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
5230 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005231 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005232 | <FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
Yingyi Bue4d919e2016-10-30 10:47:03 -07005233 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
5234 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005235 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005236 | <#DIGITS : (<DIGIT>)+ >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005237}
5238
5239<DEFAULT,IN_DBL_BRACE>
5240TOKEN :
5241{
5242 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005243 | <#IDENTIFIER_START_SPECIALCHAR : ["_"]>
5244 | <#IDENTIFIER_REST_SPECIALCHAR : ["$"]>
5245 | <#IDENTIFIER_START : <LETTER> | <IDENTIFIER_START_SPECIALCHAR> >
5246 | <#IDENTIFIER_REST : <LETTER> | <DIGIT> | <IDENTIFIER_START_SPECIALCHAR> | <IDENTIFIER_REST_SPECIALCHAR> >
5247 | <IDENTIFIER : <IDENTIFIER_START> (<IDENTIFIER_REST>)* >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005248}
5249
5250<DEFAULT,IN_DBL_BRACE>
5251TOKEN :
5252{
5253 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07005254 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005255 <EscapeQuot>
5256 | <EscapeBslash>
5257 | <EscapeSlash>
5258 | <EscapeBspace>
5259 | <EscapeFormf>
5260 | <EscapeNl>
5261 | <EscapeCr>
5262 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005263 | ~["`","\\"])* "`">
5264 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005265 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005266 | <EscapeBslash>
5267 | <EscapeSlash>
5268 | <EscapeBspace>
5269 | <EscapeFormf>
5270 | <EscapeNl>
5271 | <EscapeCr>
5272 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005273 | ~["\"","\\"])* "\"")
5274 | ("\'"(
5275 <EscapeApos>
5276 | <EscapeBslash>
5277 | <EscapeSlash>
5278 | <EscapeBspace>
5279 | <EscapeFormf>
5280 | <EscapeNl>
5281 | <EscapeCr>
5282 | <EscapeTab>
5283 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005284 | < #EscapeQuot: "\\\"" >
5285 | < #EscapeApos: "\\\'" >
5286 | < #EscapeBslash: "\\\\" >
5287 | < #EscapeSlash: "\\/" >
5288 | < #EscapeBspace: "\\b" >
5289 | < #EscapeFormf: "\\f" >
5290 | < #EscapeNl: "\\n" >
5291 | < #EscapeCr: "\\r" >
5292 | < #EscapeTab: "\\t" >
5293}
5294
5295<DEFAULT,IN_DBL_BRACE>
5296TOKEN :
5297{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005298 <DOLLAR_INTEGER_LITERAL : "$" <INTEGER_LITERAL> >
5299 | <DOLLAR_IDENTIFIER : "$" <IDENTIFIER> >
5300 | <DOLLAR_QUOTED_STRING: "$" <QUOTED_STRING> >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005301}
5302
5303<DEFAULT,IN_DBL_BRACE>
5304SKIP:
5305{
5306 " "
5307 | "\t"
5308 | "\r"
5309 | "\n"
5310}
5311
5312<DEFAULT,IN_DBL_BRACE>
5313SKIP:
5314{
5315 <"//" (~["\n"])* "\n">
5316}
5317
5318<DEFAULT,IN_DBL_BRACE>
5319SKIP:
5320{
5321 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5322}
5323
5324<DEFAULT,IN_DBL_BRACE>
5325SKIP:
5326{
Yingyi Bu93846a72016-09-13 16:30:39 -07005327 <"--" (~["\n"])* "\n">
5328}
5329
5330
5331<DEFAULT,IN_DBL_BRACE>
5332SKIP:
5333{
5334 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5335}
5336
5337<DEFAULT,IN_DBL_BRACE>
5338SKIP:
5339{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005340 <"/*"> { pushState(); } : INSIDE_COMMENT
5341}
5342
5343<INSIDE_COMMENT>
5344SPECIAL_TOKEN:
5345{
5346 <"+"(" ")*(~["*"])*>
5347}
5348
5349<INSIDE_COMMENT>
5350SKIP:
5351{
5352 <"/*"> { pushState(); }
5353}
5354
5355<INSIDE_COMMENT>
5356SKIP:
5357{
5358 <"*/"> { popState("*/"); }
5359 | <~[]>
5360}