blob: e94ad6b1a8820d4d7c812a8bde227256f20aca27 [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";
Ali Alsuliman931e7382021-08-08 21:55:59 +0300230 private static final String INCLUDE = "INCLUDE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700231 private static final String FIRST = "FIRST";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800232 private static final String FOLLOWING = "FOLLOWING";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700233 private static final String GROUPING = "GROUPING";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800234 private static final String GROUPS = "GROUPS";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700235 private static final String IGNORE = "IGNORE";
Dmitry Lychagin5476f962019-05-31 16:03:11 -0700236 private static final String LAST = "LAST";
Glenn67fd1f32021-02-25 16:04:49 -0800237 private static final String META = "META";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800238 private static final String NO = "NO";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700239 private static final String NULLS = "NULLS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800240 private static final String OTHERS = "OTHERS";
241 private static final String PARTITION = "PARTITION";
242 private static final String PRECEDING = "PRECEDING";
243 private static final String RANGE = "RANGE";
Dmitry Lychagin3346cea2019-05-28 12:11:54 -0700244 private static final String RESPECT = "RESPECT";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700245 private static final String ROLLUP = "ROLLUP";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800246 private static final String ROW = "ROW";
247 private static final String ROWS = "ROWS";
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700248 private static final String SETS = "SETS";
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800249 private static final String TIES = "TIES";
250 private static final String UNBOUNDED = "UNBOUNDED";
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700251 private static final String REPLACE = "REPLACE";
Ian Maxon38fe9402020-01-29 19:27:40 -0800252 private static final String RETURNS = "RETURNS";
Rui Guoe6986dd2020-11-25 19:50:06 -0800253 private static final String CONFIG = "CONFIG";
254
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800255
Dmitry Lychagin75f19872019-10-03 17:56:35 -0700256 private static final String INT_TYPE_NAME = "int";
Dmitry Lychagin541652082020-11-09 14:04:53 -0800257 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 -0700258
Till Westmann7199a562016-09-17 16:07:32 -0700259 // error configuration
260 protected static final boolean REPORT_EXPECTED_TOKENS = false;
261
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -0700262 private int externalVarCounter;
263
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800264 private DataverseName defaultDataverse;
265
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700266 private SqlppGroupingSetsParser groupingSetsParser;
267
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700268 private final WarningCollector warningCollector = new WarningCollector();
269
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700270 private final Map<SourceLocation, String> hintCollector = new HashMap<SourceLocation, String>();
271
Yingyi Bu391f09e2015-10-29 13:49:39 -0700272 private static class IndexParams {
273 public IndexType type;
274 public int gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800275 public String fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700276
Rui Guoe6986dd2020-11-25 19:50:06 -0800277 public IndexParams(IndexType type, int gramLength, String fullTextConfig) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700278 this.type = type;
279 this.gramLength = gramLength;
Rui Guoe6986dd2020-11-25 19:50:06 -0800280 this.fullTextConfig = fullTextConfig;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700281 }
282 };
283
284 private static class FunctionName {
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800285 public DataverseName dataverse;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700286 public String library;
287 public String function;
Caleb Herbel8d9c57a2020-08-14 20:41:13 -0600288 public Token hintToken;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700289 public SourceLocation sourceLoc;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700290 }
291
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700292 private IRecordFieldDataGen parseFieldDataGen(Token hintToken) throws ParseException {
293 String[] splits = hintToken.hintParams != null ? hintToken.hintParams.split("\\s+") : null;
294 switch (hintToken.hint) {
295 case VAL_FILE_HINT:
296 File[] valFiles = new File[splits.length];
297 for (int k=0; k<splits.length; k++) {
298 valFiles[k] = new File(splits[k]);
299 }
300 return new FieldValFileDataGen(valFiles);
301 case VAL_FILE_SAME_INDEX_HINT:
302 return new FieldValFileSameIndexDataGen(new File(splits[0]), splits[1]);
303 case LIST_VAL_FILE_HINT:
304 return new ListValFileDataGen(new File(splits[0]), Integer.parseInt(splits[1]), Integer.parseInt(splits[2]));
305 case LIST_HINT:
306 return new ListDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
307 case INTERVAL_HINT:
308 FieldIntervalDataGen.ValueType vt;
309 switch (splits[0]) {
310 case "int":
311 vt = FieldIntervalDataGen.ValueType.INT;
312 break;
313 case "long":
314 vt = FieldIntervalDataGen.ValueType.LONG;
315 break;
316 case "float":
317 vt = FieldIntervalDataGen.ValueType.FLOAT;
318 break;
319 case "double":
320 vt = FieldIntervalDataGen.ValueType.DOUBLE;
321 break;
322 default:
323 throw new SqlppParseException(getSourceLocation(hintToken),
324 "Unknown type for interval data gen: " + splits[0]);
325 }
326 return new FieldIntervalDataGen(vt, splits[1], splits[2]);
327 case INSERT_RAND_INT_HINT:
328 return new InsertRandIntDataGen(splits[0], splits[1]);
329 case DATE_BETWEEN_YEARS_HINT:
330 return new DateBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
331 case DATETIME_BETWEEN_YEARS_HINT:
332 return new DatetimeBetweenYearsDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]));
333 case DATETIME_ADD_RAND_HOURS_HINT:
334 return new DatetimeAddRandHoursDataGen(Integer.parseInt(splits[0]), Integer.parseInt(splits[1]), splits[2]);
335 case AUTO_HINT:
336 return new AutoDataGen(splits[0]);
337 default:
338 return null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700339 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700340 }
341
Till Westmann7199a562016-09-17 16:07:32 -0700342 public SQLPPParser(String s) {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700343 this(new StringReader(s));
344 super.setInput(s);
345 }
346
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800347 public static void main(String args[]) throws ParseException, TokenMgrError, IOException, FileNotFoundException, CompilationException {
Yingyi Bu391f09e2015-10-29 13:49:39 -0700348 File file = new File(args[0]);
349 Reader fis = new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8"));
350 SQLPPParser parser = new SQLPPParser(fis);
351 List<Statement> st = parser.parse();
352 //st.accept(new SQLPPPrintVisitor(), 0);
353 }
354
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800355 @Override
Taewoo Kime65e6ca2017-01-14 17:53:28 -0800356 public List<Statement> parse() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700357 return parseImpl(new ParseFunction<List<Statement>>() {
358 @Override
359 public List<Statement> parse() throws ParseException {
360 return SQLPPParser.this.Statement();
361 }
362 });
363 }
364
Dmitry Lychagin62f0beb2020-09-08 11:44:19 -0700365 @Override
366 public Expression parseExpression() throws CompilationException {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700367 return parseImpl(new ParseFunction<Expression>() {
368 @Override
369 public Expression parse() throws ParseException {
370 return SQLPPParser.this.Expression();
371 }
372 });
373 }
374
375 private static Expression parseExpression(String text) throws CompilationException {
376 return new SQLPPParser(text).parseExpression();
377 }
378
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800379 @Override
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700380 public List<String> parseMultipartIdentifier() throws CompilationException {
381 return parseImpl(new ParseFunction<List<String>>() {
382 @Override
383 public List<String> parse() throws ParseException {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700384 return SQLPPParser.this.MultipartIdentifier().first;
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700385 }
386 });
387 }
388
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800389 private List<String> parseParenthesizedIdentifierList() throws CompilationException {
390 return parseImpl(new ParseFunction<List<String>>() {
391 @Override
392 public List<String> parse() throws ParseException {
393 return SQLPPParser.this.ParenthesizedIdentifierList();
394 }
395 });
396 }
397
398 private static List<String> parseParenthesizedIdentifierList(String text) throws CompilationException {
399 return new SQLPPParser(text).parseParenthesizedIdentifierList();
400 }
401
Dmitry Lychaginef680d7d2020-10-15 13:10:41 -0700402 @Override
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700403 public FunctionDecl parseFunctionBody(FunctionSignature signature, List<String> paramNames, boolean isStored)
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800404 throws CompilationException {
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800405 return parseImpl(new ParseFunction<FunctionDecl>() {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800406 @Override
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800407 public FunctionDecl parse() throws ParseException {
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800408 DataverseName dataverse = defaultDataverse;
409 defaultDataverse = signature.getDataverseName();
410 createNewScope();
Dmitry Lychagine74454b2020-02-06 13:44:21 -0800411 List<VarIdentifier> paramVars = new ArrayList<VarIdentifier>(paramNames.size());
412 for (String paramName : paramNames) {
413 paramVars.add(SqlppVariableUtil.toInternalVariableIdentifier(paramName));
414 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800415 Expression functionBodyExpr = SQLPPParser.this.FunctionBody();
416 removeCurrentScope();
417 defaultDataverse = dataverse;
Dmitry Lychagin9ba74872021-04-05 14:38:20 -0700418 return new FunctionDecl(signature, paramVars, functionBodyExpr, isStored);
Dmitry Lychagin45de2342020-02-03 12:01:28 -0800419 }
420 });
421 }
422
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700423 @Override
424 public ViewDecl parseViewBody(DatasetFullyQualifiedName viewName) throws CompilationException {
425 return parseImpl(new ParseFunction<ViewDecl>() {
426 @Override
427 public ViewDecl parse() throws ParseException {
428 DataverseName dataverse = defaultDataverse;
429 defaultDataverse = viewName.getDataverseName();
430 createNewScope();
431 Expression viewBodyExpr = SQLPPParser.this.ViewBody();
432 removeCurrentScope();
433 defaultDataverse = dataverse;
434 return new ViewDecl(viewName, viewBodyExpr);
435 }
436 });
437 }
438
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700439 private <T> T parseImpl(ParseFunction<T> parseFunction) throws CompilationException {
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700440 warningCollector.clear();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700441 hintCollector.clear();
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800442 token_source.hintCollector = hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700443 try {
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700444 return parseFunction.parse();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700445 } catch (SqlppParseException e) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700446 if (e.getCause() instanceof AlgebricksException) {
447 AlgebricksException cause = (AlgebricksException) e.getCause();
448 if (cause.getError().isPresent() && cause.getError().get() instanceof ErrorCode) {
449 throw new CompilationException((ErrorCode) cause.getError().get(), e.getSourceLocation(),
450 cause.getParams());
451 }
452 }
453 throw new CompilationException(ErrorCode.PARSE_ERROR, e.getSourceLocation(),
454 LogRedactionUtil.userData(getMessage(e)));
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700455 } catch (ParseException e) {
456 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(getMessage(e)));
Yingyi Bu391f09e2015-10-29 13:49:39 -0700457 } catch (Error e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700458 // 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 -0700459 // 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 -0700460 final String msg = e.getClass().getSimpleName() + (e.getMessage() != null ? ": " + e.getMessage() : "");
Ali Alsulimanfe901892019-03-19 01:40:35 -0700461 throw new CompilationException(ErrorCode.PARSE_ERROR, LogRedactionUtil.userData(msg));
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700462 } finally {
463 reportUnclaimedHints();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700464 }
465 }
Till Westmann7199a562016-09-17 16:07:32 -0700466
Dmitry Lychagin1227f022019-08-01 13:14:30 -0700467 @FunctionalInterface
468 private interface ParseFunction<T> {
469 T parse() throws ParseException;
470 }
471
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700472 @Override
Dmitry Lychagin67714e22021-01-29 11:18:58 -0800473 public void getWarnings(IWarningCollector outWarningCollector) {
474 warningCollector.getWarnings(outWarningCollector);
475 }
476
477 @Override
Ali Alsuliman6a9e2b02019-09-16 20:50:41 -0700478 public void getWarnings(Collection<? super Warning> outWarnings, long maxWarnings) {
479 warningCollector.getWarnings(outWarnings, maxWarnings);
480 }
481
482 @Override
483 public long getTotalWarningsCount() {
484 return warningCollector.getTotalWarningsCount();
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700485 }
486
Till Westmann7199a562016-09-17 16:07:32 -0700487 protected String getMessage(ParseException pe) {
488 Token currentToken = pe.currentToken;
489 if (currentToken == null) {
490 return pe.getMessage();
491 }
492 int[][] expectedTokenSequences = pe.expectedTokenSequences;
493 String[] tokenImage = pe.tokenImage;
494 String sep = REPORT_EXPECTED_TOKENS ? eol : " ";
495 StringBuilder expected = REPORT_EXPECTED_TOKENS ? new StringBuilder() : null;
496 int maxSize = appendExpected(expected, expectedTokenSequences, tokenImage);
497 Token tok = currentToken.next;
498 int line = tok.beginLine;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700499 StringBuilder message = new StringBuilder(128);
500 message.append("In line ").append(line).append(" >>").append(getLine(line)).append("<<").append(sep).append("Encountered ");
Till Westmann7199a562016-09-17 16:07:32 -0700501 for (int i = 0; i < maxSize; i++) {
502 if (i != 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700503 message.append(' ');
Till Westmann7199a562016-09-17 16:07:32 -0700504 }
505 if (tok.kind == 0) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700506 message.append(fixQuotes(tokenImage[0]));
Till Westmann7199a562016-09-17 16:07:32 -0700507 break;
508 }
Till Westmanne3c9f272016-10-09 11:09:26 -0700509 final String fixedTokenImage = tokenImage[tok.kind];
510 if (! tok.image.equalsIgnoreCase(stripQuotes(fixedTokenImage))) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700511 message.append(fixQuotes(fixedTokenImage)).append(' ');
Till Westmanne3c9f272016-10-09 11:09:26 -0700512 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700513 message.append(quot).append(addEscapes(tok.image)).append(quot);
Till Westmann7199a562016-09-17 16:07:32 -0700514 tok = tok.next;
515 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700516 message.append(" at column ").append(currentToken.next.beginColumn).append('.').append(sep);
Till Westmann7199a562016-09-17 16:07:32 -0700517 if (REPORT_EXPECTED_TOKENS) {
518 if (expectedTokenSequences.length == 1) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700519 message.append("Was expecting:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700520 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700521 message.append("Was expecting one of:").append(sep).append(" ");
Till Westmann7199a562016-09-17 16:07:32 -0700522 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700523 message.append(expected);
Till Westmann7199a562016-09-17 16:07:32 -0700524 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700525 return message.toString();
Till Westmann7199a562016-09-17 16:07:32 -0700526 }
527
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700528 protected static SourceLocation getSourceLocation(Token token) {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700529 return
530 token == null ? null :
531 token.sourceLocation != null ? token.sourceLocation :
532 new SourceLocation(token.beginLine, token.beginColumn);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700533 }
534
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700535 protected static <T extends AbstractLangExpression> T addSourceLocation(T expr, Token token) {
536 expr.setSourceLocation(getSourceLocation(token));
537 return expr;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700538 }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800539
540 private boolean isToken(String image) {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700541 return isToken(token, image);
542 }
543
544 private static boolean isToken(Token token, String image) {
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800545 return token.image.equalsIgnoreCase(image);
546 }
547
548 private void expectToken(String image) throws SqlppParseException {
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700549 expectToken(token, image);
550 }
551
552 private static void expectToken(Token token, String image) throws SqlppParseException {
553 if (!isToken(token, image)) {
554 throw createUnexpectedTokenError(token);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800555 }
556 }
557
558 private SqlppParseException createUnexpectedTokenError() {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700559 return createUnexpectedTokenError(token, null);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700560 }
561
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700562 private static SqlppParseException createUnexpectedTokenError(Token t) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700563 return createUnexpectedTokenError(t, null);
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800564 }
565
566 private SqlppParseException createUnexpectedTokenError(String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700567 return createUnexpectedTokenError(token, expected);
568 }
569
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700570 private static SqlppParseException createUnexpectedTokenError(Token t, String expected) {
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700571 String message = "Unexpected token: " + LogRedactionUtil.userData(t.image) +
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -0800572 (expected == null ? "" : ". Expected: " + LogRedactionUtil.userData(expected));
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700573 return new SqlppParseException(getSourceLocation(t), message);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800574 }
575
Dmitry Lychaginac98f482020-03-31 12:26:40 -0700576 private boolean laToken(int idx, int kind) {
577 Token t = getToken(idx);
578 return t.kind == kind;
579 }
580
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -0800581 private boolean laToken(int idx, int kind, String image) {
582 Token t = getToken(idx);
583 return t.kind == kind && t.image.equalsIgnoreCase(image);
584 }
585
586 private boolean laIdentifier(int idx, String image) {
587 return laToken(idx, IDENTIFIER, image);
588 }
589
590 private boolean laIdentifier(String image) {
591 return laIdentifier(1, image);
592 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700593
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700594 private Token fetchHint(Token token, SqlppHint... expectedHints) {
595 Token hintToken = token.specialToken;
596 if (hintToken == null) {
597 return null;
598 }
599 SourceLocation sourceLoc = getSourceLocation(hintToken);
600 hintCollector.remove(sourceLoc);
601 if (hintToken.hint == null) {
602 warnUnexpectedHint(hintToken.hintParams, sourceLoc, expectedHints);
603 return null;
604 } else if (!ArrayUtils.contains(expectedHints, hintToken.hint)) {
605 warnUnexpectedHint(hintToken.hint.getIdentifier(), sourceLoc, expectedHints);
606 return null;
607 } else {
608 return hintToken;
609 }
610 }
611
612 private void reportUnclaimedHints() {
613 for (Map.Entry<SourceLocation, String> me : hintCollector.entrySet()) {
614 warnUnexpectedHint(me.getValue(), me.getKey(), "None");
615 }
616 }
617
618 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, SqlppHint... expectedHints) {
619 warnUnexpectedHint(actualHint, sourceLoc, StringUtil.join(expectedHints, ", ", "\""));
620 }
621
622 private void warnUnexpectedHint(String actualHint, SourceLocation sourceLoc, String expectedHints) {
Ali Alsulimanaa118862019-09-10 20:55:45 -0700623 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500624 warningCollector.warn(Warning.of(sourceLoc, ErrorCode.UNEXPECTED_HINT, actualHint, expectedHints));
Ali Alsulimanaa118862019-09-10 20:55:45 -0700625 }
Dmitry Lychagin1f6a6e32019-07-02 11:59:53 -0700626 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700627
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800628 private IExpressionAnnotation parseExpressionAnnotation(Token hintToken) {
629 // placeholder for the annotation that should be returned if this hint's parameters cannot be parsed
630 IExpressionAnnotation onParseErrorReturn = null;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800631 try {
632 switch (hintToken.hint) {
633 case HASH_BROADCAST_JOIN_HINT:
634 return new BroadcastExpressionAnnotation(BroadcastExpressionAnnotation.BroadcastSide.RIGHT);
635 case INDEXED_NESTED_LOOP_JOIN_HINT:
636 if (hintToken.hintParams == null) {
637 return IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
638 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800639 // if parameter parsing fails then return hint annotation without parameters
640 onParseErrorReturn = IndexedNLJoinExpressionAnnotation.INSTANCE_ANY_INDEX;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800641 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
642 return IndexedNLJoinExpressionAnnotation.newInstance(indexNames);
643 }
644 case RANGE_HINT:
645 Expression rangeExpr = parseExpression(hintToken.hintParams);
646 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
647 return new RangeAnnotation(rangeMap);
648 case SKIP_SECONDARY_INDEX_SEARCH_HINT:
649 if (hintToken.hintParams == null) {
650 return SkipSecondaryIndexSearchExpressionAnnotation.INSTANCE_ANY_INDEX;
651 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800652 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800653 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
654 return SkipSecondaryIndexSearchExpressionAnnotation.newInstance(indexNames);
655 }
656 case USE_SECONDARY_INDEX_SEARCH_HINT:
657 if (hintToken.hintParams == null) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800658 throw new SqlppParseException(getSourceLocation(hintToken), "Expected index name(s)");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800659 } else {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800660 // if parameter parsing fails then ignore this hint
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800661 List<String> indexNames = parseParenthesizedIdentifierList(hintToken.hintParams);
662 return SecondaryIndexSearchPreferenceAnnotation.newInstance(indexNames);
663 }
664 default:
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800665 throw new SqlppParseException(getSourceLocation(hintToken), "Unexpected hint");
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800666 }
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800667 } catch (SqlppParseException e) {
668 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500669 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800670 hintToken.hint.toString(), e.getMessage()));
671 }
672 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800673 } catch (CompilationException e) {
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800674 if (warningCollector.shouldWarn()) {
Michael Blow7c838de2021-02-23 16:12:48 -0500675 warningCollector.warn(Warning.of(getSourceLocation(hintToken), ErrorCode.INVALID_HINT,
Dmitry Lychagin7c624522020-12-04 18:01:45 -0800676 hintToken.hint.toString(), e.getMessage()));
677 }
678 return onParseErrorReturn;
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -0800679 }
680 }
681
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -0700682 private void ensureNoTypeDeclsInFunction(String fnName, List<Pair<VarIdentifier, TypeExpression>> paramList,
683 TypeExpression returnType, Token startToken) throws SqlppParseException {
684 for (Pair<VarIdentifier, TypeExpression> p : paramList) {
685 if (p.second != null) {
686 String paramName = SqlppVariableUtil.toUserDefinedName(p.first.getValue());
687 throw new SqlppParseException(getSourceLocation(startToken),
688 "Unexpected type declaration for parameter " + paramName + " in function " + fnName);
689 }
690 }
691 if (returnType != null) {
692 throw new SqlppParseException(getSourceLocation(startToken),
693 "Unexpected return type declaration for function " + fnName);
694 }
695 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -0700696
697 private void ensureIntegerLiteral(LiteralExpr expr, String errorPrefix) throws SqlppParseException {
698 Literal lit = expr.getValue();
699 if (lit.getLiteralType() != Literal.Type.INTEGER && lit.getLiteralType() != Literal.Type.LONG) {
700 throw new SqlppParseException(expr.getSourceLocation(), errorPrefix + " should be an INTEGER");
701 }
702 }
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700703
704 private DataverseName createDataverseName(List<String> parts, int fromIndex, int toIndex, Token startToken)
705 throws SqlppParseException {
706 try {
707 return DataverseName.create(parts, fromIndex, toIndex);
708 } catch (AsterixException e) {
709 SqlppParseException pe = new SqlppParseException(getSourceLocation(startToken), e.getMessage());
710 pe.initCause(e);
711 throw pe;
712 }
713 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700714}
715
716PARSER_END(SQLPPParser)
717
Yingyi Bu391f09e2015-10-29 13:49:39 -0700718List<Statement> Statement() throws ParseException:
719{
720 scopeStack.push(RootScopeFactory.createRootScope(this));
721 List<Statement> decls = new ArrayList<Statement>();
722 Statement stmt = null;
723}
724{
Murtadha Hubaildc775222017-08-21 15:22:45 +0300725 (
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700726 (stmt = ExplainStatement()
Murtadha Hubaildc775222017-08-21 15:22:45 +0300727 {
728 decls.add(stmt);
729 }
730 )?
731 (<SEMICOLON>)+
Yingyi Bu391f09e2015-10-29 13:49:39 -0700732 )*
733 <EOF>
734 {
735 return decls;
736 }
737}
738
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700739Statement ExplainStatement() throws ParseException:
740{
741 Statement stmt = null;
742 Token explainToken = null;
743}
744{
745 ( <EXPLAIN> { explainToken = token; } )?
746 stmt = SingleStatement()
747 {
748 if (explainToken != null) {
749 if (stmt.getKind() == Statement.Kind.QUERY) {
750 ((Query)stmt).setExplain(true);
751 } else {
752 throw new SqlppParseException(getSourceLocation(explainToken),
753 "EXPLAIN is not supported for this kind of statement");
754 }
755 }
756 return stmt;
757 }
758}
759
Yingyi Bu391f09e2015-10-29 13:49:39 -0700760Statement SingleStatement() throws ParseException:
761{
762 Statement stmt = null;
763}
764{
765 (
766 stmt = DataverseDeclaration()
767 | stmt = FunctionDeclaration()
768 | stmt = CreateStatement()
769 | stmt = LoadStatement()
770 | stmt = DropStatement()
771 | stmt = WriteStatement()
772 | stmt = SetStatement()
773 | stmt = InsertStatement()
774 | stmt = DeleteStatement()
775 | stmt = UpdateStatement()
Yingyi Bucb5bf332017-01-02 22:19:50 -0800776 | stmt = UpsertStatement()
Yingyi Buab817482016-08-19 21:29:31 -0700777 | stmt = ConnectionStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700778 | stmt = CompactStatement()
Dmitry Lychaginaebd0312019-06-04 08:55:39 -0700779 | stmt = Query()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700780 | stmt = RefreshExternalDatasetStatement()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700781 )
782 {
783 return stmt;
784 }
785}
786
787DataverseDecl DataverseDeclaration() throws ParseException:
788{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700789 Token startToken = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700790 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700791}
792{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700793 <USE> { startToken = token; } dvName = DataverseName()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700794 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -0700795 defaultDataverse = dvName;
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800796 DataverseDecl dvDecl = new DataverseDecl(defaultDataverse);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -0700797 return addSourceLocation(dvDecl, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700798 }
799}
800
801Statement CreateStatement() throws ParseException:
802{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700803 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700804 Statement stmt = null;
805}
806{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700807 <CREATE> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700808 (
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700809 stmt = CreateOrReplaceStatement(startToken)
810 | stmt = CreateTypeStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800811 | stmt = CreateNodegroupStatement(startToken)
812 | stmt = CreateDatasetStatement(startToken)
813 | stmt = CreateIndexStatement(startToken)
814 | stmt = CreateDataverseStatement(startToken)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700815 | stmt = CreateFunctionStatement(startToken, false)
Ian Maxon38fe9402020-01-29 19:27:40 -0800816 | stmt = CreateAdapterStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -0800817 | stmt = CreateSynonymStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800818 | stmt = CreateFeedStatement(startToken)
819 | stmt = CreateFeedPolicyStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -0800820 | stmt = CreateFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700821 | stmt = CreateViewStatement(startToken, false)
Yingyi Bu391f09e2015-10-29 13:49:39 -0700822 )
823 {
824 return stmt;
825 }
826}
827
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700828Statement CreateOrReplaceStatement(Token startStmtToken) throws ParseException:
829{
830 Statement stmt = null;
831 Token replaceToken = null;
832}
833{
834 <OR> <IDENTIFIER> { replaceToken = token; }
835 (
836 stmt = CreateFunctionStatement(startStmtToken, true)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -0700837 | stmt = CreateViewStatement(startStmtToken, true)
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -0700838 )
839 {
840 // check expected token here to make the grammar extension plugin happy
841 expectToken(replaceToken, REPLACE);
842 return stmt;
843 }
844}
845
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800846TypeDecl CreateTypeStatement(Token startStmtToken) throws ParseException:
847{
848 TypeDecl stmt = null;
849}
850{
851 <TYPE> stmt = TypeSpecification(startStmtToken)
852 {
853 return stmt;
854 }
855}
856
Dmitry Lychagine77cdac2019-08-05 11:37:47 -0700857TypeDecl TypeSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700858{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800859 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700860 boolean ifNotExists = false;
861 TypeExpression typeExpr = null;
862}
863{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800864 nameComponents = TypeName() ifNotExists = IfNotExists()
Till Westmannf6028272016-09-30 14:34:42 -0700865 <AS> typeExpr = RecordTypeDef()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800866 {
867 boolean dgen = false;
868 long numValues = -1;
869 String filename = null;
870 Token hintToken = fetchHint(startStmtToken, SqlppHint.DGEN_HINT);
871 if (hintToken != null) {
872 String hintParams = hintToken.hintParams;
873 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
874 if (splits == null || splits.length != 2) {
875 throw new SqlppParseException(getSourceLocation(hintToken),
876 "Expecting /*+ dgen <filename> <numberOfItems> */");
877 }
878 dgen = true;
879 filename = splits[0];
880 numValues = Long.parseLong(splits[1]);
Yingyi Bu391f09e2015-10-29 13:49:39 -0700881 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800882 TypeDataGen tddg = new TypeDataGen(dgen, filename, numValues);
883 TypeDecl stmt = new TypeDecl(nameComponents.first, nameComponents.second, typeExpr, tddg, ifNotExists);
884 return addSourceLocation(stmt, startStmtToken);
885 }
886}
887
888NodegroupDecl CreateNodegroupStatement(Token startStmtToken) throws ParseException:
889{
890 NodegroupDecl stmt = null;
891}
892{
893 <NODEGROUP> stmt = NodegroupSpecification(startStmtToken)
894 {
895 return stmt;
896 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700897}
898
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700899NodegroupDecl NodegroupSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700900{
901 String name = null;
902 String tmp = null;
903 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800904 List<Identifier> ncNames = new ArrayList<Identifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -0700905}
906{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800907 name = Identifier() ifNotExists = IfNotExists()
908 <ON> tmp = Identifier()
909 {
910 ncNames.add(new Identifier(tmp));
911 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700912 ( <COMMA> tmp = Identifier()
913 {
914 ncNames.add(new Identifier(tmp));
915 }
916 )*
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800917 {
918 NodegroupDecl stmt = new NodegroupDecl(new Identifier(name), ncNames, ifNotExists);
919 return addSourceLocation(stmt, startStmtToken);
920 }
921}
922
923void Dataset() throws ParseException:
924{
925}
926{
927 (<DATASET>|<COLLECTION>)
928}
929
930DatasetDecl CreateDatasetStatement(Token startStmtToken) throws ParseException:
931{
932 DatasetDecl stmt = null;
933}
934{
935 (
936 (<INTERNAL>)? Dataset() stmt = DatasetSpecification(startStmtToken)
937 | <EXTERNAL> Dataset() stmt = ExternalDatasetSpecification(startStmtToken)
938 )
939 {
940 return stmt;
941 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700942}
943
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700944DatasetDecl DatasetSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -0700945{
Dmitry Lychagin54c06012019-11-13 15:54:20 -0800946 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700947 boolean ifNotExists = false;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -0700948 TypeExpression typeExpr = null;
949 TypeExpression metaTypeExpr = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800950 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700951 Map<String,String> hints = new HashMap<String,String>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -0700952 DatasetDecl stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700953 boolean autogenerated = false;
Yingyi Buc9bfe252016-03-01 00:02:40 -0800954 Pair<Integer, List<String>> filterField = null;
Till Westmannf3aa19f2017-12-01 17:42:35 -0800955 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -0700956}
957{
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800958 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700959 typeExpr = DatasetTypeSpecification()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700960 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800961 { String name; }
962 <WITH>
963 name = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -0700964 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800965 if (!name.equalsIgnoreCase("meta")){
966 throw new SqlppParseException(getSourceLocation(startStmtToken),
967 "We can only support one additional associated field called \"meta\".");
968 }
Yingyi Bu391f09e2015-10-29 13:49:39 -0700969 }
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700970 metaTypeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800971 )?
972 ifNotExists = IfNotExists()
973 primaryKeyFields = PrimaryKey()
974 (<AUTOGENERATED> { autogenerated = true; } )?
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800975 ( <HINTS> hints = Properties() )?
976 ( LOOKAHEAD(2) <WITH> <FILTER> <ON> filterField = NestedField() )?
977 ( <WITH> withRecord = RecordConstructor() )?
978 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800979 try {
Xikui Wangca7927f2020-09-03 11:27:57 -0700980 InternalDetailsDecl idd = new InternalDetailsDecl(primaryKeyFields.second, primaryKeyFields.first, autogenerated,
981 filterField == null? null : filterField.first, filterField == null? null : filterField.second);
Murtadha Hubail353e95f2020-08-24 22:18:04 +0300982 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, metaTypeExpr, hints,
983 DatasetType.INTERNAL, idd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800984 return addSourceLocation(stmt, startStmtToken);
985 } catch (CompilationException e) {
986 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
987 }
988 }
989}
990
991DatasetDecl ExternalDatasetSpecification(Token startStmtToken) throws ParseException:
992{
993 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -0700994 TypeExpression typeExpr = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800995 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800996 String adapterName = null;
997 Map<String,String> properties = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -0800998 Map<String,String> hints = new HashMap<String,String>();
999 DatasetDecl stmt = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001000 RecordConstructor withRecord = null;
1001}
1002{
1003 nameComponents = QualifiedName()
Dmitry Lychagin5cc254a2020-05-08 17:12:47 -07001004 typeExpr = DatasetTypeSpecification()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001005 ifNotExists = IfNotExists()
1006 <USING> adapterName = AdapterName() properties = Configuration()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001007 ( <HINTS> hints = Properties() )?
1008 ( <WITH> withRecord = RecordConstructor() )?
1009 {
1010 ExternalDetailsDecl edd = new ExternalDetailsDecl();
1011 edd.setAdapter(adapterName);
1012 edd.setProperties(properties);
1013 try {
Murtadha Hubail353e95f2020-08-24 22:18:04 +03001014 stmt = new DatasetDecl(nameComponents.first, nameComponents.second, typeExpr, null, hints, DatasetType.EXTERNAL,
1015 edd, withRecord, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001016 return addSourceLocation(stmt, startStmtToken);
1017 } catch (CompilationException e) {
1018 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1019 }
1020 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001021}
1022
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001023TypeExpression DatasetTypeSpecification() throws ParseException:
1024{
1025 TypeExpression typeExpr = null;
1026}
1027{
1028 (
1029 LOOKAHEAD(3) typeExpr = DatasetRecordTypeSpecification(true)
1030 | typeExpr = DatasetReferenceTypeSpecification()
1031 )
1032 {
1033 return typeExpr;
1034 }
1035}
1036
1037TypeExpression DatasetReferenceTypeSpecification() throws ParseException:
1038{
1039 TypeExpression typeExpr = null;
1040}
1041{
1042 <LEFTPAREN> typeExpr = TypeReference() <RIGHTPAREN>
1043 {
1044 return typeExpr;
1045 }
1046}
1047
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001048RecordTypeDefinition DatasetRecordTypeSpecification(boolean allowRecordKindModifier) throws ParseException:
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001049{
1050 RecordTypeDefinition recordTypeDef = null;
1051 RecordTypeDefinition.RecordKind recordKind = null;
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001052 Token startToken = null, recordKindToken = null;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001053}
1054{
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001055 <LEFTPAREN> { startToken = token; } recordTypeDef = DatasetRecordTypeDef() <RIGHTPAREN>
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001056 ( recordKind = RecordTypeKind() { recordKindToken = token; } <TYPE> )?
1057 {
1058 if (recordKind == null) {
1059 recordKind = RecordTypeDefinition.RecordKind.CLOSED;
1060 } else if (!allowRecordKindModifier) {
1061 throw createUnexpectedTokenError(recordKindToken);
1062 }
1063 recordTypeDef.setRecordKind(recordKind);
Dmitry Lychagin6c239e72020-06-19 19:46:27 -07001064 return addSourceLocation(recordTypeDef, startToken);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001065 }
1066}
1067
1068RecordTypeDefinition DatasetRecordTypeDef() throws ParseException:
1069{
1070 RecordTypeDefinition recType = new RecordTypeDefinition();
1071}
1072{
1073 DatasetRecordField(recType) ( <COMMA> DatasetRecordField(recType) )*
1074 {
1075 return recType;
1076 }
1077}
1078
1079void DatasetRecordField(RecordTypeDefinition recType) throws ParseException:
1080{
1081 String fieldName;
1082 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001083 boolean nullable = true, missable = true;
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001084}
1085{
1086 fieldName = Identifier()
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001087 type = TypeReference() ( <NOT> <UNKNOWN> { nullable = false; missable = false; } )?
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001088 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07001089 recType.addField(fieldName, type, nullable, missable);
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07001090 }
1091}
1092
Yingyi Bu391f09e2015-10-29 13:49:39 -07001093RefreshExternalDatasetStatement RefreshExternalDatasetStatement() throws ParseException:
1094{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001095 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001096 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001097 String datasetName = null;
1098}
1099{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001100 <REFRESH> { startToken = token; } <EXTERNAL> Dataset() nameComponents = QualifiedName()
1101 {
1102 RefreshExternalDatasetStatement stmt = new RefreshExternalDatasetStatement();
1103 stmt.setDataverseName(nameComponents.first);
1104 stmt.setDatasetName(nameComponents.second);
1105 return addSourceLocation(stmt, startToken);
1106 }
1107}
1108
1109CreateIndexStatement CreateIndexStatement(Token startStmtToken) throws ParseException:
1110{
1111 CreateIndexStatement stmt = null;
1112}
1113{
1114 (
1115 <INDEX> stmt = IndexSpecification(startStmtToken)
1116 | <PRIMARY> <INDEX> stmt = PrimaryIndexSpecification(startStmtToken)
1117 )
1118 {
1119 return stmt;
1120 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001121}
1122
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001123CreateIndexStatement IndexSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001124{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001125 Pair<DataverseName,Identifier> nameComponents = null;
Glenn67fd1f32021-02-25 16:04:49 -08001126 String indexName = null;
1127 IndexParams indexParams = null;
1128 CreateIndexStatement.IndexedElement indexedElement = null;
1129 List<CreateIndexStatement.IndexedElement> indexedElementList = new ArrayList<CreateIndexStatement.IndexedElement>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001130 boolean enforced = false;
Glenn67fd1f32021-02-25 16:04:49 -08001131 boolean ifNotExists = false;
1132 boolean hasUnnest = false;
1133 String fullTextConfigName = null;
1134 Token startElementToken = null;
Ali Alsuliman931e7382021-08-08 21:55:59 +03001135 Boolean excludeUnknown = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001136}
1137{
Ali Alsuliman8351d252017-09-24 00:43:15 -07001138 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001139 indexName = Identifier() ifNotExists = IfNotExists()
Ali Alsuliman8351d252017-09-24 00:43:15 -07001140 <ON> nameComponents = QualifiedName()
Glenn67fd1f32021-02-25 16:04:49 -08001141 <LEFTPAREN> { startElementToken = token; }
1142 indexedElement = IndexedElement(startElementToken) {
1143 indexedElementList.add(indexedElement);
1144 hasUnnest |= indexedElement.hasUnnest();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001145 }
Glenn67fd1f32021-02-25 16:04:49 -08001146 (<COMMA> { startElementToken = token; }
1147 indexedElement = IndexedElement(startElementToken) {
1148 indexedElementList.add(indexedElement);
1149 hasUnnest |= indexedElement.hasUnnest();
1150 }
1151 )*
1152 <RIGHTPAREN>
1153 ( <TYPE> indexParams = IndexType() )? ( <ENFORCED> { enforced = true; } )?
Ali Alsuliman931e7382021-08-08 21:55:59 +03001154 ( <IDENTIFIER>
1155 {
1156 if (isToken(EXCLUDE)) {
1157 excludeUnknown = true;
1158 } else if (isToken(INCLUDE)) {
1159 excludeUnknown = false;
1160 } else {
1161 throw createUnexpectedTokenError();
1162 }
1163 } <UNKNOWN> <KEY>
1164 )?
Ali Alsuliman8351d252017-09-24 00:43:15 -07001165 )
1166 {
Glenn67fd1f32021-02-25 16:04:49 -08001167 IndexType indexType;
1168 int gramLength;
1169 if (indexParams != null) {
1170 indexType = indexParams.type;
1171 gramLength = indexParams.gramLength;
1172 fullTextConfigName = indexParams.fullTextConfig;
1173 } else {
1174 indexType = hasUnnest ? IndexType.ARRAY : IndexType.BTREE;
1175 gramLength = -1;
1176 fullTextConfigName = null;
Ali Alsuliman8351d252017-09-24 00:43:15 -07001177 }
Glenn67fd1f32021-02-25 16:04:49 -08001178 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman931e7382021-08-08 21:55:59 +03001179 new Identifier(indexName), indexType, indexedElementList, enforced, gramLength, fullTextConfigName, ifNotExists,
1180 excludeUnknown);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001181 return addSourceLocation(stmt, startStmtToken);
Ali Alsuliman8351d252017-09-24 00:43:15 -07001182 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001183}
1184
Glenn67fd1f32021-02-25 16:04:49 -08001185CreateIndexStatement.IndexedElement IndexedElement(Token startElementToken) throws ParseException:
1186{
1187 Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1188 Pair<List<String>, IndexedTypeExpression> elementSimple = null;
1189 int elementSimpleSource = 0;
1190}
1191{
1192 (
1193 element = IndexedElementUnnestSelect()
1194 | (
1195 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1196 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1197 <DOT> elementSimple = IndexedField()
1198 { elementSimpleSource = 1; }
1199 )
1200 | elementSimple = IndexedField()
1201 | <LEFTPAREN> ( element = IndexedElementUnnestSelect() | elementSimple = IndexedField() ) <RIGHTPAREN>
1202 )
1203 {
1204 int source;
1205 List<List<String>> unnestList;
1206 List<Pair<List<String>, IndexedTypeExpression>> projectList;
1207 if (elementSimple != null) {
1208 source = elementSimpleSource;
1209 unnestList = null;
1210 projectList = Collections.singletonList(elementSimple);
1211 } else {
1212 source = element.first;
1213 unnestList = element.second;
1214 projectList = element.third;
1215 }
1216 CreateIndexStatement.IndexedElement ie = new CreateIndexStatement.IndexedElement(source, unnestList, projectList);
1217 ie.setSourceLocation(getSourceLocation(startElementToken));
1218 return ie;
1219 }
1220}
1221
1222Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelect()
1223 throws ParseException:
1224{
1225 int source = 0;
1226 Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> element = null;
1227}
1228{
1229 <UNNEST>
1230 (
1231 (
1232 LOOKAHEAD({ laIdentifier(META) && laToken(2, LEFTPAREN) && laToken(3, RIGHTPAREN) })
1233 <IDENTIFIER> { expectToken(META); } <LEFTPAREN> <RIGHTPAREN>
1234 <DOT> element = IndexedElementUnnestSelectBody() { source = 1; }
1235 ) | element = IndexedElementUnnestSelectBody()
1236 )
1237 {
1238 return new Triple<Integer, List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(
1239 source, element.first, element.second
1240 );
1241 }
1242}
1243
1244Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>> IndexedElementUnnestSelectBody()
1245 throws ParseException:
1246{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001247 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001248 IndexedTypeExpression type = null;
1249 List<List<String>> unnestList = new ArrayList();
Glenn67fd1f32021-02-25 16:04:49 -08001250 List<Pair<List<String>, IndexedTypeExpression>> projectList = new ArrayList();
1251}
1252{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001253 path = MultipartIdentifier() { unnestList.add(path.first); }
1254 ( <UNNEST> path = MultipartIdentifier() { unnestList.add(path.first); })*
Glenn67fd1f32021-02-25 16:04:49 -08001255 (
Glenn58329202021-04-09 12:03:46 -07001256 ( <COLON> type = IndexedTypeExpr(false)
Glenn67fd1f32021-02-25 16:04:49 -08001257 {
1258 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, type));
1259 }
1260 ) |
1261 (
Glenn58329202021-04-09 12:03:46 -07001262 <SELECT> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1263 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001264 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001265 }
1266 ( <COMMA> path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(false) )?
1267 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001268 projectList.add(new Pair<List<String>, IndexedTypeExpression>(path.first, type));
Glenn58329202021-04-09 12:03:46 -07001269 }
1270 )*
Glenn67fd1f32021-02-25 16:04:49 -08001271 )
1272 )?
1273 {
1274 if (projectList.isEmpty()) {
1275 // To support the case (<UNNEST> IDENTIFIER)* IDENTIFIER w/o any type specification.
1276 projectList.add(new Pair<List<String>, IndexedTypeExpression>(null, null));
1277 }
1278
1279 return new Pair<List<List<String>>, List<Pair<List<String>, IndexedTypeExpression>>>(unnestList, projectList);
1280 }
1281}
1282
1283Pair<List<String>, IndexedTypeExpression> IndexedField() throws ParseException:
1284{
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001285 Triple<List<String>, Token, Token> path = null;
Glenn67fd1f32021-02-25 16:04:49 -08001286 IndexedTypeExpression type = null;
1287}
1288{
Glenn58329202021-04-09 12:03:46 -07001289 path = MultipartIdentifier() ( <COLON> type = IndexedTypeExpr(true) )?
Glenn67fd1f32021-02-25 16:04:49 -08001290 {
Dmitry Lychaginb9393132021-04-12 17:21:22 -07001291 return new Pair<List<String>, IndexedTypeExpression>(path.first, type);
Glenn67fd1f32021-02-25 16:04:49 -08001292 }
1293}
1294
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001295CreateIndexStatement PrimaryIndexSpecification(Token startStmtToken) throws ParseException:
1296{
Glenn67fd1f32021-02-25 16:04:49 -08001297 Pair<DataverseName,Identifier> nameComponents = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001298 String indexName = null;
1299 boolean ifNotExists = false;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001300}
1301{
1302 (indexName = Identifier())? ifNotExists = IfNotExists()
1303 <ON> nameComponents = QualifiedName() (<TYPE> <BTREE>)?
1304 {
1305 if (indexName == null) {
1306 indexName = "primary_idx_" + nameComponents.second;
1307 }
Glenn67fd1f32021-02-25 16:04:49 -08001308 CreateIndexStatement stmt = new CreateIndexStatement(nameComponents.first, nameComponents.second,
Ali Alsuliman931e7382021-08-08 21:55:59 +03001309 new Identifier(indexName), IndexType.BTREE, Collections.emptyList(), false, -1, null, ifNotExists, null);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001310 return addSourceLocation(stmt, startStmtToken);
1311 }
1312}
1313
Yingyi Bu391f09e2015-10-29 13:49:39 -07001314String FilterField() throws ParseException :
1315{
1316 String filterField = null;
1317}
1318{
1319 filterField = Identifier()
1320 {
1321 return filterField;
1322 }
1323}
1324
1325IndexParams IndexType() throws ParseException:
1326{
1327 IndexType type = null;
1328 int gramLength = 0;
Rui Guoe6986dd2020-11-25 19:50:06 -08001329 String fullTextConfig = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001330}
1331{
1332 (<BTREE>
1333 {
1334 type = IndexType.BTREE;
1335 }
1336 | <RTREE>
1337 {
1338 type = IndexType.RTREE;
1339 }
1340 | <KEYWORD>
1341 {
1342 type = IndexType.LENGTH_PARTITIONED_WORD_INVIX;
1343 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001344 | <FULLTEXT>
Taewoo Kimc49405a2017-01-04 00:30:43 -08001345 {
1346 type = IndexType.SINGLE_PARTITION_WORD_INVIX;
1347 }
Rui Guoe6986dd2020-11-25 19:50:06 -08001348 // For now we don't allow inverted index creation using a full-text config in another data verse.
1349 // We may want to support corss-dataverse full-text config access later
1350 // If so, replace the Identifier() with QualifiedName() to get the dataverse name
1351 ( <USING> Identifier()
1352 {
1353 fullTextConfig = token.image;
1354 }
1355 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001356 | <NGRAM> <LEFTPAREN> <INTEGER_LITERAL>
1357 {
1358 type = IndexType.LENGTH_PARTITIONED_NGRAM_INVIX;
1359 gramLength = Integer.valueOf(token.image);
1360 }
1361 <RIGHTPAREN>)
1362 {
Rui Guoe6986dd2020-11-25 19:50:06 -08001363 return new IndexParams(type, gramLength, fullTextConfig);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001364 }
1365}
1366
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001367CreateDataverseStatement CreateDataverseStatement(Token startStmtToken) throws ParseException :
1368{
1369 CreateDataverseStatement stmt = null;
1370}
1371{
1372 <DATAVERSE> stmt = DataverseSpecification(startStmtToken)
1373 {
1374 return stmt;
1375 }
1376}
1377
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001378CreateDataverseStatement DataverseSpecification(Token startStmtToken) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07001379{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001380 DataverseName dvName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001381 boolean ifNotExists = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001382}
1383{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001384 dvName = DataverseName() ifNotExists = IfNotExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001385 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07001386 CreateDataverseStatement stmt = new CreateDataverseStatement(dvName, null, ifNotExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001387 return addSourceLocation(stmt, startStmtToken);
1388 }
1389}
1390
Ian Maxon38fe9402020-01-29 19:27:40 -08001391CreateAdapterStatement CreateAdapterStatement(Token startStmtToken) throws ParseException:
1392{
1393 CreateAdapterStatement stmt = null;
1394}
1395{
1396 <ADAPTER> stmt = AdapterSpecification(startStmtToken)
1397 {
1398 return stmt;
1399 }
1400}
1401
1402CreateAdapterStatement AdapterSpecification(Token startStmtToken) throws ParseException:
1403{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001404 Pair<DataverseName,Identifier> adapterName = null;
1405 Pair<DataverseName,Identifier> libraryName = null;
1406 List<String> externalIdentifier = null;
Ian Maxon38fe9402020-01-29 19:27:40 -08001407 boolean ifNotExists = false;
1408}
1409{
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001410 adapterName = QualifiedName()
1411 ifNotExists = IfNotExists()
1412 <AS> externalIdentifier = FunctionExternalIdentifier()
1413 <AT> libraryName = QualifiedName()
1414 {
1415 CreateAdapterStatement stmt = new CreateAdapterStatement(adapterName.first, adapterName.second.getValue(),
1416 libraryName.first, libraryName.second.getValue(), externalIdentifier, ifNotExists);
1417 return addSourceLocation(stmt, startStmtToken);
1418 }
1419}
Ian Maxon38fe9402020-01-29 19:27:40 -08001420
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001421CreateViewStatement CreateViewStatement(Token startStmtToken, boolean orReplace) throws ParseException:
1422{
1423 CreateViewStatement stmt = null;
1424}
1425{
1426 <VIEW> stmt = ViewSpecification(startStmtToken, orReplace)
1427 {
1428 return stmt;
1429 }
1430}
1431
1432CreateViewStatement ViewSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
1433{
1434 Pair<DataverseName, Identifier> nameComponents = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001435 TypeExpression typeExpr = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001436 boolean ifNotExists = false;
1437 Token beginPos = null, endPos = null;
1438 Expression viewBodyExpr = null;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001439 Boolean defaultNull = null;
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001440 Map<String, String> viewConfig = null;
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001441 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001442 DataverseName currentDataverse = defaultDataverse;
1443}
1444{
1445 nameComponents = QualifiedName()
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001446 (
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001447 (
1448 typeExpr = DatasetTypeSpecification()
1449 ifNotExists = IfNotExists()
1450 <IDENTIFIER> { expectToken(DEFAULT); } <NULL> { defaultNull = true; }
1451 viewConfig = ViewConfiguration()
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001452 ( <PRIMARY> <KEY> <LEFTPAREN> primaryKeyFields = PrimaryKeyFields() <RIGHTPAREN> <NOT> <ENFORCED> )?
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001453 )
1454 |
1455 ( ifNotExists = IfNotExists() )
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001456 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001457 {
1458 if (orReplace && ifNotExists) {
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001459 throw new SqlppParseException(getSourceLocation(startStmtToken), "Unexpected IF NOT EXISTS");
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001460 }
1461 }
1462 <AS>
1463 {
1464 beginPos = token;
1465 createNewScope();
1466 if (nameComponents.first != null) {
1467 defaultDataverse = nameComponents.first;
1468 }
1469 }
1470 viewBodyExpr = ViewBody()
1471 {
1472 endPos = token;
1473 String viewBody = extractFragment(beginPos.beginLine, beginPos.beginColumn + 1, endPos.endLine,
1474 endPos.endColumn + 1);
1475 removeCurrentScope();
1476 defaultDataverse = currentDataverse;
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001477 try {
1478 CreateViewStatement stmt = new CreateViewStatement(nameComponents.first, nameComponents.second.getValue(),
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001479 typeExpr, viewBody, viewBodyExpr, defaultNull, viewConfig, primaryKeyFields, orReplace, ifNotExists);
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001480 return addSourceLocation(stmt, startStmtToken);
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001481 } catch (CompilationException e) {
1482 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1483 }
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001484 }
1485}
1486
1487Expression ViewBody() throws ParseException:
1488{
1489 Expression viewBodyExpr = null;
1490}
1491{
Dmitry Lychagin33c77f92021-07-21 15:59:04 -07001492 (
1493 ( viewBodyExpr = VariableRef() ( viewBodyExpr = FieldAccessor(viewBodyExpr) )* )
1494 | viewBodyExpr = SelectExpression(true)
1495 )
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001496 {
1497 return viewBodyExpr;
1498 }
1499}
1500
Dmitry Lychagin1f6f24e2021-08-19 21:41:57 -07001501Map<String, String> ViewConfiguration() throws ParseException:
1502{
1503 Map<String, String> config = null;
1504 String name = null, value = null;
1505}
1506{
1507 (
1508 <IDENTIFIER> { name = token.image.toLowerCase(); }
1509 value = StringLiteral()
1510 {
1511 if (config == null) {
1512 config = new LinkedHashMap<String, String>();
1513 }
1514 config.put(name, value);
1515 }
1516 )*
1517 {
1518 return config;
1519 }
1520}
1521
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001522CreateFunctionStatement CreateFunctionStatement(Token startStmtToken, boolean orReplace) throws ParseException:
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001523{
1524 CreateFunctionStatement stmt = null;
1525}
1526{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001527 <FUNCTION> stmt = FunctionSpecification(startStmtToken, orReplace)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001528 {
1529 return stmt;
1530 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001531}
1532
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001533CreateFunctionStatement FunctionSpecification(Token startStmtToken, boolean orReplace) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001534{
Ian Maxon38fe9402020-01-29 19:27:40 -08001535 FunctionSignature signature = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001536 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001537 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001538 List<Pair<VarIdentifier,TypeExpression>> params = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001539 int arity = 0;
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001540 TypeExpression returnType = null;
1541 Token beginPos = null, endPos = null;
1542 Expression functionBodyExpr = null;
1543 Pair<DataverseName,Identifier> libraryName = null;
1544 List<String> externalIdentifier = null;
1545 RecordConstructor withOptions = null;
1546 boolean ifNotExists = false;
1547 CreateFunctionStatement stmt = null;
1548 DataverseName currentDataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001549}
1550{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001551 fctName = FunctionName()
Steven Glenn Jacobs665e9fe2017-12-27 10:30:39 -08001552 {
1553 defaultDataverse = fctName.dataverse;
1554 }
Dmitry Lychagin541652082020-11-09 14:04:53 -08001555 paramsWithArity = FunctionParameters()
1556 {
1557 arity = paramsWithArity.first;
1558 params = paramsWithArity.second;
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001559 signature = new FunctionSignature(fctName.dataverse, fctName.function, arity);
Dmitry Lychagin541652082020-11-09 14:04:53 -08001560 }
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001561 ifNotExists = IfNotExists()
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07001562 {
1563 if (orReplace && ifNotExists) {
1564 throw new SqlppParseException(getSourceLocation(token), "Unexpected IF NOT EXISTS");
1565 }
1566 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001567 returnType = FunctionReturnType()
Ian Maxon38fe9402020-01-29 19:27:40 -08001568 (
1569 (
1570 <LEFTBRACE>
1571 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001572 beginPos = token;
1573 createNewScope();
Ian Maxon38fe9402020-01-29 19:27:40 -08001574 }
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001575 functionBodyExpr = FunctionBody()
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001576 <RIGHTBRACE>
1577 {
1578 endPos = token;
1579 String functionBody = extractFragment(beginPos.beginLine, beginPos.beginColumn, endPos.beginLine,
1580 endPos.beginColumn);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001581 getCurrentScope().addFunctionDescriptor(signature, false);
1582 removeCurrentScope();
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001583 ensureNoTypeDeclsInFunction(fctName.function, params, returnType, startStmtToken);
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001584 stmt = new CreateFunctionStatement(signature, params, functionBody, functionBodyExpr, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001585 }
Ian Maxon38fe9402020-01-29 19:27:40 -08001586 )
1587 |
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001588 (
1589 <AS> externalIdentifier = FunctionExternalIdentifier()
1590 <AT> libraryName = QualifiedName()
1591 (<WITH> withOptions = RecordConstructor())?
1592 {
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001593 try {
1594 stmt = new CreateFunctionStatement(signature, params, returnType, libraryName.first,
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001595 libraryName.second.getValue(), externalIdentifier, withOptions, orReplace, ifNotExists);
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001596 } catch (AlgebricksException e) {
1597 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1598 }
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001599 }
1600 )
Ian Maxon38fe9402020-01-29 19:27:40 -08001601 )
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001602 {
Dmitry Lychagin265e3ce2021-03-02 12:22:38 -08001603 defaultDataverse = currentDataverse;
1604 return addSourceLocation(stmt, startStmtToken);
1605 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001606}
1607
Dmitry Lychagin541652082020-11-09 14:04:53 -08001608Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameters() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001609{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001610 Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> params = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001611}
1612{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001613 <LEFTPAREN> (params = FunctionParameterList())? <RIGHTPAREN>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001614 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001615 if (params == null) {
1616 params = new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(
1617 0, Collections.<Pair<VarIdentifier, TypeExpression>>emptyList()
1618 );
1619 }
1620 return params;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001621 }
1622}
1623
Dmitry Lychagin541652082020-11-09 14:04:53 -08001624Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>> FunctionParameterList() :
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001625{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001626 List<Pair<VarIdentifier, TypeExpression>> paramList = null;
1627 Pair<VarIdentifier, TypeExpression> param = null;
1628 int arity = 0;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001629}
1630{
Dmitry Lychagin541652082020-11-09 14:04:53 -08001631 (
1632 (
1633 <DOT> <DOT> <DOT>
1634 {
1635 param = new Pair<VarIdentifier, TypeExpression>(
1636 SqlppVariableUtil.toInternalVariableIdentifier(UDF_VARARGS_PARAM_NAME), null
1637 );
1638 paramList = Collections.<Pair<VarIdentifier, TypeExpression>>singletonList(param);
1639 arity = FunctionIdentifier.VARARGS;
1640 }
1641 )
1642 |
1643 (
1644 param = FunctionParameter()
1645 {
1646 paramList = new ArrayList<Pair<VarIdentifier, TypeExpression>>();
1647 paramList.add(param);
1648 }
1649 ( <COMMA> param = FunctionParameter() { paramList.add(param); } )*
1650 {
1651 arity = paramList.size();
1652 }
1653 )
1654 )
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001655 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08001656 return new Pair<Integer, List<Pair<VarIdentifier, TypeExpression>>>(arity, paramList);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001657 }
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001658}
1659
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001660Pair<VarIdentifier,TypeExpression> FunctionParameter() throws ParseException:
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001661{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001662 String name = null;
1663 TypeExpression type = null;
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001664}
1665{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07001666 name = VariableIdentifier()
1667 ( LOOKAHEAD(3) (<COLON>)? type = TypeExpr(false) )?
1668 {
1669 return new Pair<VarIdentifier,TypeExpression>(new VarIdentifier(name), type);
1670 }
1671}
1672
1673TypeExpression FunctionReturnType() throws ParseException:
1674{
1675 TypeExpression returnType = null;
1676}
1677{
1678 ( LOOKAHEAD({laIdentifier(RETURNS)}) <IDENTIFIER> returnType = TypeExpr(false) )?
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08001679 {
1680 return returnType;
1681 }
1682}
1683
Dmitry Lychagin45de2342020-02-03 12:01:28 -08001684Expression FunctionBody() throws ParseException:
1685{
1686 Expression functionBodyExpr = null;
1687}
1688{
1689 ( functionBodyExpr = SelectExpression(true) | functionBodyExpr = Expression() )
1690 {
1691 return functionBodyExpr;
1692 }
1693}
1694
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07001695List<String> FunctionExternalIdentifier() throws ParseException:
1696{
1697 String ident = null;
1698 List<String> identList = new ArrayList(2);
1699}
1700{
1701 ident = StringLiteral() { identList.add(ident.trim()); }
1702 ( <COMMA> ident = StringLiteral() { identList.add(ident.trim()); } )*
1703 {
1704 return identList;
1705 }
1706}
1707
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001708CreateFeedStatement CreateFeedStatement(Token startStmtToken) throws ParseException:
1709{
1710 CreateFeedStatement stmt = null;
1711}
1712{
1713 <FEED> stmt = FeedSpecification(startStmtToken)
1714 {
1715 return stmt;
1716 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001717}
1718
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001719CreateFeedStatement FeedSpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001720{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001721 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001722 boolean ifNotExists = false;
1723 String adapterName = null;
1724 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001725 CreateFeedStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001726 Pair<Identifier,Identifier> sourceNameComponents = null;
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001727 RecordConstructor withRecord = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001728}
1729{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001730 nameComponents = QualifiedName() ifNotExists = IfNotExists()
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001731 <WITH> withRecord = RecordConstructor()
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001732 {
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001733 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001734 stmt = new CreateFeedStatement(nameComponents, withRecord, ifNotExists);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07001735 return addSourceLocation(stmt, startStmtToken);
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001736 } catch (AlgebricksException e) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001737 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
Xikui Wang5a61b2a2018-01-02 14:14:03 -08001738 }
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08001739 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001740}
1741
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001742CreateFeedPolicyStatement CreateFeedPolicyStatement(Token startStmtToken) throws ParseException:
1743{
1744 CreateFeedPolicyStatement stmt = null;
1745}
1746{
1747 <INGESTION> <POLICY> stmt = FeedPolicySpecification(startStmtToken)
1748 {
1749 return stmt;
1750 }
1751}
1752
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001753CreateFeedPolicyStatement FeedPolicySpecification(Token startStmtToken) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001754{
Michael Blowd6cf6412016-06-30 02:44:35 -04001755 String policyName = null;
1756 String basePolicyName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001757 String sourcePolicyFile = null;
1758 String definition = null;
1759 boolean ifNotExists = false;
1760 Map<String,String> properties = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001761 CreateFeedPolicyStatement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001762}
1763{
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001764 policyName = Identifier() ifNotExists = IfNotExists()
1765 <FROM>
1766 (<POLICY> basePolicyName = Identifier() properties = Configuration() (<DEFINITION> definition = ConstantString())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07001767 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001768 stmt = new CreateFeedPolicyStatement(policyName, basePolicyName, properties, definition, ifNotExists);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001769 }
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001770 | <PATH> sourcePolicyFile = ConstantString() (<DEFINITION> definition = ConstantString())?
1771 {
1772 stmt = new CreateFeedPolicyStatement(policyName, sourcePolicyFile, definition, ifNotExists);
1773 }
1774 )
1775 {
1776 return addSourceLocation(stmt, startStmtToken);
1777 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001778}
1779
Rui Guoe6986dd2020-11-25 19:50:06 -08001780Statement CreateFullTextStatement(Token startStmtToken) throws ParseException:
1781{
1782 Statement stmt = null;
1783}
1784{
1785 (
1786 <FULLTEXT>
1787 (
1788 <FILTER> stmt = CreateFullTextFilterSpec(startStmtToken)
1789 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = CreateFullTextConfigSpec(startStmtToken))
1790 )
1791 )
1792 {
1793 return stmt;
1794 }
1795}
1796
1797CreateFullTextFilterStatement CreateFullTextFilterSpec(Token startStmtToken) throws ParseException:
1798{
1799 CreateFullTextFilterStatement stmt = null;
1800 Pair<DataverseName,Identifier> nameComponents = null;
1801 boolean ifNotExists = false;
1802 RecordConstructor expr = null;
1803}
1804{
1805 (
1806 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1807 <AS>
1808 expr = RecordConstructor()
1809 )
1810 {
1811 try {
1812 stmt = new CreateFullTextFilterStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
1813 return addSourceLocation(stmt, startStmtToken);
1814 } catch (CompilationException e) {
1815 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1816 }
1817 }
1818}
1819
1820CreateFullTextConfigStatement CreateFullTextConfigSpec(Token startStmtToken) throws ParseException:
1821{
1822 CreateFullTextConfigStatement stmt = null;
1823 Pair<DataverseName,Identifier> nameComponents = null;
1824 boolean ifNotExists = false;
1825 RecordConstructor expr = null;
1826}
1827{
1828 (
1829 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1830 <AS>
1831 expr = RecordConstructor()
1832 )
1833 {
1834 try {
1835 stmt = new CreateFullTextConfigStatement(nameComponents.first, nameComponents.second.getValue(), ifNotExists, expr);
1836 return addSourceLocation(stmt, startStmtToken);
1837 } catch (CompilationException e) {
1838 throw new SqlppParseException(getSourceLocation(startStmtToken), e.getMessage());
1839 }
1840 }
1841}
1842
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08001843CreateSynonymStatement CreateSynonymStatement(Token startStmtToken) throws ParseException:
1844{
1845 CreateSynonymStatement stmt = null;
1846}
1847{
1848 <SYNONYM> stmt = SynonymSpecification(startStmtToken)
1849 {
1850 return stmt;
1851 }
1852}
1853
1854CreateSynonymStatement SynonymSpecification(Token startStmtToken) throws ParseException:
1855{
1856 Pair<DataverseName,Identifier> nameComponents = null;
1857 Pair<DataverseName,Identifier> objectNameComponents = null;
1858 boolean ifNotExists = false;
1859}
1860{
1861 nameComponents = QualifiedName() ifNotExists = IfNotExists()
1862 <FOR> objectNameComponents = QualifiedName()
1863 {
1864 CreateSynonymStatement stmt = new CreateSynonymStatement(nameComponents.first, nameComponents.second.getValue(),
1865 objectNameComponents.first, objectNameComponents.second.getValue(), ifNotExists);
1866 return addSourceLocation(stmt, startStmtToken);
1867 }
1868}
1869
Yingyi Bu391f09e2015-10-29 13:49:39 -07001870boolean IfNotExists() throws ParseException:
1871{
1872}
1873{
Yingyi Budaa549c2016-06-28 22:30:52 -07001874 ( LOOKAHEAD(1) <IF> <NOT> <EXISTS>
Yingyi Bu391f09e2015-10-29 13:49:39 -07001875 {
1876 return true;
1877 }
1878 )?
1879 {
1880 return false;
1881 }
1882}
1883
Xikui Wang9d63f622017-05-18 17:50:44 -07001884void ApplyFunction(List<FunctionSignature> funcSigs) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001885{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001886 FunctionName functionName = null;
Xikui Wang261dc6d2017-03-29 21:23:15 -07001887 String fqFunctionName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001888}
1889{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001890 <APPLY> <FUNCTION> functionName = FunctionName()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001891 {
1892 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1893 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
1894 }
Xikui Wang261dc6d2017-03-29 21:23:15 -07001895 (
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001896 <COMMA> functionName = FunctionName()
Xikui Wang261dc6d2017-03-29 21:23:15 -07001897 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08001898 fqFunctionName = functionName.library == null ? functionName.function : functionName.library + "#" + functionName.function;
1899 funcSigs.add(new FunctionSignature(functionName.dataverse, fqFunctionName, 1));
Xikui Wang261dc6d2017-03-29 21:23:15 -07001900 }
1901 )*
Yingyi Bu391f09e2015-10-29 13:49:39 -07001902}
1903
1904String GetPolicy() throws ParseException:
1905{
1906 String policy = null;
1907}
1908{
1909 <USING> <POLICY> policy = Identifier()
1910 {
1911 return policy;
1912 }
1913
1914}
1915
1916FunctionSignature FunctionSignature() throws ParseException:
1917{
1918 FunctionName fctName = null;
Dmitry Lychagin541652082020-11-09 14:04:53 -08001919 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> params = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001920 int arity = 0;
1921}
1922{
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001923 fctName = FunctionName()
1924 (
Dmitry Lychagin541652082020-11-09 14:04:53 -08001925 LOOKAHEAD(2) params = FunctionParameters() { arity = params.first; }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001926 | ( <LEFTPAREN> arity = FunctionArity() <RIGHTPAREN> )
1927 | ( <ATT> arity = FunctionArity() ) // back-compat
1928 )
1929 {
1930 return new FunctionSignature(fctName.dataverse, fctName.function, arity);
1931 }
1932}
Yingyi Bu391f09e2015-10-29 13:49:39 -07001933
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001934int FunctionArity() throws ParseException:
1935{
1936 int arity;
1937}
1938{
1939 <INTEGER_LITERAL>
1940 {
1941 try {
1942 arity = Integer.parseInt(token.image);
1943 } catch (NumberFormatException e) {
1944 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001945 }
Dmitry Lychagindcb4ad22020-08-26 15:49:09 -07001946 if (arity < 0 && arity != FunctionIdentifier.VARARGS) {
1947 throw new SqlppParseException(getSourceLocation(token), "Invalid arity: " + arity);
1948 }
1949 return arity;
1950 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001951}
1952
Yingyi Buc9bfe252016-03-01 00:02:40 -08001953Pair<List<Integer>, List<List<String>>> PrimaryKey() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07001954{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001955 Pair<List<Integer>, List<List<String>>> primaryKeyFields = null;
1956}
1957{
1958 <PRIMARY> <KEY> primaryKeyFields = PrimaryKeyFields()
1959 {
1960 return primaryKeyFields;
1961 }
1962}
1963
1964Pair<List<Integer>, List<List<String>>> PrimaryKeyFields() throws ParseException:
1965{
Yingyi Buc9bfe252016-03-01 00:02:40 -08001966 Pair<Integer, List<String>> tmp = null;
1967 List<Integer> keyFieldSourceIndicators = new ArrayList<Integer>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07001968 List<List<String>> primaryKeyFields = new ArrayList<List<String>>();
1969}
1970{
Dmitry Lychagin81578f92021-08-24 11:57:07 -07001971 tmp = NestedField()
Yingyi Bu391f09e2015-10-29 13:49:39 -07001972 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001973 keyFieldSourceIndicators.add(tmp.first);
1974 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001975 }
1976 ( <COMMA> tmp = NestedField()
1977 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001978 keyFieldSourceIndicators.add(tmp.first);
1979 primaryKeyFields.add(tmp.second);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001980 }
1981 )*
1982 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08001983 return new Pair<List<Integer>, List<List<String>>> (keyFieldSourceIndicators, primaryKeyFields);
Yingyi Bu391f09e2015-10-29 13:49:39 -07001984 }
1985}
1986
1987Statement DropStatement() throws ParseException:
1988{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001989 Token startToken = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001990 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07001991}
1992{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07001993 <DROP> { startToken = token; }
Yingyi Bu391f09e2015-10-29 13:49:39 -07001994 (
Dmitry Lychagin314e2792019-11-26 14:35:42 -08001995 stmt = DropDatasetStatement(startToken)
1996 | stmt = DropIndexStatement(startToken)
1997 | stmt = DropNodeGroupStatement(startToken)
1998 | stmt = DropTypeStatement(startToken)
1999 | stmt = DropDataverseStatement(startToken)
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002000 | stmt = DropAdapterStatement(startToken)
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002001 | stmt = DropFunctionStatement(startToken)
2002 | stmt = DropFeedStatement(startToken)
2003 | stmt = DropFeedPolicyStatement(startToken)
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002004 | stmt = DropSynonymStatement(startToken)
Rui Guoe6986dd2020-11-25 19:50:06 -08002005 | stmt = DropFullTextStatement(startToken)
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002006 | stmt = DropViewStatement(startToken)
Yingyi Bu391f09e2015-10-29 13:49:39 -07002007 )
2008 {
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002009 return stmt;
2010 }
2011}
2012
2013DropDatasetStatement DropDatasetStatement(Token startStmtToken) throws ParseException:
2014{
2015 DropDatasetStatement stmt = null;
2016}
2017{
2018 Dataset() stmt = DropDatasetSpecification(startStmtToken)
2019 {
2020 return stmt;
2021 }
2022}
2023
2024DropDatasetStatement DropDatasetSpecification(Token startStmtToken) throws ParseException:
2025{
2026 Pair<DataverseName,Identifier> pairId = null;
2027 boolean ifExists = false;
2028}
2029{
2030 pairId = QualifiedName() ifExists = IfExists()
2031 {
2032 DropDatasetStatement stmt = new DropDatasetStatement(pairId.first, pairId.second, ifExists);
2033 return addSourceLocation(stmt, startStmtToken);
2034 }
2035}
2036
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07002037ViewDropStatement DropViewStatement(Token startStmtToken) throws ParseException:
2038{
2039 ViewDropStatement stmt = null;
2040}
2041{
2042 <VIEW> stmt = DropViewSpecification(startStmtToken)
2043 {
2044 return stmt;
2045 }
2046}
2047
2048ViewDropStatement DropViewSpecification(Token startStmtToken) throws ParseException:
2049{
2050 Pair<DataverseName,Identifier> pairId = null;
2051 boolean ifExists = false;
2052}
2053{
2054 pairId = QualifiedName() ifExists = IfExists()
2055 {
2056 ViewDropStatement stmt = new ViewDropStatement(pairId.first, pairId.second, ifExists);
2057 return addSourceLocation(stmt, startStmtToken);
2058 }
2059}
2060
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002061IndexDropStatement DropIndexStatement(Token startStmtToken) throws ParseException:
2062{
2063 IndexDropStatement stmt = null;
2064}
2065{
2066 <INDEX> stmt = DropIndexSpecification(startStmtToken)
2067 {
2068 return stmt;
2069 }
2070}
2071
2072IndexDropStatement DropIndexSpecification(Token startStmtToken) throws ParseException:
2073{
2074 Triple<DataverseName,Identifier,Identifier> tripleId = null;
2075 boolean ifExists = false;
2076}
2077{
2078 tripleId = DoubleQualifiedName() ifExists = IfExists()
2079 {
2080 IndexDropStatement stmt = new IndexDropStatement(tripleId.first, tripleId.second, tripleId.third, ifExists);
2081 return addSourceLocation(stmt, startStmtToken);
2082 }
2083}
2084
Rui Guoe6986dd2020-11-25 19:50:06 -08002085Statement DropFullTextStatement(Token startStmtToken) throws ParseException:
2086{
2087 Statement stmt = null;
2088}
2089{
2090 <FULLTEXT>
2091 (
2092 <FILTER> stmt = DropFullTextFilterSpec(startStmtToken)
2093 | (<IDENTIFIER> { expectToken(CONFIG); } stmt = DropFullTextConfigSpec(startStmtToken))
2094 )
2095 {
2096 return stmt;
2097 }
2098}
2099
2100FullTextFilterDropStatement DropFullTextFilterSpec(Token startStmtToken) throws ParseException:
2101{
2102 FullTextFilterDropStatement stmt = null;
2103 Pair<DataverseName,Identifier> nameComponents = null;
2104 boolean ifExists = false;
2105}
2106{
2107 nameComponents = QualifiedName() ifExists = IfExists()
2108 {
2109 stmt = new FullTextFilterDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2110 return addSourceLocation(stmt, startStmtToken);
2111 }
2112}
2113
2114FullTextConfigDropStatement DropFullTextConfigSpec(Token startStmtToken) throws ParseException:
2115{
2116 FullTextConfigDropStatement stmt = null;
2117 Pair<DataverseName,Identifier> nameComponents = null;
2118 boolean ifExists = false;
2119}
2120{
2121 nameComponents = QualifiedName() ifExists = IfExists()
2122 {
2123 stmt = new FullTextConfigDropStatement(nameComponents.first, nameComponents.second.getValue(), ifExists);
2124 return addSourceLocation(stmt, startStmtToken);
2125 }
2126}
2127
2128
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002129NodeGroupDropStatement DropNodeGroupStatement(Token startStmtToken) throws ParseException:
2130{
2131 NodeGroupDropStatement stmt = null;
2132}
2133{
2134 <NODEGROUP> stmt = DropNodeGroupSpecification(startStmtToken)
2135 {
2136 return stmt;
2137 }
2138}
2139
2140NodeGroupDropStatement DropNodeGroupSpecification(Token startStmtToken) throws ParseException:
2141{
2142 String id = null;
2143 boolean ifExists = false;
2144}
2145{
2146 id = Identifier() ifExists = IfExists()
2147 {
2148 NodeGroupDropStatement stmt = new NodeGroupDropStatement(new Identifier(id), ifExists);
2149 return addSourceLocation(stmt, startStmtToken);
2150 }
2151}
2152
2153TypeDropStatement DropTypeStatement(Token startStmtToken) throws ParseException:
2154{
2155 TypeDropStatement stmt = null;
2156}
2157{
2158 <TYPE> stmt = DropTypeSpecification(startStmtToken)
2159 {
2160 return stmt;
2161 }
2162}
2163
2164TypeDropStatement DropTypeSpecification(Token startStmtToken) throws ParseException:
2165{
2166 Pair<DataverseName,Identifier> pairId = null;
2167 boolean ifExists = false;
2168}
2169{
2170 pairId = TypeName() ifExists = IfExists()
2171 {
2172 TypeDropStatement stmt = new TypeDropStatement(pairId.first, pairId.second, ifExists);
2173 return addSourceLocation(stmt, startStmtToken);
2174 }
2175}
2176
2177DataverseDropStatement DropDataverseStatement(Token startStmtToken) throws ParseException:
2178{
2179 DataverseDropStatement stmt = null;
2180}
2181{
2182 <DATAVERSE> stmt = DropDataverseSpecification(startStmtToken)
2183 {
2184 return stmt;
2185 }
2186}
2187
2188DataverseDropStatement DropDataverseSpecification(Token startStmtToken) throws ParseException:
2189{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002190 DataverseName dvName = null;
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002191 boolean ifExists = false;
2192}
2193{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002194 dvName = DataverseName() ifExists = IfExists()
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002195 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002196 DataverseDropStatement stmt = new DataverseDropStatement(dvName, ifExists);
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002197 return addSourceLocation(stmt, startStmtToken);
2198 }
2199}
2200
Dmitry Lychagin20905cd2020-08-06 20:34:55 -07002201AdapterDropStatement DropAdapterStatement(Token startStmtToken) throws ParseException:
2202{
2203 AdapterDropStatement stmt = null;
2204}
2205{
2206 <ADAPTER> stmt = DropAdapterSpecification(startStmtToken)
2207 {
2208 return stmt;
2209 }
2210}
2211
2212AdapterDropStatement DropAdapterSpecification(Token startStmtToken) throws ParseException:
2213{
2214 Pair<DataverseName,Identifier> adapterName = null;
2215 boolean ifExists = false;
2216}
2217{
2218 adapterName = QualifiedName() ifExists = IfExists()
2219 {
2220 AdapterDropStatement stmt = new AdapterDropStatement(adapterName.first, adapterName.second.getValue(), ifExists);
2221 return addSourceLocation(stmt, startStmtToken);
2222 }
2223}
2224
Dmitry Lychagin314e2792019-11-26 14:35:42 -08002225FunctionDropStatement DropFunctionStatement(Token startStmtToken) throws ParseException:
2226{
2227 FunctionDropStatement stmt = null;
2228}
2229{
2230 <FUNCTION> stmt = DropFunctionSpecification(startStmtToken)
2231 {
2232 return stmt;
2233 }
2234}
2235
2236FunctionDropStatement DropFunctionSpecification(Token startStmtToken) throws ParseException:
2237{
2238 FunctionSignature funcSig = null;
2239 boolean ifExists = false;
2240}
2241{
2242 funcSig = FunctionSignature() ifExists = IfExists()
2243 {
2244 FunctionDropStatement stmt = new FunctionDropStatement(funcSig, ifExists);
2245 return addSourceLocation(stmt, startStmtToken);
2246 }
2247}
2248
2249FeedDropStatement DropFeedStatement(Token startStmtToken) throws ParseException:
2250{
2251 FeedDropStatement stmt = null;
2252}
2253{
2254 <FEED> stmt = DropFeedSpecification(startStmtToken)
2255 {
2256 return stmt;
2257 }
2258}
2259
2260FeedDropStatement DropFeedSpecification(Token startStmtToken) throws ParseException:
2261{
2262 Pair<DataverseName,Identifier> pairId = null;
2263 boolean ifExists = false;
2264}
2265{
2266 pairId = QualifiedName() ifExists = IfExists()
2267 {
2268 FeedDropStatement stmt = new FeedDropStatement(pairId.first, pairId.second, ifExists);
2269 return addSourceLocation(stmt, startStmtToken);
2270 }
2271}
2272
2273FeedPolicyDropStatement DropFeedPolicyStatement(Token startStmtToken) throws ParseException:
2274{
2275 FeedPolicyDropStatement stmt = null;
2276}
2277{
2278 <INGESTION> <POLICY> stmt = DropFeedPolicySpecification(startStmtToken)
2279 {
2280 return stmt;
2281 }
2282}
2283
2284FeedPolicyDropStatement DropFeedPolicySpecification(Token startStmtToken) throws ParseException:
2285{
2286 Pair<DataverseName,Identifier> pairId = null;
2287 boolean ifExists = false;
2288}
2289{
2290 pairId = QualifiedName() ifExists = IfExists()
2291 {
2292 FeedPolicyDropStatement stmt = new FeedPolicyDropStatement(pairId.first, pairId.second, ifExists);
2293 return addSourceLocation(stmt, startStmtToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002294 }
2295}
2296
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002297SynonymDropStatement DropSynonymStatement(Token startStmtToken) throws ParseException:
2298{
2299 SynonymDropStatement stmt = null;
2300}
2301{
2302 <SYNONYM> stmt = DropSynonymSpecification(startStmtToken)
2303 {
2304 return stmt;
2305 }
2306}
2307
2308SynonymDropStatement DropSynonymSpecification(Token startStmtToken) throws ParseException:
2309{
2310 Pair<DataverseName,Identifier> pairId = null;
2311 boolean ifExists = false;
2312}
2313{
2314 pairId = QualifiedName() ifExists = IfExists()
2315 {
2316 SynonymDropStatement stmt = new SynonymDropStatement(pairId.first, pairId.second.getValue(), ifExists);
2317 return addSourceLocation(stmt, startStmtToken);
2318 }
2319}
2320
Yingyi Bu391f09e2015-10-29 13:49:39 -07002321boolean IfExists() throws ParseException :
2322{
2323}
2324{
2325 ( LOOKAHEAD(1) <IF> <EXISTS>
2326 {
2327 return true;
2328 }
2329 )?
2330 {
2331 return false;
2332 }
2333}
2334
2335InsertStatement InsertStatement() throws ParseException:
2336{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002337 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002338 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002339 VariableExpr var = null;
2340 Query query = null;
2341 Expression returnExpression = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002342}
2343{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002344 <INSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002345 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002346 ( <RETURNING> returnExpression = Expression())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002347 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002348 if (var == null && returnExpression != null) {
2349 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2350 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002351 }
Yingyi Bucaea8f02015-11-16 15:12:15 -08002352 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002353 InsertStatement stmt = new InsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2354 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002355 return addSourceLocation(stmt, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002356 }
2357}
2358
2359UpsertStatement UpsertStatement() throws ParseException:
2360{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002361 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002362 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bucb5bf332017-01-02 22:19:50 -08002363 VariableExpr var = null;
2364 Query query = null;
2365 Expression returnExpression = null;
2366}
2367{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002368 <UPSERT> { startToken = token; } <INTO> nameComponents = QualifiedName() (<AS> var = Variable())?
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07002369 query = Query()
Yingyi Bucb5bf332017-01-02 22:19:50 -08002370 ( <RETURNING> returnExpression = Expression())?
2371 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002372 if (var == null && returnExpression != null) {
2373 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2374 addSourceLocation(var, startToken);
Yingyi Bucb5bf332017-01-02 22:19:50 -08002375 }
2376 query.setTopLevel(true);
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002377 UpsertStatement stmt = new UpsertStatement(nameComponents.first, nameComponents.second.getValue(), query,
2378 getVarCounter(), var, returnExpression);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002379 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002380 }
2381}
2382
2383DeleteStatement DeleteStatement() throws ParseException:
2384{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002385 Token startToken = null;
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002386 VariableExpr var = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002387 Expression condition = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002388 Pair<DataverseName, Identifier> nameComponents;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002389}
2390{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002391 <DELETE> { startToken = token; }
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002392 <FROM> nameComponents = QualifiedName() ((<AS>)? var = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002393 (<WHERE> condition = Expression())?
Yingyi Bu20e085b2016-07-06 12:57:27 -07002394 {
Dmitry Lychagin393215e2019-04-11 10:26:56 -07002395 if (var == null) {
2396 var = new VariableExpr(SqlppVariableUtil.toInternalVariableIdentifier(nameComponents.second.getValue()));
2397 addSourceLocation(var, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002398 }
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002399 DeleteStatement stmt = new DeleteStatement(var, nameComponents.first, nameComponents.second.getValue(),
Abdullah Alamoudi6eb01752017-04-01 21:51:19 -07002400 condition, getVarCounter());
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002401 return addSourceLocation(stmt, startToken);
Yingyi Bu20e085b2016-07-06 12:57:27 -07002402 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002403}
2404
2405UpdateStatement UpdateStatement() throws ParseException:
2406{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002407 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002408 VariableExpr vars;
2409 Expression target;
2410 Expression condition;
2411 UpdateClause uc;
2412 List<UpdateClause> ucs = new ArrayList<UpdateClause>();
2413}
2414{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002415 <UPDATE> { startToken = token; } vars = Variable() <IN> target = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002416 <WHERE> condition = Expression()
2417 <LEFTPAREN> (uc = UpdateClause()
2418 {
2419 ucs.add(uc);
2420 }
2421 (<COMMA> uc = UpdateClause()
2422 {
2423 ucs.add(uc);
2424 }
2425 )*) <RIGHTPAREN>
2426 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002427 UpdateStatement stmt = new UpdateStatement(vars, target, condition, ucs);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002428 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002429 }
2430}
2431
2432UpdateClause UpdateClause() throws ParseException:
2433{
2434 Expression target = null;
2435 Expression value = null ;
2436 InsertStatement is = null;
2437 DeleteStatement ds = null;
2438 UpdateStatement us = null;
2439 Expression condition = null;
2440 UpdateClause ifbranch = null;
2441 UpdateClause elsebranch = null;
2442}
2443{
2444 (<SET> target = Expression() <EQ> value = Expression()
2445 | is = InsertStatement()
2446 | ds = DeleteStatement()
2447 | us = UpdateStatement()
2448 | <IF> <LEFTPAREN> condition = Expression() <RIGHTPAREN>
2449 <THEN> ifbranch = UpdateClause()
2450 [LOOKAHEAD(1) <ELSE> elsebranch = UpdateClause()]
2451 {
2452 return new UpdateClause(target, value, is, ds, us, condition, ifbranch, elsebranch);
2453 }
2454 )
2455}
2456
2457Statement SetStatement() throws ParseException:
2458{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002459 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002460 String pn = null;
2461 String pv = null;
2462}
2463{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002464 <SET> { startToken = token; } pn = Identifier() pv = ConstantString()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002465 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002466 SetStatement stmt = new SetStatement(pn, pv);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002467 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002468 }
2469}
2470
2471Statement WriteStatement() throws ParseException:
2472{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002473 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002474 String nodeName = null;
2475 String fileName = null;
2476 Query query;
2477 String writerClass = null;
2478 Pair<Identifier,Identifier> nameComponents = null;
2479}
2480{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002481 <WRITE> { startToken = token; } <OUTPUT> <TO> nodeName = Identifier() <COLON> fileName = ConstantString()
Yingyi Bu6d57e492016-06-06 21:24:42 -07002482 ( <USING> writerClass = ConstantString() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002483 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002484 WriteStatement stmt = new WriteStatement(new Identifier(nodeName), fileName, writerClass);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002485 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002486 }
2487}
2488
2489LoadStatement LoadStatement() throws ParseException:
2490{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002491 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002492 DataverseName dataverseName = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002493 Identifier datasetName = null;
2494 boolean alreadySorted = false;
2495 String adapterName;
2496 Map<String,String> properties;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002497 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002498}
2499{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002500 <LOAD> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002501 {
2502 dataverseName = nameComponents.first;
2503 datasetName = nameComponents.second;
2504 }
2505 <USING> adapterName = AdapterName() properties = Configuration()
2506 (<PRESORTED>
2507 {
2508 alreadySorted = true;
2509 }
2510 )?
2511 {
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08002512 LoadStatement stmt = new LoadStatement(dataverseName, datasetName.getValue(), adapterName, properties,
2513 alreadySorted);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002514 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002515 }
2516}
2517
2518
2519String AdapterName() throws ParseException :
2520{
2521 String adapterName = null;
2522}
2523{
2524 adapterName = Identifier()
2525 {
2526 return adapterName;
2527 }
2528}
2529
2530Statement CompactStatement() throws ParseException:
2531{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002532 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002533 Pair<DataverseName,Identifier> nameComponents = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002534}
2535{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002536 <COMPACT> { startToken = token; } Dataset() nameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002537 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002538 CompactStatement stmt = new CompactStatement(nameComponents.first, nameComponents.second);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002539 return addSourceLocation(stmt, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002540 }
2541}
2542
Yingyi Buab817482016-08-19 21:29:31 -07002543Statement ConnectionStatement() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002544{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002545 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002546 Statement stmt = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002547}
2548{
2549 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002550 <CONNECT> { startToken = token; } stmt = ConnectStatement(startToken)
2551 | <DISCONNECT> { startToken = token; } stmt = DisconnectStatement(startToken)
2552 | <START> { startToken = token; } stmt = StartStatement(startToken)
2553 | <STOP> { startToken = token; } stmt = StopStatement(startToken)
Yingyi Buab817482016-08-19 21:29:31 -07002554 )
2555 {
2556 return stmt;
2557 }
2558}
2559
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002560Statement StartStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002561{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002562 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002563 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002564}
2565{
2566 <FEED> feedNameComponents = QualifiedName()
2567 {
2568 stmt = new StartFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002569 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002570 }
2571}
2572
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002573AbstractStatement StopStatement(Token startStmtToken) throws ParseException:
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002574{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002575 Pair<DataverseName,Identifier> feedNameComponents = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002576 AbstractStatement stmt = null;
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002577}
2578{
2579 <FEED> feedNameComponents = QualifiedName()
2580 {
2581 stmt = new StopFeedStatement (feedNameComponents);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002582 return addSourceLocation(stmt, startStmtToken);
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002583 }
2584}
2585
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002586AbstractStatement DisconnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002587{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002588 Pair<DataverseName,Identifier> feedNameComponents = null;
2589 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002590
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002591 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002592}
2593{
2594 (
2595 <FEED> feedNameComponents = QualifiedName() <FROM> Dataset() datasetNameComponents = QualifiedName()
Yingyi Bu391f09e2015-10-29 13:49:39 -07002596 {
2597 stmt = new DisconnectFeedStatement(feedNameComponents, datasetNameComponents);
2598 }
2599 )
Yingyi Buab817482016-08-19 21:29:31 -07002600 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002601 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002602 }
2603}
2604
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002605AbstractStatement ConnectStatement(Token startStmtToken) throws ParseException:
Yingyi Buab817482016-08-19 21:29:31 -07002606{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002607 Pair<DataverseName,Identifier> feedNameComponents = null;
2608 Pair<DataverseName,Identifier> datasetNameComponents = null;
Yingyi Buab817482016-08-19 21:29:31 -07002609
2610 Map<String,String> configuration = null;
Xikui Wang9d63f622017-05-18 17:50:44 -07002611 List<FunctionSignature> appliedFunctions = new ArrayList<FunctionSignature>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002612 AbstractStatement stmt = null;
Yingyi Buab817482016-08-19 21:29:31 -07002613 String policy = null;
Xikui Wangf6741682018-02-22 19:17:17 -08002614 String whereClauseBody = null;
2615 WhereClause whereClause = null;
2616 Token beginPos = null;
2617 Token endPos = null;
Yingyi Buab817482016-08-19 21:29:31 -07002618}
2619{
2620 (
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08002621 <FEED> feedNameComponents = QualifiedName() <TO> Dataset() datasetNameComponents = QualifiedName()
Xikui Wangf6741682018-02-22 19:17:17 -08002622 (ApplyFunction(appliedFunctions))?
2623 (policy = GetPolicy())?
2624 (
2625 <WHERE>
2626 {
2627 beginPos = token;
2628 whereClause = new WhereClause();
2629 Expression whereExpr;
2630 }
2631 whereExpr = Expression()
2632 {
2633 whereClause.setWhereExpr(whereExpr);
2634 }
2635 )?
2636 {
2637 if (whereClause != null) {
2638 endPos = token;
2639 whereClauseBody = extractFragment(beginPos.endLine, beginPos.endColumn, endPos.endLine, endPos.endColumn + 1);
2640 }
2641 }
Yingyi Buab817482016-08-19 21:29:31 -07002642 {
Xikui Wang261dc6d2017-03-29 21:23:15 -07002643 stmt = new ConnectFeedStatement(feedNameComponents, datasetNameComponents, appliedFunctions,
Xikui Wangf6741682018-02-22 19:17:17 -08002644 policy, whereClauseBody, getVarCounter());
Yingyi Buab817482016-08-19 21:29:31 -07002645 }
2646 )
2647 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002648 return addSourceLocation(stmt, startStmtToken);
Yingyi Buab817482016-08-19 21:29:31 -07002649 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002650}
2651
2652Map<String,String> Configuration() throws ParseException :
2653{
2654 Map<String,String> configuration = new LinkedHashMap<String,String>();
2655 Pair<String, String> keyValuePair = null;
2656}
2657{
2658 <LEFTPAREN> ( keyValuePair = KeyValuePair()
2659 {
2660 configuration.put(keyValuePair.first, keyValuePair.second);
2661 }
2662 ( <COMMA> keyValuePair = KeyValuePair()
2663 {
2664 configuration.put(keyValuePair.first, keyValuePair.second);
2665 }
2666 )* )? <RIGHTPAREN>
2667 {
2668 return configuration;
2669 }
2670}
2671
2672Pair<String, String> KeyValuePair() throws ParseException:
2673{
2674 String key;
2675 String value;
2676}
2677{
Ali Alsulimane2986012020-05-14 18:14:22 -07002678 <LEFTPAREN> key = ConstantString()
2679 <EQ> ( value = ConstantString() | (<TRUE> | <FALSE>) {value = token.image.toLowerCase();} )
2680 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002681 {
2682 return new Pair<String, String>(key, value);
2683 }
2684}
2685
2686Map<String,String> Properties() throws ParseException:
2687{
2688 Map<String,String> properties = new HashMap<String,String>();
2689 Pair<String, String> property;
2690}
2691{
2692 (LOOKAHEAD(1) <LEFTPAREN> property = Property()
2693 {
2694 properties.put(property.first, property.second);
2695 }
2696 ( <COMMA> property = Property()
2697 {
2698 properties.put(property.first, property.second);
2699 }
2700 )* <RIGHTPAREN> )?
2701 {
2702 return properties;
2703 }
2704}
2705
2706Pair<String, String> Property() throws ParseException:
2707{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002708 String key = null;
2709 String value = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002710}
2711{
Yingyi Bu6d57e492016-06-06 21:24:42 -07002712 (key = Identifier() | key = StringLiteral())
2713 <EQ>
2714 ( value = ConstantString() | <INTEGER_LITERAL>
Yingyi Bu391f09e2015-10-29 13:49:39 -07002715 {
2716 try {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002717 value = String.valueOf(Long.parseLong(token.image));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002718 } catch (NumberFormatException nfe) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002719 throw new SqlppParseException(getSourceLocation(token), "inapproriate value: " + token.image);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002720 }
2721 }
2722 )
2723 {
2724 return new Pair<String, String>(key.toUpperCase(), value);
2725 }
2726}
2727
Glenn58329202021-04-09 12:03:46 -07002728IndexedTypeExpression IndexedTypeExpr(boolean allowQues) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002729{
2730 TypeExpression typeExpr = null;
Glenn58329202021-04-09 12:03:46 -07002731 boolean isUnknownable = !allowQues;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002732}
2733{
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002734 typeExpr = TypeExpr(false)
Glenn58329202021-04-09 12:03:46 -07002735 ( <QUES>
2736 {
2737 if (!allowQues) {
2738 throw new SqlppParseException(getSourceLocation(token), "'?' quantifier is illegal for the type expression.");
2739 }
2740 isUnknownable = true;
2741 }
2742 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002743 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002744 return new IndexedTypeExpression(typeExpr, isUnknownable);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002745 }
2746}
2747
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002748TypeExpression TypeExpr(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002749{
2750 TypeExpression typeExpr = null;
2751}
2752{
2753 (
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002754 typeExpr = TypeReference()
2755 | typeExpr = OrderedListTypeDef(allowRecordTypeDef)
2756 | typeExpr = UnorderedListTypeDef(allowRecordTypeDef)
2757 | typeExpr = RecordTypeDef() {
2758 if (!allowRecordTypeDef) {
2759 throw new SqlppParseException(typeExpr.getSourceLocation(), "Unexpected record type declaration");
2760 }
2761 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002762 )
2763 {
2764 return typeExpr;
2765 }
2766}
2767
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002768RecordTypeDefinition.RecordKind RecordTypeKind() throws ParseException:
2769{
2770 RecordTypeDefinition.RecordKind recordKind = null;
2771}
2772{
2773 (
2774 <CLOSED> { recordKind = RecordTypeDefinition.RecordKind.CLOSED; }
2775 | <OPEN> { recordKind = RecordTypeDefinition.RecordKind.OPEN; }
2776 )
2777 {
2778 return recordKind;
2779 }
2780}
2781
Yingyi Bu391f09e2015-10-29 13:49:39 -07002782RecordTypeDefinition RecordTypeDef() throws ParseException:
2783{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002784 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002785 RecordTypeDefinition recType = new RecordTypeDefinition();
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002786 RecordTypeDefinition.RecordKind recordKind = RecordTypeDefinition.RecordKind.OPEN;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002787}
2788{
Dmitry Lychagin3e2623c2020-04-01 11:20:04 -07002789 ( recordKind = RecordTypeKind() )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002790 <LEFTBRACE>
2791 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002792 startToken = token;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002793 Token hintToken = fetchHint(token, SqlppHint.GEN_FIELDS_HINT);
2794 if (hintToken != null) {
2795 String hintParams = hintToken.hintParams;
2796 String[] splits = hintParams != null ? hintParams.split("\\s+") : null;
2797 if (splits == null || splits.length != 4) {
2798 throw new SqlppParseException(getSourceLocation(hintToken),
2799 "Expecting: /*+ gen-fields <type> <min> <max> <prefix>*/");
Yingyi Bu391f09e2015-10-29 13:49:39 -07002800 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002801 if (!splits[0].equals("int")) {
2802 throw new SqlppParseException(getSourceLocation(hintToken),
2803 "The only supported type for gen-fields is int.");
2804 }
2805 UndeclaredFieldsDataGen ufdg = new UndeclaredFieldsDataGen(UndeclaredFieldsDataGen.Type.INT,
2806 Integer.parseInt(splits[1]), Integer.parseInt(splits[2]), splits[3]);
2807 recType.setUndeclaredFieldsDataGen(ufdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002808 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002809 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002810 (
2811 RecordField(recType)
2812 ( <COMMA> RecordField(recType) )*
2813 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002814 <RIGHTBRACE>
2815 {
Yingyi Bu391f09e2015-10-29 13:49:39 -07002816 recType.setRecordKind(recordKind);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002817 return addSourceLocation(recType, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002818 }
2819}
2820
2821void RecordField(RecordTypeDefinition recType) throws ParseException:
2822{
2823 String fieldName;
2824 TypeExpression type = null;
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002825 boolean nullable = false, missable = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002826}
2827{
2828 fieldName = Identifier()
2829 {
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002830 Token hintToken = fetchHint(token, SqlppHint.VAL_FILE_HINT, SqlppHint.VAL_FILE_SAME_INDEX_HINT,
2831 SqlppHint.LIST_VAL_FILE_HINT, SqlppHint.LIST_HINT, SqlppHint.INTERVAL_HINT, SqlppHint.INSERT_RAND_INT_HINT,
2832 SqlppHint.DATE_BETWEEN_YEARS_HINT, SqlppHint.DATETIME_ADD_RAND_HOURS_HINT, SqlppHint.AUTO_HINT);
2833 IRecordFieldDataGen rfdg = hintToken != null ? parseFieldDataGen(hintToken) : null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002834 }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002835 <COLON> type = TypeExpr(true) ( <QUES> { nullable = true; missable = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002836 {
Dmitry Lychagin3a628022020-05-12 18:09:02 -07002837 recType.addField(fieldName, type, nullable, missable, rfdg);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002838 }
2839}
2840
2841TypeReferenceExpression TypeReference() throws ParseException:
2842{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002843 Pair<DataverseName,Identifier> id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002844}
2845{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002846 id = QualifiedName()
2847 {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07002848 if (id.first == null && id.second.getValue().equalsIgnoreCase(INT_TYPE_NAME)) {
2849 id.second = new Identifier(BuiltinType.AINT64.getTypeName());
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002850 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002851
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002852 TypeReferenceExpression typeRef = new TypeReferenceExpression(id);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002853 return addSourceLocation(typeRef, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002854 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002855}
2856
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002857OrderedListTypeDefinition OrderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002858{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002859 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002860 TypeExpression type = null;
2861}
2862{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002863 <LEFTBRACKET> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002864 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002865 <RIGHTBRACKET>
2866 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002867 OrderedListTypeDefinition typeDef = new OrderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002868 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002869 }
2870}
2871
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002872UnorderedListTypeDefinition UnorderedListTypeDef(boolean allowRecordTypeDef) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002873{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002874 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002875 TypeExpression type = null;
2876}
2877{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002878 <LEFTDBLBRACE> { startToken = token; }
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07002879 ( type = TypeExpr(allowRecordTypeDef) )
Yingyi Bu391f09e2015-10-29 13:49:39 -07002880 <RIGHTDBLBRACE>
2881 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002882 UnorderedListTypeDefinition typeDef = new UnorderedListTypeDefinition(type);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07002883 return addSourceLocation(typeDef, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07002884 }
2885}
2886
2887FunctionName FunctionName() throws ParseException:
2888{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002889 Triple<List<String>, Token, Token> prefix = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002890 String suffix = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002891}
2892{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002893 // Note: there's a copy of this production in PrimaryExpr() (LOOKAHEAD for FunctionCallExpr())
2894 // that copy must be kept in sync with this code
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002895 prefix = MultipartIdentifierWithHints(
2896 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.RANGE_HINT,
2897 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
2898 )
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002899 (<SHARP> suffix = Identifier())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002900 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002901 Token startToken = prefix.second;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002902 FunctionName result = new FunctionName();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002903 result.sourceLoc = getSourceLocation(startToken);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06002904 result.hintToken = prefix.third;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002905 List<String> list = prefix.first;
2906 int ln = list.size();
2907 String last = list.get(ln - 1);
2908 if (suffix == null) {
2909 // prefix = (dv_part1.dv_part2...dv_partN.)?func_name
2910 // no library name
2911 result.function = last;
2912 } else {
2913 // prefix = (dv_part1.dv_part2...dv_partN.)?lib_name
2914 // suffix = func_name
2915 result.library = last;
2916 result.function = suffix;
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07002917 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002918 if (ln > 1) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07002919 result.dataverse = createDataverseName(list, 0, ln - 1, startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002920 } else {
2921 result.dataverse = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002922 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002923
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002924 if (result.function.equalsIgnoreCase(INT_TYPE_NAME)) {
2925 result.function = BuiltinType.AINT64.getTypeName();
Yingyi Bu391f09e2015-10-29 13:49:39 -07002926 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002927 return result;
2928 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07002929}
2930
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002931Pair<DataverseName,Identifier> TypeName() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002932{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002933 Pair<DataverseName,Identifier> name = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002934}
2935{
2936 name = QualifiedName()
2937 {
2938 if (name.first == null) {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08002939 name.first = defaultDataverse;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002940 }
2941 return name;
2942 }
2943}
2944
2945String Identifier() throws ParseException:
2946{
2947 String lit = null;
2948}
2949{
2950 (<IDENTIFIER>
2951 {
2952 return token.image;
2953 }
2954 | lit = QuotedString()
2955 {
2956 return lit;
2957 }
2958 )
2959}
2960
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08002961List<String> ParenthesizedIdentifierList() throws ParseException:
2962{
2963 List<String> list = new ArrayList<String>();
2964 String ident = null;
2965}
2966{
2967 <LEFTPAREN>
2968 ident = Identifier() { list.add(ident); }
2969 ( <COMMA> ident = Identifier() { list.add(ident); } )*
2970 <RIGHTPAREN>
2971 {
2972 return list;
2973 }
2974}
2975
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002976Pair<Integer, Pair<List<String>, IndexedTypeExpression>> OpenField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002977{
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002978 IndexedTypeExpression fieldType = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002979 Pair<Integer, List<String>> fieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002980}
2981{
2982 fieldList = NestedField()
Glenn58329202021-04-09 12:03:46 -07002983 ( <COLON> fieldType = IndexedTypeExpr(true) )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07002984 {
Dmitry Lychagin8ba59442017-06-16 14:19:45 -07002985 return new Pair<Integer, Pair<List<String>, IndexedTypeExpression>>
2986 (fieldList.first, new Pair<List<String>, IndexedTypeExpression>(fieldList.second, fieldType));
Yingyi Bu391f09e2015-10-29 13:49:39 -07002987 }
2988}
2989
Yingyi Buc9bfe252016-03-01 00:02:40 -08002990Pair<Integer, List<String>> NestedField() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07002991{
2992 List<String> exprList = new ArrayList<String>();
2993 String lit = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07002994 Token litToken = null;
Yingyi Buc9bfe252016-03-01 00:02:40 -08002995 int source = 0;
Yingyi Bu391f09e2015-10-29 13:49:39 -07002996}
2997{
2998 lit = Identifier()
2999 {
Yingyi Bub9169b62016-02-26 21:21:49 -08003000 boolean meetParens = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003001 litToken = token;
Yingyi Bub9169b62016-02-26 21:21:49 -08003002 }
3003 (
Yingyi Buc9bfe252016-03-01 00:02:40 -08003004 LOOKAHEAD(1)
Yingyi Bub9169b62016-02-26 21:21:49 -08003005 <LEFTPAREN><RIGHTPAREN>
3006 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003007 if(!lit.equalsIgnoreCase("meta")){
3008 throw new SqlppParseException(getSourceLocation(litToken), "The string before () has to be \"meta\".");
Yingyi Bub9169b62016-02-26 21:21:49 -08003009 }
3010 meetParens = true;
Yingyi Buc9bfe252016-03-01 00:02:40 -08003011 source = 1;
Yingyi Bub9169b62016-02-26 21:21:49 -08003012 }
3013 )?
3014 {
3015 if(!meetParens){
3016 exprList.add(lit);
3017 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003018 }
3019 (<DOT>
3020 lit = Identifier()
3021 {
3022 exprList.add(lit);
3023 }
3024 )*
3025 {
Yingyi Buc9bfe252016-03-01 00:02:40 -08003026 return new Pair<Integer, List<String>>(source, exprList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003027 }
3028}
3029
Yingyi Bu6d57e492016-06-06 21:24:42 -07003030String ConstantString() throws ParseException:
3031{
3032 String value = null;
3033}
3034{
3035 (value = QuotedString() | value = StringLiteral())
3036 {
3037 return value;
3038 }
3039}
3040
Yingyi Bu391f09e2015-10-29 13:49:39 -07003041String QuotedString() throws ParseException:
3042{
3043}
3044{
3045 <QUOTED_STRING>
3046 {
3047 return removeQuotesAndEscapes(token.image);
3048 }
3049}
3050
Yingyi Bu391f09e2015-10-29 13:49:39 -07003051String StringLiteral() throws ParseException:
3052{
3053}
3054{
3055 <STRING_LITERAL>
3056 {
3057 return removeQuotesAndEscapes(token.image);
3058 }
3059}
3060
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003061Triple<List<String>, Token, Token> MultipartIdentifier() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003062{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003063 Triple<List<String>, Token, Token> result = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003064}
3065{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003066 result = MultipartIdentifierWithHints(null)
Yingyi Bu391f09e2015-10-29 13:49:39 -07003067 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003068 return result;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003069 }
3070}
3071
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003072Triple<List<String>, Token, Token> MultipartIdentifierWithHints(SqlppHint... expectedHints)
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003073 throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003074{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003075 List<String> list = new ArrayList<String>();
3076 SourceLocation sourceLoc = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003077 Token startToken, hintToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003078 String item = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003079}
3080{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003081 item = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003082 {
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003083 list.add(item);
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003084 startToken = token;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003085 if (expectedHints != null && expectedHints.length > 0) {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003086 hintToken = fetchHint(token, expectedHints);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003087 }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003088 }
3089 (<DOT> item = Identifier() { list.add(item); } )*
3090 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003091 return new Triple<List<String>, Token, Token>(list, startToken, hintToken);
3092 }
3093}
3094
3095DataverseName DataverseName() throws ParseException:
3096{
3097 Triple<List<String>, Token, Token> ident = null;
3098}
3099{
3100 ident = MultipartIdentifier()
3101 {
3102 List<String> list = ident.first;
3103 Token startToken = ident.second;
3104 return createDataverseName(list, 0, list.size(), startToken);
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003105 }
3106}
3107
3108Pair<DataverseName,Identifier> QualifiedName() throws ParseException:
3109{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003110 Triple<List<String>, Token, Token> ident = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003111}
3112{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003113 ident = MultipartIdentifier()
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003114 {
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003115 List<String> list = ident.first;
3116 Token startToken = ident.second;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003117 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003118 DataverseName id1 = len > 1 ? createDataverseName(list, 0, len - 1, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003119 Identifier id2 = new Identifier(list.get(len - 1));
3120 return new Pair<DataverseName,Identifier>(id1, id2);
3121 }
3122}
3123
3124Triple<DataverseName, Identifier, Identifier> DoubleQualifiedName() throws ParseException:
3125{
3126 List<String> list = new ArrayList<String>();
3127 String item = null;
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003128 Token startToken = null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003129}
3130{
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003131 item = Identifier() { list.add(item); startToken = token; }
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003132 (<DOT> item = Identifier() { list.add(item); } )+
3133 {
3134 int len = list.size();
Dmitry Lychagin13a40e12021-04-08 18:22:57 -07003135 DataverseName id1 = len > 2 ? createDataverseName(list, 0, len - 2, startToken) : null;
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003136 Identifier id2 = new Identifier(list.get(len - 2));
3137 Identifier id3 = new Identifier(list.get(len - 1));
3138 return new Triple<DataverseName,Identifier,Identifier>(id1, id2, id3);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003139 }
3140}
3141
3142FunctionDecl FunctionDeclaration() throws ParseException:
3143{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003144 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003145 String functionName;
Dmitry Lychagin541652082020-11-09 14:04:53 -08003146 Pair<Integer, List<Pair<VarIdentifier,TypeExpression>>> paramsWithArity = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003147 Expression funcBody;
3148 createNewScope();
3149}
3150{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003151 <DECLARE> { startToken = token; } <FUNCTION>
Xikui Wang3de700a2018-03-15 16:32:55 -07003152 functionName = Identifier()
Dmitry Lychagin541652082020-11-09 14:04:53 -08003153 paramsWithArity = FunctionParameters()
Xikui Wang3de700a2018-03-15 16:32:55 -07003154 <LEFTBRACE>
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003155 funcBody = FunctionBody()
Xikui Wang3de700a2018-03-15 16:32:55 -07003156 <RIGHTBRACE>
3157 {
Dmitry Lychagin541652082020-11-09 14:04:53 -08003158 int arity = paramsWithArity.first;
3159 List<Pair<VarIdentifier,TypeExpression>> paramList = paramsWithArity.second;
3160 FunctionSignature signature = new FunctionSignature(defaultDataverse, functionName, arity);
Xikui Wang3de700a2018-03-15 16:32:55 -07003161 getCurrentScope().addFunctionDescriptor(signature, false);
Dmitry Lychaginf7bc6c12020-06-01 18:19:44 -07003162 ensureNoTypeDeclsInFunction(functionName, paramList, null, startToken);
3163 List<VarIdentifier> params = new ArrayList<VarIdentifier>(paramList.size());
3164 for (Pair<VarIdentifier,TypeExpression> p: paramList) {
3165 params.add(p.getFirst());
Ian Maxon38fe9402020-01-29 19:27:40 -08003166 }
Dmitry Lychagin9ba74872021-04-05 14:38:20 -07003167 FunctionDecl stmt = new FunctionDecl(signature, params, funcBody, false);
Xikui Wang3de700a2018-03-15 16:32:55 -07003168 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003169 return addSourceLocation(stmt, startToken);
Xikui Wang3de700a2018-03-15 16:32:55 -07003170 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003171}
3172
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003173Query Query() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003174{
Dmitry Lychaginaebd0312019-06-04 08:55:39 -07003175 Query query = new Query();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003176 Expression expr;
3177}
3178{
3179 (
3180 expr = Expression()
3181 |
3182 expr = SelectExpression(false)
3183 )
3184 {
3185 query.setBody(expr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003186 query.setSourceLocation(expr.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07003187 return query;
3188 }
3189}
3190
3191
Yingyi Bu391f09e2015-10-29 13:49:39 -07003192Expression Expression():
3193{
3194 Expression expr = null;
3195 Expression exprP = null;
3196}
3197{
3198(
3199 LOOKAHEAD(2)
3200 expr = OperatorExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003201 | expr = QuantifiedExpression()
3202)
3203 {
3204 return (exprP==null) ? expr : exprP;
3205 }
3206}
3207
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003208Expression OperatorExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003209{
3210 OperatorExpr op = null;
3211 Expression operand = null;
3212}
3213{
3214 operand = AndExpr()
3215 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003216 <OR>
3217 {
3218 if (op == null) {
3219 op = new OperatorExpr();
3220 op.addOperand(operand);
Xikui Wang3de700a2018-03-15 16:32:55 -07003221 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003222 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003223 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003224 try{
3225 op.addOperator(token.image.toLowerCase());
3226 } catch (Exception e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003227 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003228 }
Xikui Wang3de700a2018-03-15 16:32:55 -07003229 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003230
Xikui Wang3de700a2018-03-15 16:32:55 -07003231 operand = AndExpr()
3232 {
3233 op.addOperand(operand);
3234 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003235
3236 )*
3237
3238 {
3239 return op==null? operand: op;
3240 }
3241}
3242
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003243Expression AndExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003244{
3245 OperatorExpr op = null;
3246 Expression operand = null;
3247}
3248{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003249 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003250 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07003251 <AND>
3252 {
3253 if (op == null) {
3254 op = new OperatorExpr();
3255 op.addOperand(operand);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003256 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003257 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003258 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003259 try{
3260 op.addOperator(token.image.toLowerCase());
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003261 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003262 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003263 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003264 }
3265
Yingyi Bu196db5d2016-07-15 19:07:20 -07003266 operand = NotExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003267 {
3268 op.addOperand(operand);
3269 }
3270
3271 )*
3272
3273 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003274 return op==null ? operand: op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003275 }
3276}
3277
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003278Expression NotExpr() throws ParseException:
Yingyi Bu196db5d2016-07-15 19:07:20 -07003279{
3280 Expression inputExpr;
3281 boolean not = false;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003282 Token startToken = null;
Yingyi Bu196db5d2016-07-15 19:07:20 -07003283}
3284{
Dmitry Lychagin5476f962019-05-31 16:03:11 -07003285 (LOOKAHEAD(2) <NOT> { not = true; startToken = token; } )? inputExpr = RelExpr()
Yingyi Bu196db5d2016-07-15 19:07:20 -07003286 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003287 if(not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003288 FunctionSignature signature = new FunctionSignature(BuiltinFunctions.NOT);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003289 CallExpr callExpr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(inputExpr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003290 return addSourceLocation(callExpr, startToken);
Yingyi Bu196db5d2016-07-15 19:07:20 -07003291 } else {
3292 return inputExpr;
3293 }
3294 }
3295}
Yingyi Bu391f09e2015-10-29 13:49:39 -07003296
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003297Expression RelExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003298{
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003299 boolean not = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003300 OperatorExpr op = null;
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003301 Token opToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003302 Expression operand = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003303 IExpressionAnnotation annotation = null;
3304}
3305{
Yingyi Bu6c638342016-09-02 17:54:34 -07003306 operand = BetweenExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003307
3308 (
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003309 LOOKAHEAD(3)( <LT> | <GT> | <LE> | <GE> | <EQ> | <NE> | <LG> |<SIMILAR> | (<NOT> { not = true; })? <IN> |
3310 <IS> (<NOT> { not = true; })? <DISTINCT> { opToken = token; } <FROM> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003311 {
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003312 if (opToken == null) {
3313 opToken = token;
3314 }
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003315 Token hintToken = fetchHint(token,
3316 SqlppHint.HASH_BROADCAST_JOIN_HINT, SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT,
3317 SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT, SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
3318 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003319 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003320 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003321 }
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003322 String operator = opToken.image.toLowerCase();
Yingyi Bu4a4b8962016-09-16 12:09:11 -07003323 if (operator.equals("<>")){
3324 operator = "!=";
3325 }
3326 if (not) {
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003327 operator = "not_" + operator;
3328 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003329 if (op == null) {
3330 op = new OperatorExpr();
Shiva2ea73232019-10-09 18:04:05 -07003331 op.addOperand(operand);
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003332 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003333 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003334 }
Yingyi Bu196db5d2016-07-15 19:07:20 -07003335 try{
3336 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003337 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003338 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu196db5d2016-07-15 19:07:20 -07003339 }
Yingyi Bua8baf6d2016-07-05 21:40:44 -07003340 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003341
Yingyi Bu6c638342016-09-02 17:54:34 -07003342 operand = BetweenExpr()
Yingyi Buea4ec722016-11-04 01:26:16 -07003343 {
Shiva2ea73232019-10-09 18:04:05 -07003344 op.addOperand(operand);
Yingyi Buea4ec722016-11-04 01:26:16 -07003345 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003346 )?
3347
3348 {
3349 if (annotation != null) {
3350 op.addHint(annotation);
3351 }
3352 return op==null? operand: op;
3353 }
3354}
3355
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003356Expression BetweenExpr() throws ParseException:
Yingyi Bu6c638342016-09-02 17:54:34 -07003357{
3358 boolean not = false;
3359 OperatorExpr op = null;
3360 Expression operand = null;
3361 IExpressionAnnotation annotation = null;
3362}
3363{
3364 operand = IsExpr()
3365 (
3366 LOOKAHEAD(2)
3367 (<NOT> { not = true; })? <BETWEEN>
3368 {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003369 Token hintToken = fetchHint(token,
3370 SqlppHint.INDEXED_NESTED_LOOP_JOIN_HINT, SqlppHint.SKIP_SECONDARY_INDEX_SEARCH_HINT,
3371 SqlppHint.USE_SECONDARY_INDEX_SEARCH_HINT
3372 );
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07003373 if (hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08003374 annotation = parseExpressionAnnotation(hintToken);
Yingyi Bu6c638342016-09-02 17:54:34 -07003375 }
3376 String operator = token.image.toLowerCase();
3377 if(not){
3378 operator = "not_" + operator;
3379 }
3380 if (op == null) {
3381 op = new OperatorExpr();
3382 op.addOperand(operand);
3383 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003384 addSourceLocation(op, token);
Yingyi Bu6c638342016-09-02 17:54:34 -07003385 }
3386 try{
3387 op.addOperator(operator);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003388 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003389 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6c638342016-09-02 17:54:34 -07003390 }
3391 }
3392
3393 operand = IsExpr()
3394 {
3395 op.addOperand(operand);
3396 }
3397
3398 <AND>
3399 operand = IsExpr()
3400 {
Dmitry Lychaginef1719e2017-12-15 08:33:07 -08003401 op.addOperator(OperatorType.AND);
Yingyi Bu6c638342016-09-02 17:54:34 -07003402 op.addOperand(operand);
3403 }
3404 )?
3405
3406 {
3407 if (annotation != null) {
3408 op.addHint(annotation);
3409 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003410 return op==null ? operand: op;
Yingyi Bu6c638342016-09-02 17:54:34 -07003411 }
3412}
3413
Yingyi Budaa549c2016-06-28 22:30:52 -07003414Expression IsExpr() throws ParseException:
3415{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003416 Token notToken = null;
3417 CallExpr expr = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003418 Expression operand = null;
3419 boolean not = false;
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003420 FunctionIdentifier fn = null;
Yingyi Budaa549c2016-06-28 22:30:52 -07003421}
3422{
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003423 operand = LikeExpr()
Dmitry Lychagin4d5476e2021-08-18 19:43:22 -07003424 (
3425 LOOKAHEAD(3)
3426 <IS>
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003427 (<NOT> { not = true; notToken = token; })?
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003428 (
3429 <NULL> { fn = BuiltinFunctions.IS_NULL; } |
3430 <MISSING> { fn = BuiltinFunctions.IS_MISSING; } |
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08003431 <UNKNOWN> { fn = BuiltinFunctions.IS_UNKNOWN; } |
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07003432 (<KNOWN> | <VALUED>) { not = !not; fn = BuiltinFunctions.IS_UNKNOWN; }
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003433 )
Yingyi Budaa549c2016-06-28 22:30:52 -07003434 {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003435 FunctionSignature signature = new FunctionSignature(fn);
Yingyi Budaa549c2016-06-28 22:30:52 -07003436 expr = new CallExpr(signature, new ArrayList<Expression>(Collections.singletonList(operand)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003437 addSourceLocation(expr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003438 if (not) {
Dmitry Lychagin7c53fcf2017-11-07 12:07:41 -08003439 FunctionSignature notSignature = new FunctionSignature(BuiltinFunctions.NOT);
Yingyi Budaa549c2016-06-28 22:30:52 -07003440 expr = new CallExpr(notSignature, new ArrayList<Expression>(Collections.singletonList(expr)));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003441 addSourceLocation(expr, notToken);
Yingyi Budaa549c2016-06-28 22:30:52 -07003442 }
3443 }
3444 )?
3445 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003446 return expr == null ? operand : expr;
Yingyi Budaa549c2016-06-28 22:30:52 -07003447 }
3448}
3449
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003450Expression LikeExpr() throws ParseException:
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003451{
3452 boolean not = false;
3453 OperatorExpr op = null;
3454 Expression operand = null;
3455}
3456{
3457 operand = ConcatExpr()
3458 (
3459 LOOKAHEAD(2)
3460 (<NOT> { not = true; })? <LIKE>
3461 {
3462 op = new OperatorExpr();
3463 op.addOperand(operand);
3464 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003465 addSourceLocation(op, token);
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003466
3467 String operator = token.image.toLowerCase();
3468 if (not) {
3469 operator = "not_" + operator;
3470 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003471 try {
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003472 op.addOperator(operator);
3473 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003474 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Bu6e6a80c2017-01-21 20:18:49 -08003475 }
3476 }
3477
3478 operand = ConcatExpr()
3479 {
3480 op.addOperand(operand);
3481 }
3482 )?
3483
3484 {
3485 return op == null ? operand : op;
3486 }
3487}
3488
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003489Expression ConcatExpr() throws ParseException:
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003490{
3491 OperatorExpr op = null;
3492 Expression operand = null;
3493}
3494{
3495 operand = AddExpr()
3496 (
3497 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003498 <CONCAT>
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003499 {
3500 if (op == null) {
3501 op = new OperatorExpr();
3502 op.addOperand(operand);
3503 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003504 addSourceLocation(op, token);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003505 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003506 op.addOperator(OperatorType.CONCAT);
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003507 }
3508 operand = AddExpr()
3509 {
3510 op.addOperand(operand);
3511 }
3512 )*
3513
3514 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003515 return op == null ? operand : op;
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003516 }
3517}
Yingyi Budaa549c2016-06-28 22:30:52 -07003518
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003519Expression AddExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003520{
3521 OperatorExpr op = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003522 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003523 Expression operand = null;
3524}
3525{
3526 operand = MultExpr()
Yingyi Budaa549c2016-06-28 22:30:52 -07003527 (
Yingyi Bufdc71eb2016-08-24 22:41:57 -07003528 LOOKAHEAD(1)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003529 (<PLUS> { opType = OperatorType.PLUS; } | <MINUS> { opType = OperatorType.MINUS; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003530 {
3531 if (op == null) {
3532 op = new OperatorExpr();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003533 op.addOperand(operand);
3534 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003535 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003536 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003537 op.addOperator(opType);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003538 }
3539
3540 operand = MultExpr()
3541 {
3542 op.addOperand(operand);
3543 }
3544 )*
3545
3546 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003547 return op == null ? operand : op;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003548 }
3549}
3550
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003551Expression MultExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003552{
3553 OperatorExpr op = null;
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003554 OperatorType opType = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003555 Expression operand = null;
3556}
3557{
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003558 operand = ExponentExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003559
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003560 ( (
3561 <MUL> { opType = OperatorType.MUL; } |
3562 <DIVIDE> { opType = OperatorType.DIVIDE; } |
3563 <DIV> { opType = OperatorType.DIV; } |
3564 ( <MOD> | <PERCENT> ) { opType = OperatorType.MOD; }
3565 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003566 {
3567 if (op == null) {
3568 op = new OperatorExpr();
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003569 op.addOperand(operand);
3570 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003571 addSourceLocation(op, token);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003572 }
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07003573 op.addOperator(opType);
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003574 }
3575 operand = ExponentExpr()
3576 {
3577 op.addOperand(operand);
3578 }
3579 )*
3580
3581 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003582 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003583 }
3584}
3585
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003586Expression ExponentExpr() throws ParseException:
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003587{
3588 OperatorExpr op = null;
3589 Expression operand = null;
3590}
3591{
3592 operand = UnaryExpr()
3593 (<CARET>
3594 {
3595 if (op == null) {
3596 op = new OperatorExpr();
3597 op.addOperand(operand);
3598 op.setCurrentop(true);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003599 addSourceLocation(op, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003600 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003601 op.addOperator(OperatorType.CARET);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003602 }
3603 operand = UnaryExpr()
3604 {
3605 op.addOperand(operand);
3606 }
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003607 )?
3608 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003609 return op == null ? operand : op;
Yingyi Bu79ccdac2016-07-26 23:49:24 -07003610 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003611}
3612
3613Expression UnaryExpr() throws ParseException:
3614{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003615 boolean not = false;
3616 UnaryExpr uexpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003617 Expression expr = null;
3618}
Yingyi Budaa549c2016-06-28 22:30:52 -07003619{
Yingyi Bu196db5d2016-07-15 19:07:20 -07003620 ( (<PLUS> | <MINUS> | (<NOT> { not = true; } )? <EXISTS> )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003621 {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003622 String exprType = token.image.toLowerCase();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003623 if (not) {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003624 exprType = "not_" + exprType;
3625 }
3626 uexpr = new UnaryExpr();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003627 addSourceLocation(uexpr, token);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003628 try {
Yingyi Bu196db5d2016-07-15 19:07:20 -07003629 uexpr.setExprType(exprType);
Taewoo Kime65e6ca2017-01-14 17:53:28 -08003630 } catch (CompilationException e){
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003631 throw new SqlppParseException(getSourceLocation(token), e.getMessage());
Yingyi Budaa549c2016-06-28 22:30:52 -07003632 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003633 }
3634 )?
3635
3636 expr = ValueExpr()
3637 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003638 if (uexpr == null) {
3639 return expr;
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003640 } else {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003641 uexpr.setExpr(expr);
3642 return uexpr;
3643 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003644 }
3645}
3646
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003647Expression ValueExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003648{
3649 Expression expr = null;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003650 AbstractAccessor accessor = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003651}
3652{
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003653 expr = PrimaryExpr()
3654 (
3655 accessor = FieldAccessor(accessor != null ? accessor : expr)
3656 |
3657 accessor = IndexAccessor(accessor != null ? accessor : expr)
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003658 )*
3659 {
3660 return accessor == null ? expr : accessor;
3661 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003662}
3663
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003664FieldAccessor FieldAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003665{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003666 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003667 String ident = null;
3668}
3669{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003670 <DOT> { startToken = token; } ident = Identifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003671 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003672 FieldAccessor fa = new FieldAccessor(inputExpr, new Identifier(ident));
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003673 return addSourceLocation(fa, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003674 }
3675}
3676
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003677AbstractAccessor IndexAccessor(Expression inputExpr) throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003678{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003679 Token startToken = null;
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003680 boolean star = false, slice = false;
3681 Expression expr1 = null, expr2 = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003682}
3683{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003684 <LEFTBRACKET> { startToken = token; }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003685 (
3686 <MUL> { star = true; }
3687 |
3688 ( expr1 = Expression() ( <COLON> { slice = true; } ( expr2 = Expression() )? )? )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003689 )
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003690 <RIGHTBRACKET>
3691 {
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003692 if (expr1 != null && expr1.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3693 ensureIntegerLiteral( (LiteralExpr) expr1, "Index");
Hussain Towailebc5b5deb2018-12-15 18:48:48 +03003694 }
Dmitry Lychagin0a722532020-08-27 18:39:25 -07003695 if (expr2 != null && expr2.getKind() == Expression.Kind.LITERAL_EXPRESSION) {
3696 ensureIntegerLiteral( (LiteralExpr) expr2, "Index");
3697 }
3698 AbstractAccessor resultAccessor;
3699 if (slice) {
3700 resultAccessor = new ListSliceExpression(inputExpr, expr1, expr2);
3701 } else if (star) {
3702 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.STAR, null);
3703 } else {
3704 resultAccessor = new IndexAccessor(inputExpr, IndexAccessor.IndexKind.ELEMENT, expr1);
3705 }
3706 return addSourceLocation(resultAccessor, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003707 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003708}
3709
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003710Expression PrimaryExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07003711{
3712 Expression expr = null;
3713}
3714{
Dmitry Lychagin54c06012019-11-13 15:54:20 -08003715 (
3716 LOOKAHEAD(Identifier() (<DOT> Identifier())* (<SHARP> Identifier())? <LEFTPAREN>) expr = FunctionCallExpr()
Dmitry Lychagin5fbd04b2018-04-19 16:54:28 -07003717 | expr = CaseExpr()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003718 | expr = Literal()
3719 | expr = VariableRef()
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003720 | expr = ExternalVariableRef()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003721 | expr = ListConstructor()
3722 | expr = RecordConstructor()
3723 | expr = ParenthesizedExpression()
3724 )
3725 {
3726 return expr;
3727 }
3728}
3729
3730Expression Literal() throws ParseException:
3731{
3732 LiteralExpr lit = new LiteralExpr();
3733 String str = null;
3734}
3735{
3736 ( str = StringLiteral()
3737 {
3738 lit.setValue(new StringLiteral(str));
3739 }
3740 | <INTEGER_LITERAL>
3741 {
Till Westmann68c6a992016-09-30 00:19:12 -07003742 try {
3743 lit.setValue(new LongIntegerLiteral(Long.valueOf(token.image)));
3744 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003745 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003746 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003747 }
3748 | <FLOAT_LITERAL>
3749 {
Till Westmann68c6a992016-09-30 00:19:12 -07003750 try {
3751 lit.setValue(new FloatLiteral(Float.valueOf(token.image)));
3752 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003753 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003754 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003755 }
3756 | <DOUBLE_LITERAL>
3757 {
Till Westmann68c6a992016-09-30 00:19:12 -07003758 try {
3759 lit.setValue(new DoubleLiteral(Double.valueOf(token.image)));
3760 } catch (NumberFormatException e) {
Ali Alsuliman587b7902019-01-21 14:33:50 -08003761 throw new SqlppParseException(getSourceLocation(token), "Could not parse numeric literal \"" + LogRedactionUtil.userData(token.image) +'"');
Till Westmann68c6a992016-09-30 00:19:12 -07003762 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003763 }
Yingyi Bu535d86b2016-05-23 16:44:25 -07003764 | <MISSING>
3765 {
3766 lit.setValue(MissingLiteral.INSTANCE);
3767 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003768 | <NULL>
3769 {
3770 lit.setValue(NullLiteral.INSTANCE);
3771 }
3772 | <TRUE>
3773 {
3774 lit.setValue(TrueLiteral.INSTANCE);
3775 }
3776 | <FALSE>
3777 {
3778 lit.setValue(FalseLiteral.INSTANCE);
3779 }
3780 )
3781 {
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003782 return addSourceLocation(lit, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003783 }
3784}
3785
Yingyi Bu391f09e2015-10-29 13:49:39 -07003786VariableExpr VariableRef() throws ParseException:
3787{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003788 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003789}
3790{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003791 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003792 {
3793 Identifier ident = lookupSymbol(id);
3794 if (isInForbiddenScopes(id)) {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003795 throw new SqlppParseException(getSourceLocation(token),
3796 "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 -07003797 }
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003798 VariableExpr varExp;
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003799 if (ident != null) { // exist such ident
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003800 varExp = new VariableExpr((VarIdentifier)ident);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003801 } else {
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003802 varExp = new VariableExpr(new VarIdentifier(id));
Yingyi Buacc12a92016-03-26 17:25:05 -07003803 varExp.setIsNewVar(false);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003804 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003805 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003806 }
3807}
3808
Yingyi Bu391f09e2015-10-29 13:49:39 -07003809VariableExpr Variable() throws ParseException:
3810{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003811 String id = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003812}
3813{
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003814 id = VariableIdentifier()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003815 {
3816 Identifier ident = lookupSymbol(id);
Dmitry Lychagin75f19872019-10-03 17:56:35 -07003817 VariableExpr varExp = new VariableExpr(new VarIdentifier(id));
3818 if (ident != null) { // exist such ident
Yingyi Bu391f09e2015-10-29 13:49:39 -07003819 varExp.setIsNewVar(false);
3820 }
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003821 return addSourceLocation(varExp, token);
Yingyi Bu391f09e2015-10-29 13:49:39 -07003822 }
3823}
3824
Dmitry Lychagin0d4c1b62020-02-07 17:10:16 -08003825String VariableIdentifier() throws ParseException:
3826{
3827 String id = null;
3828}
3829{
3830 (<IDENTIFIER> { id = token.image; } | id = QuotedString())
3831 {
3832 return SqlppVariableUtil.toInternalVariableName(id); // prefix user-defined variables with "$".
3833 }
3834}
3835
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08003836Pair<VariableExpr, List<Pair<Expression, Identifier>>> VariableWithFieldMap() throws ParseException:
3837{
3838 VariableExpr var = null;
3839 List<Pair<Expression, Identifier>> fieldList = new ArrayList<Pair<Expression, Identifier>>();
3840}
3841{
3842 var = Variable()
3843 ( LOOKAHEAD(1)
3844 {
3845 VariableExpr fieldVarExpr = null;
3846 String fieldIdentifierStr = null;
3847 }
3848 <LEFTPAREN>
3849 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3850 {
3851 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3852 }
3853 (<COMMA>
3854 fieldVarExpr = VariableRef() <AS> fieldIdentifierStr = Identifier()
3855 {
3856 fieldList.add(new Pair<Expression, Identifier>(fieldVarExpr, new Identifier(fieldIdentifierStr)));
3857 }
3858 )*
3859 <RIGHTPAREN>
3860 )?
3861 {
3862 return new Pair<VariableExpr, List<Pair<Expression, Identifier>>>(var, fieldList);
3863 }
3864}
3865
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003866VariableExpr ExternalVariableRef() throws ParseException:
3867{
3868 String name = null;
3869}
3870{
3871 (
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08003872 <DOLLAR_IDENTIFIER> { name = token.image.substring(1); }
3873 | <DOLLAR_INTEGER_LITERAL> { name = token.image.substring(1); }
3874 | <DOLLAR_QUOTED_STRING> { name = removeQuotesAndEscapes(token.image.substring(1)); }
3875 | <QUES> { name = String.valueOf(++externalVarCounter); }
Dmitry Lychagin1bdf8082018-06-13 14:48:23 -07003876 )
3877 {
3878 String idName = SqlppVariableUtil.toExternalVariableName(name);
3879 VarIdentifier id = new VarIdentifier(idName);
3880 VariableExpr varExp = new VariableExpr(id);
3881 return addSourceLocation(varExp, token);
3882 }
3883}
3884
Yingyi Bu391f09e2015-10-29 13:49:39 -07003885Expression ListConstructor() throws ParseException:
3886{
3887 Expression expr = null;
3888}
3889{
3890 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003891 expr = OrderedListConstructor() |
3892 expr = UnorderedListConstructor()
Yingyi Bu391f09e2015-10-29 13:49:39 -07003893 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07003894 {
3895 return expr;
3896 }
3897}
3898
Yingyi Bu391f09e2015-10-29 13:49:39 -07003899ListConstructor OrderedListConstructor() throws ParseException:
3900{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003901 Token startToken = null;
3902 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003903}
3904{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003905 <LEFTBRACKET> { startToken = token; }
3906 exprList = ExpressionList()
3907 <RIGHTBRACKET>
3908 {
3909 ListConstructor expr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003910 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003911 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003912}
3913
3914ListConstructor UnorderedListConstructor() throws ParseException:
3915{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003916 Token startToken = null;
3917 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003918}
3919{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003920 <LEFTDBLBRACE> { startToken = token; }
3921 exprList = ExpressionList()
3922 <RIGHTDBLBRACE>
3923 {
3924 ListConstructor expr = new ListConstructor(ListConstructor.Type.UNORDERED_LIST_CONSTRUCTOR, exprList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003925 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003926 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003927}
3928
3929List<Expression> ExpressionList() throws ParseException:
3930{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003931 Expression expr = null;
3932 List<Expression> exprList = new ArrayList<Expression>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003933}
3934{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003935 (
3936 expr = Expression()
3937 {
3938 exprList.add(expr);
3939 }
3940 ( <COMMA> expr = Expression()
Till Westmann60f89982017-08-11 18:14:20 -07003941 {
3942 exprList.add(expr);
3943 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003944 )*
3945 )?
3946 {
3947 return exprList;
3948 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003949}
3950
Yingyi Bu391f09e2015-10-29 13:49:39 -07003951RecordConstructor RecordConstructor() throws ParseException:
3952{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003953 Token startToken = null;
3954 FieldBinding fb = null;
3955 List<FieldBinding> fbList = new ArrayList<FieldBinding>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07003956}
3957{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003958 <LEFTBRACE> { startToken = token; }
3959 (
3960 fb = FieldBinding() { fbList.add(fb); }
3961 (<COMMA> fb = FieldBinding() { fbList.add(fb); })*
3962 )?
3963 <RIGHTBRACE>
3964 {
3965 RecordConstructor expr = new RecordConstructor(fbList);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07003966 return addSourceLocation(expr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003967 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003968}
3969
3970FieldBinding FieldBinding() throws ParseException:
3971{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003972 Expression left, right = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003973}
3974{
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003975 left = Expression() ( <COLON> right = Expression() )?
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003976 {
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003977 if (right == null) {
3978 String generatedIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(left, false);
3979 if (generatedIdentifier == null) {
3980 throw new SqlppParseException(getSourceLocation(token), "Cannot infer field name");
3981 }
3982 String generatedName = SqlppVariableUtil.toUserDefinedName(generatedIdentifier);
Ali Alsuliman5760e9e2019-08-22 16:37:36 -07003983 LiteralExpr generatedNameExpr = new LiteralExpr(new StringLiteral(generatedName));
3984 generatedNameExpr.setSourceLocation(left.getSourceLocation());
3985 return new FieldBinding(generatedNameExpr, left);
Dmitry Lychagincdcb9232019-06-04 13:27:03 -07003986 } else {
3987 return new FieldBinding(left, right);
3988 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07003989 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07003990}
3991
Yingyi Bu391f09e2015-10-29 13:49:39 -07003992Expression FunctionCallExpr() throws ParseException:
3993{
Yingyi Bu391f09e2015-10-29 13:49:39 -07003994 List<Expression> argList = new ArrayList<Expression>();
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003995 Expression argExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07003996 FunctionName funcName = null;
Yingyi Buf4d09842016-08-26 00:03:52 -07003997 boolean star = false;
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07003998 boolean distinct = false;
Dmitry Lychagin9d469592020-06-01 15:42:51 -07003999 Expression filterExpr = null;
4000 WindowExpression windowExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004001}
4002{
4003 funcName = FunctionName()
Dmitry Lychagin7a4b5682017-09-11 18:53:07 -07004004 <LEFTPAREN> (
4005 ( <DISTINCT> { distinct = true; } )?
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004006 ( argExpr = Expression() | <MUL> { star = true; } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004007 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004008 if (star) {
4009 if (funcName.function.equalsIgnoreCase(BuiltinFunctions.SCALAR_COUNT.getName())) {
4010 argExpr = new LiteralExpr(new LongIntegerLiteral(1L));
4011 } else {
4012 throw new SqlppParseException(getSourceLocation(token),
4013 "The parameter * can only be used in " + BuiltinFunctions.SCALAR_COUNT.getName() + "().");
Yingyi Buf4d09842016-08-26 00:03:52 -07004014 }
Yingyi Buf4d09842016-08-26 00:03:52 -07004015 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004016 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004017 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004018 (<COMMA> argExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004019 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004020 argList.add(argExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004021 }
4022 )*)? <RIGHTPAREN>
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004023
4024 {
4025 String name = funcName.function;
4026 if (distinct) {
4027 name += FunctionMapUtil.DISTINCT_AGGREGATE_SUFFIX;
4028 }
4029 String fqFunctionName = funcName.library == null ? name : funcName.library + "#" + name;
4030 int arity = argList.size();
4031 FunctionSignature signature = lookupFunctionSignature(funcName.dataverse, fqFunctionName, arity);
4032 if (signature == null) {
4033 signature = new FunctionSignature(funcName.dataverse, fqFunctionName, arity);
4034 }
4035 }
4036
4037 ( <FILTER> <LEFTPAREN> <WHERE> filterExpr = Expression() <RIGHTPAREN> )?
4038
4039 ( LOOKAHEAD(5) windowExpr = WindowExpr(signature, argList, filterExpr) )?
4040
4041 {
4042 if (windowExpr != null) {
4043 return windowExpr;
4044 } else {
4045 CallExpr callExpr = new CallExpr(signature, argList, filterExpr);
Caleb Herbel8d9c57a2020-08-14 20:41:13 -06004046 if (funcName.hintToken != null) {
Dmitry Lychagin6fd8ab32020-12-03 09:04:14 -08004047 IExpressionAnnotation annotation = parseExpressionAnnotation(funcName.hintToken);
4048 if (annotation != null) {
4049 callExpr.addHint(annotation);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004050 }
4051 }
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004052 FunctionMapUtil.normalizedListInputFunctions(callExpr);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004053 callExpr.setSourceLocation(funcName.sourceLoc);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004054 return callExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004055 }
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07004056 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004057}
4058
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004059WindowExpression WindowExpr(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr)
4060 throws ParseException:
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004061{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004062 WindowExpression windowExpr = null;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004063 Boolean fromLast = null, ignoreNulls = null;
4064}
4065{
4066 (
4067 // FROM ( FIRST | LAST ) ( ( RESPECT | IGNORE ) NULLS )? OVER
4068 LOOKAHEAD(5, <FROM> <IDENTIFIER> ( <IDENTIFIER> <IDENTIFIER> )? <OVER>)
4069 <FROM> <IDENTIFIER>
4070 {
4071 if (isToken(FIRST)) {
4072 fromLast = false;
4073 } else if (isToken(LAST)) {
4074 fromLast = true;
4075 } else {
4076 throw createUnexpectedTokenError();
4077 }
4078 }
4079 )?
4080 (
4081 // ( RESPECT | IGNORE ) NULLS OVER
4082 LOOKAHEAD(3, <IDENTIFIER> <IDENTIFIER> <OVER>)
4083 <IDENTIFIER>
4084 {
4085 if (isToken(RESPECT)) {
4086 ignoreNulls = false;
4087 } else if (isToken(IGNORE)) {
4088 ignoreNulls = true;
4089 } else {
4090 throw createUnexpectedTokenError();
4091 }
4092 }
4093 <IDENTIFIER>
4094 {
4095 if (!isToken(NULLS)) {
4096 throw createUnexpectedTokenError();
4097 }
4098 }
4099 )?
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004100 <OVER> windowExpr = OverClause(signature, argList, aggFilterExpr, token, fromLast, ignoreNulls)
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004101 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004102 return windowExpr;
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004103 }
4104}
4105
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004106WindowExpression OverClause(FunctionSignature signature, List<Expression> argList, Expression aggFilterExpr,
4107 Token startToken, Boolean fromLast, Boolean ignoreNulls) throws ParseException:
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004108{
4109 Expression partitionExpr = null;
4110 List<Expression> partitionExprs = new ArrayList<Expression>();
4111 OrderbyClause orderByClause = null;
4112 List<Expression> orderbyList = null;
4113 List<OrderbyClause.OrderModifier> orderbyModifierList = null;
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004114 List<OrderbyClause.NullOrderModifier> orderbyNullModifierList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004115 WindowExpression.FrameMode frameMode = null;
4116 Pair<WindowExpression.FrameBoundaryKind, Expression> frameStart = null, frameEnd = null;
4117 WindowExpression.FrameBoundaryKind frameStartKind = null, frameEndKind = null;
4118 Expression frameStartExpr = null, frameEndExpr = null;
4119 WindowExpression.FrameExclusionKind frameExclusionKind = null;
4120 Pair<VariableExpr, List<Pair<Expression, Identifier>>> windowVarWithFieldList = null;
4121 VariableExpr windowVar = null;
4122 List<Pair<Expression, Identifier>> windowFieldList = null;
4123}
4124{
4125 (
4126 windowVarWithFieldList = VariableWithFieldMap() <AS>
4127 {
4128 windowVar = windowVarWithFieldList.first;
4129 windowFieldList = windowVarWithFieldList.second;
4130 }
4131 )?
4132 <LEFTPAREN>
4133 (
4134 <IDENTIFIER> { expectToken(PARTITION); } <BY>
4135 partitionExpr = Expression() { partitionExprs.add(partitionExpr); }
4136 ( <COMMA> partitionExpr = Expression() { partitionExprs.add(partitionExpr); } )*
4137 )?
4138 (
4139 orderByClause = OrderbyClause()
4140 {
4141 orderbyList = orderByClause.getOrderbyList();
4142 orderbyModifierList = orderByClause.getModifierList();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004143 orderbyNullModifierList = orderByClause.getNullModifierList();
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004144 }
4145 (
4146 frameMode = WindowFrameMode()
4147 (
4148 frameStart = WindowFrameBoundary() |
4149 ( <BETWEEN> frameStart = WindowFrameBoundary() <AND> frameEnd = WindowFrameBoundary() )
4150 )
4151 ( frameExclusionKind = WindowFrameExclusion() )?
4152 {
4153 frameStartKind = frameStart.first;
4154 frameStartExpr = frameStart.second;
4155 if (frameEnd == null) {
4156 frameEndKind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4157 } else {
4158 frameEndKind = frameEnd.first;
4159 frameEndExpr = frameEnd.second;
4160 }
4161 if (frameExclusionKind == null) {
4162 frameExclusionKind = WindowExpression.FrameExclusionKind.NO_OTHERS;
4163 }
4164 }
4165 )?
4166 )?
4167 <RIGHTPAREN>
4168 {
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004169 WindowExpression winExpr = new WindowExpression(signature, argList, aggFilterExpr, partitionExprs, orderbyList,
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004170 orderbyModifierList, orderbyNullModifierList, frameMode, frameStartKind, frameStartExpr, frameEndKind,
4171 frameEndExpr, frameExclusionKind, windowVar, windowFieldList, ignoreNulls, fromLast);
Dmitry Lychagin9d469592020-06-01 15:42:51 -07004172 return addSourceLocation(winExpr, startToken);
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004173 }
4174}
4175
4176WindowExpression.FrameMode WindowFrameMode() throws ParseException:
4177{
4178}
4179{
4180 <IDENTIFIER>
4181 {
4182 if (isToken(RANGE)) {
4183 return WindowExpression.FrameMode.RANGE;
4184 } else if (isToken(ROWS)) {
4185 return WindowExpression.FrameMode.ROWS;
4186 } else if (isToken(GROUPS)) {
4187 return WindowExpression.FrameMode.GROUPS;
4188 } else {
4189 throw createUnexpectedTokenError();
4190 }
4191 }
4192}
4193
4194Pair<WindowExpression.FrameBoundaryKind, Expression> WindowFrameBoundary() throws ParseException:
4195{
4196 boolean current = false;
4197 Expression expr = null;
4198}
4199{
4200 (
Dmitry Lychagin5476f962019-05-31 16:03:11 -07004201 LOOKAHEAD({ laIdentifier(CURRENT) || laIdentifier(UNBOUNDED) }) <IDENTIFIER> { current = isToken(CURRENT); }
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004202 | expr = Expression()
4203 )
4204 <IDENTIFIER>
4205 {
4206 WindowExpression.FrameBoundaryKind kind;
4207 if (current && isToken(ROW)) {
4208 kind = WindowExpression.FrameBoundaryKind.CURRENT_ROW;
4209 } else if (!current && isToken(PRECEDING)) {
4210 kind = expr == null
4211 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_PRECEDING
4212 : WindowExpression.FrameBoundaryKind.BOUNDED_PRECEDING;
4213 } else if (!current && isToken(FOLLOWING)) {
4214 kind = expr == null
4215 ? WindowExpression.FrameBoundaryKind.UNBOUNDED_FOLLOWING
4216 : WindowExpression.FrameBoundaryKind.BOUNDED_FOLLOWING;
4217 } else {
4218 throw createUnexpectedTokenError();
4219 }
4220 return new Pair<WindowExpression.FrameBoundaryKind, Expression>(kind, expr);
4221 }
4222}
4223
4224WindowExpression.FrameExclusionKind WindowFrameExclusion() throws ParseException:
4225{
4226 boolean current = false, no = false;
4227}
4228{
4229 <IDENTIFIER>
4230 {
4231 expectToken(EXCLUDE);
4232 }
4233 (
4234 <GROUP>
4235 {
4236 return WindowExpression.FrameExclusionKind.GROUP;
4237 }
4238 |
4239 (
4240 <IDENTIFIER>
4241 {
4242 if (isToken(TIES)) {
4243 return WindowExpression.FrameExclusionKind.TIES;
4244 } else if (isToken(CURRENT)) {
4245 current = true;
4246 } else if (isToken(NO)) {
4247 no = true;
4248 } else {
4249 throw createUnexpectedTokenError();
4250 }
4251 }
4252 <IDENTIFIER>
4253 {
4254 if (current && isToken(ROW)) {
4255 return WindowExpression.FrameExclusionKind.CURRENT_ROW;
4256 } else if (no && isToken(OTHERS)) {
4257 return WindowExpression.FrameExclusionKind.NO_OTHERS;
4258 } else {
4259 throw createUnexpectedTokenError();
4260 }
4261 }
4262 )
4263 )
4264}
4265
Yingyi Bu391f09e2015-10-29 13:49:39 -07004266Expression ParenthesizedExpression() throws ParseException:
4267{
Dmitry Lychagine9630592021-08-30 15:29:30 -07004268 Token startToken = null;
4269 Expression expr = null, expr2 = null;
4270 List<Expression> exprList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004271}
4272{
4273 (
4274 LOOKAHEAD(2)
Dmitry Lychagine9630592021-08-30 15:29:30 -07004275 <LEFTPAREN> { startToken = token; } expr = Expression()
4276 (
4277 <COMMA> expr2 = Expression()
4278 {
4279 if (exprList == null) {
4280 exprList = new ArrayList<Expression>();
4281 exprList.add(expr);
4282 }
4283 exprList.add(expr2);
4284 }
4285 )*
4286 <RIGHTPAREN>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004287 |
4288 expr = Subquery()
4289 )
4290 {
Dmitry Lychagine9630592021-08-30 15:29:30 -07004291 if (exprList != null) {
4292 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, exprList);
4293 return addSourceLocation(listExpr, startToken);
4294 } else {
4295 return expr;
4296 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004297 }
4298}
4299
Yingyi Buc8c067c2016-07-25 23:37:19 -07004300Expression CaseExpr() throws ParseException:
Yingyi Bu391f09e2015-10-29 13:49:39 -07004301{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004302 Token startToken = null;
4303 Expression conditionExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004304 List<Expression> whenExprs = new ArrayList<Expression>();
4305 List<Expression> thenExprs = new ArrayList<Expression>();
4306 Expression elseExpr = null;
Yingyi Buc8c067c2016-07-25 23:37:19 -07004307 Expression whenExpr = null;
4308 Expression thenExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004309}
4310{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004311 <CASE> { startToken = token; }
4312 ( conditionExpr = Expression() )?
Yingyi Buc8c067c2016-07-25 23:37:19 -07004313 (
4314 <WHEN> whenExpr = Expression()
4315 {
4316 whenExprs.add(whenExpr);
4317 }
4318 <THEN> thenExpr = Expression()
4319 {
4320 thenExprs.add(thenExpr);
4321 }
4322 )*
4323 (<ELSE> elseExpr = Expression() )?
4324 <END>
4325 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004326 if (conditionExpr == null) {
4327 LiteralExpr litExpr = new LiteralExpr(TrueLiteral.INSTANCE);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004328 conditionExpr = addSourceLocation(litExpr, startToken);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004329 }
4330 CaseExpression caseExpr = new CaseExpression(conditionExpr, whenExprs, thenExprs, elseExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004331 return addSourceLocation(caseExpr, startToken);
Yingyi Buc8c067c2016-07-25 23:37:19 -07004332 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004333}
4334
Yingyi Buab817482016-08-19 21:29:31 -07004335SelectExpression SelectExpression(boolean subquery) throws ParseException:
4336{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004337 List<LetClause> letClauses = new ArrayList<LetClause>();
4338 SelectSetOperation selectSetOperation;
4339 OrderbyClause orderbyClause = null;
4340 LimitClause limitClause = null;
4341 createNewScope();
Yingyi Buab817482016-08-19 21:29:31 -07004342}
4343{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004344 ( letClauses = LetClause() )?
4345 selectSetOperation = SelectSetOperation()
4346 (orderbyClause = OrderbyClause() {})?
4347 (limitClause = LimitClause() {})?
4348 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004349 SelectExpression selectExpr =
4350 new SelectExpression(letClauses, selectSetOperation, orderbyClause, limitClause, subquery);
4351 selectExpr.setSourceLocation((!letClauses.isEmpty() ? letClauses.get(0) : selectSetOperation).getSourceLocation());
4352 return selectExpr;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004353 }
4354}
4355
Yingyi Buab817482016-08-19 21:29:31 -07004356SelectSetOperation SelectSetOperation() throws ParseException:
4357{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004358 SetOperationInput setOperationInputLeft;
4359 List<SetOperationRight> setOperationRights = new ArrayList<SetOperationRight>();
4360}
4361{
4362 {
4363 SelectBlock selectBlockLeft = null;
4364 SelectExpression subqueryLeft = null;
4365 Expression expr = null;
4366 }
4367 selectBlockLeft = SelectBlock()
4368 {
4369 setOperationInputLeft = new SetOperationInput(selectBlockLeft, subqueryLeft);
4370 }
4371 (
4372 {
4373 SetOpType opType = SetOpType.UNION;
4374 boolean setSemantics = true;
4375 SelectBlock selectBlockRight = null;
4376 SelectExpression subqueryRight = null;
4377 }
4378 (<UNION> {opType = SetOpType.UNION;} |<INTERSECT> {opType = SetOpType.INTERSECT;} |<EXCEPT> {opType = SetOpType.EXCEPT;}) (<ALL> {setSemantics = false;} )?
4379 (selectBlockRight = SelectBlock()| subqueryRight = Subquery())
4380 {
4381 setOperationRights.add(new SetOperationRight(opType, setSemantics, new SetOperationInput(selectBlockRight, subqueryRight)));
4382 }
4383 )*
4384 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004385 SelectSetOperation selectSetOp = new SelectSetOperation(setOperationInputLeft, setOperationRights);
4386 selectSetOp.setSourceLocation(selectBlockLeft.getSourceLocation());
4387 return selectSetOp;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004388 }
4389}
4390
Yingyi Buab817482016-08-19 21:29:31 -07004391SelectExpression Subquery() throws ParseException:
4392{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004393 SelectExpression selectExpr = null;
4394}
4395{
4396 <LEFTPAREN> selectExpr = SelectExpression(true) {} <RIGHTPAREN>
4397 {
4398 return selectExpr;
4399 }
4400}
4401
Yingyi Buab817482016-08-19 21:29:31 -07004402SelectBlock SelectBlock() throws ParseException:
4403{
Yingyi Bu391f09e2015-10-29 13:49:39 -07004404 SelectClause selectClause = null;
4405 FromClause fromClause = null;
4406 List<LetClause> fromLetClauses = null;
4407 WhereClause whereClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004408 List<AbstractClause> fromLetWhereClauses = new ArrayList<AbstractClause>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004409 GroupbyClause groupbyClause = null;
4410 List<LetClause> gbyLetClauses = null;
4411 HavingClause havingClause = null;
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004412 List<AbstractClause> gbyLetHavingClauses = new ArrayList<AbstractClause>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004413 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004414}
4415{
4416 (
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004417 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004418 selectClause = SelectClause() { startSrcLoc = selectClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004419 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004420 (
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004421 fromClause = FromClause()
4422 (
4423 fromLetClauses = LetClause()
4424 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004425 (whereClause = WhereClause())?
4426 (
4427 groupbyClause = GroupbyClause()
4428 (
4429 gbyLetClauses = LetClause()
4430 )?
4431 (havingClause = HavingClause())?
4432 )?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004433 )
4434 |
4435 (
4436 fromLetClauses = LetClause()
4437 {
4438 // LET without FROM -> create dummy FROM clause: FROM {{missing}} AS #0
4439 SourceLocation sourceLoc = getSourceLocation(token);
4440 LiteralExpr missingExpr = new LiteralExpr(MissingLiteral.INSTANCE);
4441 missingExpr.setSourceLocation(sourceLoc);
4442 List<Expression> list = new ArrayList<Expression>(1);
4443 list.add(missingExpr);
4444 ListConstructor listExpr = new ListConstructor(ListConstructor.Type.ORDERED_LIST_CONSTRUCTOR, list);
4445 listExpr.setSourceLocation(sourceLoc);
4446 List<FromTerm> fromTerms = new ArrayList<FromTerm>(1);
4447 VariableExpr fromVar = new VariableExpr(new VarIdentifier("#0"));
4448 fromVar.setSourceLocation(sourceLoc);
4449 fromTerms.add(new FromTerm(listExpr, fromVar, null, new ArrayList<AbstractBinaryCorrelateClause>()));
4450 fromClause = new FromClause(fromTerms);
4451 }
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004452 (whereClause = WhereClause())?
Dmitry Lychagin368092d2019-06-06 14:39:38 -07004453 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004454 )?
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004455 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004456 |
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004457 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004458 fromClause = FromClause() { startSrcLoc = fromClause.getSourceLocation(); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004459 (
Yingyi Bu391f09e2015-10-29 13:49:39 -07004460 fromLetClauses = LetClause()
4461 )?
4462 (whereClause = WhereClause())?
4463 (
4464 groupbyClause = GroupbyClause()
4465 (
4466 gbyLetClauses = LetClause()
4467 )?
4468 (havingClause = HavingClause())?
4469 )?
4470 selectClause = SelectClause()
Dmitry Lychagin214ff552021-07-22 18:55:41 -07004471 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004472 )
4473 {
Dmitry Lychagin276adf92019-01-15 13:16:40 -08004474 if (fromLetClauses != null) {
4475 fromLetWhereClauses.addAll(fromLetClauses);
4476 }
4477 if (whereClause != null) {
4478 fromLetWhereClauses.add(whereClause);
4479 }
4480 if (gbyLetClauses != null) {
4481 gbyLetHavingClauses.addAll(gbyLetClauses);
4482 }
4483 if (havingClause != null) {
4484 gbyLetHavingClauses.add(havingClause);
4485 }
4486 SelectBlock selectBlock = new SelectBlock(selectClause, fromClause, fromLetWhereClauses, groupbyClause,
4487 gbyLetHavingClauses);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004488 selectBlock.setSourceLocation(startSrcLoc);
4489 return selectBlock;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004490 }
4491}
4492
Yingyi Buab817482016-08-19 21:29:31 -07004493SelectClause SelectClause() throws ParseException:
4494{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004495 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004496 SelectRegular selectRegular = null;
4497 SelectElement selectElement = null;
4498 boolean distinct = false;
4499}
4500{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004501 <SELECT> { startToken = token; } (<ALL>|<DISTINCT> { distinct = true; } )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004502 (
4503 selectRegular = SelectRegular()
Michael Blowd6cf6412016-06-30 02:44:35 -04004504 |
Yingyi Bu391f09e2015-10-29 13:49:39 -07004505 selectElement = SelectElement()
Yingyi Bua89fae62016-07-06 07:58:55 -07004506 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004507 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004508 SourceLocation sourceLoc = getSourceLocation(startToken);
4509 if (selectRegular == null && selectElement == null){
Yingyi Bua89fae62016-07-06 07:58:55 -07004510 Projection projection = new Projection(null, null, true, false);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004511 projection.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004512 List<Projection> projections = new ArrayList<Projection>();
4513 projections.add(projection);
4514 selectRegular = new SelectRegular(projections);
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004515 selectRegular.setSourceLocation(sourceLoc);
Yingyi Bua89fae62016-07-06 07:58:55 -07004516 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004517 SelectClause selectClause = new SelectClause(selectElement, selectRegular, distinct);
4518 selectClause.setSourceLocation(sourceLoc);
4519 return selectClause;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004520 }
4521}
4522
Yingyi Buab817482016-08-19 21:29:31 -07004523SelectRegular SelectRegular() throws ParseException:
4524{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004525 SourceLocation startSrcLoc = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004526 List<Projection> projections = new ArrayList<Projection>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004527 Projection projection = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004528}
4529{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004530 projection = Projection()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004531 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004532 projections.add(projection);
4533 startSrcLoc = projection.getSourceLocation();
4534 }
4535 ( LOOKAHEAD(2) <COMMA> projection = Projection()
4536 {
4537 projections.add(projection);
4538 }
4539 )*
4540 {
4541 SelectRegular selectRegular = new SelectRegular(projections);
4542 selectRegular.setSourceLocation(startSrcLoc);
4543 return selectRegular;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004544 }
4545}
4546
Yingyi Buab817482016-08-19 21:29:31 -07004547SelectElement SelectElement() throws ParseException:
4548{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004549 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004550 Expression expr = null;
4551 String name = null;
4552}
4553{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004554 (<RAW>|<ELEMENT>|<VALUE>) { startToken = token; } expr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004555 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004556 SelectElement selectElement = new SelectElement(expr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004557 return addSourceLocation(selectElement, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004558 }
4559}
4560
Yingyi Buab817482016-08-19 21:29:31 -07004561Projection Projection() throws ParseException :
4562{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004563 SourceLocation startSrcLoc = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004564 Expression expr = null;
4565 Identifier identifier = null;
4566 String name = null;
4567 boolean star = false;
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004568 boolean varStar = false;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004569}
4570{
4571 (
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004572 <MUL> { star = true; startSrcLoc = getSourceLocation(token); }
4573 | LOOKAHEAD(3) expr = VariableRef() <DOT> <MUL> { varStar = true; }
Dmitry Lychagin5cdaa5d2018-02-21 11:11:26 -08004574 | expr = Expression() ((<AS>)? name = Identifier())?
4575 {
4576 if (name == null) {
4577 String generatedColumnIdentifier = ExpressionToVariableUtil.getGeneratedIdentifier(expr, false);
4578 if (generatedColumnIdentifier != null) {
4579 name = SqlppVariableUtil.toUserDefinedName(generatedColumnIdentifier);
4580 }
4581 }
4582 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004583 )
4584 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004585 Projection projection = new Projection(expr, name, star, varStar);
4586 projection.setSourceLocation(expr != null ? expr.getSourceLocation() : startSrcLoc);
4587 return projection;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004588 }
4589}
4590
4591FromClause FromClause() throws ParseException :
4592{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004593 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004594 List<FromTerm> fromTerms = new ArrayList<FromTerm>();
4595 extendCurrentScope();
4596}
4597{
4598 {
4599 FromTerm fromTerm = null;
4600 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004601 <FROM> { startToken = token; } fromTerm = FromTerm() { fromTerms.add(fromTerm); }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004602 (LOOKAHEAD(2) <COMMA> fromTerm = FromTerm() { fromTerms.add(fromTerm); } )*
4603 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004604 FromClause fromClause = new FromClause(fromTerms);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004605 return addSourceLocation(fromClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004606 }
4607}
4608
4609FromTerm FromTerm() throws ParseException :
4610{
4611 Expression leftExpr = null;
Michael Blowd6cf6412016-06-30 02:44:35 -04004612 VariableExpr leftVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004613 VariableExpr posVar = null;
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004614 AbstractBinaryCorrelateClause correlateClause = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004615 List<AbstractBinaryCorrelateClause> correlateClauses = new ArrayList<AbstractBinaryCorrelateClause>();
4616}
4617{
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004618 leftExpr = Expression() ((<AS>)? leftVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004619 (
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004620 (
4621 correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER)
4622 | ( <INNER> correlateClause = JoinOrUnnestClause(JoinType.INNER, UnnestType.INNER) )
4623 | ( <LEFT> ( <OUTER> )? correlateClause = JoinOrUnnestClause(JoinType.LEFTOUTER, UnnestType.LEFTOUTER) )
4624 | ( <RIGHT> ( <OUTER> )? correlateClause = JoinClause(JoinType.RIGHTOUTER) )
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004625 | ( <CROSS> correlateClause = CrossJoinClause() )
Yingyi Bu391f09e2015-10-29 13:49:39 -07004626 )
4627 {
4628 correlateClauses.add(correlateClause);
4629 }
4630 )*
4631 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004632 if (leftVar == null) {
Yingyi Bu5b2d4c82016-07-13 17:56:48 -07004633 leftVar = ExpressionToVariableUtil.getGeneratedVariable(leftExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004634 }
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004635 FromTerm fromTerm = new FromTerm(leftExpr, leftVar, posVar, correlateClauses);
4636 fromTerm.setSourceLocation(leftExpr.getSourceLocation());
4637 return fromTerm;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004638 }
4639}
4640
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004641AbstractBinaryCorrelateClause JoinOrUnnestClause(JoinType joinType, UnnestType unnestType) throws ParseException :
4642{
4643 AbstractBinaryCorrelateClause correlateClause = null;
4644}
4645{
4646 ( correlateClause = JoinClause(joinType) | correlateClause = UnnestClause(unnestType) )
4647 {
4648 return correlateClause;
4649 }
4650}
4651
Yingyi Bu391f09e2015-10-29 13:49:39 -07004652JoinClause JoinClause(JoinType joinType) throws ParseException :
4653{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004654 Token startToken = null;
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004655 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004656 Expression conditionExpr = null;
4657}
4658{
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004659 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput() <ON> conditionExpr = Expression()
4660 {
4661 JoinClause joinClause = new JoinClause(joinType, rightInput.first, rightInput.second, rightInput.third,
4662 conditionExpr);
4663 return addSourceLocation(joinClause, startToken);
4664 }
4665}
4666
4667JoinClause CrossJoinClause() throws ParseException :
4668{
4669 Token startToken = null;
4670 Triple<Expression, VariableExpr, VariableExpr> rightInput = null;
4671 Expression conditionExpr = null;
4672}
4673{
4674 <JOIN> { startToken = token; } rightInput = JoinClauseRightInput()
4675 {
4676 JoinClause joinClause = new JoinClause(JoinType.INNER, rightInput.first, rightInput.second, rightInput.third,
4677 new LiteralExpr(TrueLiteral.INSTANCE));
4678 return addSourceLocation(joinClause, startToken);
4679 }
4680}
4681
4682Triple<Expression, VariableExpr, VariableExpr> JoinClauseRightInput() throws ParseException :
4683{
4684 Expression rightExpr = null;
4685 VariableExpr rightVar = null;
4686 VariableExpr posVar = null;
4687}
4688{
4689 rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004690 {
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004691 if (rightVar == null) {
4692 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004693 }
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07004694 return new Triple<Expression, VariableExpr, VariableExpr>(rightExpr, rightVar, posVar);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004695 }
4696}
4697
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004698UnnestClause UnnestClause(UnnestType unnestType) throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004699{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004700 Token startToken = null;
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08004701 Expression rightExpr = null;
4702 VariableExpr rightVar = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004703 VariableExpr posVar = null;
4704}
4705{
Dmitry Lychagin4cf91e42021-03-12 12:53:15 -08004706 (<UNNEST>|<CORRELATE>|<FLATTEN>) { startToken = token; } rightExpr = Expression() ((<AS>)? rightVar = Variable())? (<AT> posVar = Variable())?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004707 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004708 if (rightVar == null) {
4709 rightVar = ExpressionToVariableUtil.getGeneratedVariable(rightExpr, true);
Yingyi Bu9e3f9be2016-07-01 10:07:37 -07004710 }
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07004711 UnnestClause unnestClause = new UnnestClause(unnestType, rightExpr, rightVar, posVar);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004712 return addSourceLocation(unnestClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004713 }
4714}
4715
Yingyi Bu391f09e2015-10-29 13:49:39 -07004716List<LetClause> LetClause() throws ParseException:
4717{
4718 List<LetClause> letList = new ArrayList<LetClause>();
4719 LetClause letClause;
4720}
4721{
4722 (
4723 (<LET>|<LETTING>) letClause = LetElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = LetElement() { letList.add(letClause); })*
4724 |
4725 <WITH> letClause = WithElement() { letList.add(letClause); } (LOOKAHEAD(1) <COMMA> letClause = WithElement() { letList.add(letClause); })*
4726 )
4727 {
4728 return letList;
4729 }
4730}
4731
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004732WhereClause WhereClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004733{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004734 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004735 Expression whereExpr;
4736}
4737{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004738 <WHERE> { startToken = token; } whereExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004739 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004740 WhereClause wc = new WhereClause(whereExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004741 return addSourceLocation(wc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004742 }
4743}
4744
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004745OrderbyClause OrderbyClause() throws ParseException :
Yingyi Bu391f09e2015-10-29 13:49:39 -07004746{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004747 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004748 OrderbyClause oc = new OrderbyClause();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004749 Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> orderbyExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004750 List<Expression> orderbyList = new ArrayList<Expression>();
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004751 List<OrderbyClause.OrderModifier> modifierList = new ArrayList<OrderbyClause.OrderModifier>();
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004752 List<OrderbyClause.NullOrderModifier> nullModifierList = new ArrayList<OrderbyClause.NullOrderModifier>();
Yingyi Bu391f09e2015-10-29 13:49:39 -07004753}
4754{
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004755 <ORDER>
4756 {
4757 startToken = token;
4758 Token hintToken = fetchHint(token, SqlppHint.INMEMORY_HINT, SqlppHint.RANGE_HINT);
4759 if (hintToken != null) {
4760 switch (hintToken.hint) {
4761 case INMEMORY_HINT:
4762 String[] splits = hintToken.hintParams.split("\\s+");
4763 int numFrames = Integer.parseInt(splits[0]);
4764 int numTuples = Integer.parseInt(splits[1]);
4765 oc.setNumFrames(numFrames);
4766 oc.setNumTuples(numTuples);
4767 break;
4768 case RANGE_HINT:
4769 try {
4770 Expression rangeExpr = parseExpression(hintToken.hintParams);
4771 RangeMap rangeMap = RangeMapBuilder.parseHint(rangeExpr);
4772 oc.setRangeMap(rangeMap);
4773 } catch (CompilationException e) {
4774 throw new SqlppParseException(getSourceLocation(hintToken), e.getMessage());
Ali Alsuliman80225e22018-10-15 14:17:07 -07004775 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004776 break;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004777 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004778 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004779 }
4780 <BY> orderbyExpr = OrderByExpression()
4781 {
4782 orderbyList.add(orderbyExpr.first);
4783 modifierList.add(orderbyExpr.second);
4784 nullModifierList.add(orderbyExpr.third);
4785 }
4786 (
4787 LOOKAHEAD(2) <COMMA> orderbyExpr = OrderByExpression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07004788 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004789 orderbyList.add(orderbyExpr.first);
4790 modifierList.add(orderbyExpr.second);
4791 nullModifierList.add(orderbyExpr.third);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004792 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004793 )*
4794 {
4795 oc.setOrderbyList(orderbyList);
4796 oc.setModifierList(modifierList);
4797 oc.setNullModifierList(nullModifierList);
4798 return addSourceLocation(oc, startToken);
4799 }
4800}
Yingyi Bu391f09e2015-10-29 13:49:39 -07004801
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004802Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier> OrderByExpression()
4803 throws ParseException:
4804{
4805 Expression orderbyExpr = null;
4806 OrderbyClause.OrderModifier modif = OrderbyClause.OrderModifier.ASC;
4807 OrderbyClause.NullOrderModifier nullModif = null;
4808}
4809{
4810 orderbyExpr = Expression()
4811 (
4812 <ASC> { modif = OrderbyClause.OrderModifier.ASC; }
4813 |
4814 <DESC> { modif = OrderbyClause.OrderModifier.DESC; }
4815 )?
4816 (
4817 LOOKAHEAD({ laIdentifier(NULLS) }) <IDENTIFIER> { expectToken(NULLS); } <IDENTIFIER>
Yingyi Bu391f09e2015-10-29 13:49:39 -07004818 {
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004819 if (isToken(FIRST)) {
4820 nullModif = OrderbyClause.NullOrderModifier.FIRST;
4821 } else if (isToken(LAST)) {
4822 nullModif = OrderbyClause.NullOrderModifier.LAST;
4823 } else {
4824 throw createUnexpectedTokenError();
4825 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004826 }
Dmitry Lychagine5e3ad92021-07-15 12:44:47 -07004827 )?
4828 {
4829 return new Triple<Expression, OrderbyClause.OrderModifier, OrderbyClause.NullOrderModifier>(orderbyExpr, modif,
4830 nullModif);
4831 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004832}
4833
4834GroupbyClause GroupbyClause()throws ParseException :
4835{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004836 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004837 GroupbyClause gbc = new GroupbyClause();
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004838 List<List<GbyVariableExpressionPair>> gbyList = null;
4839 List<GroupingElement> groupingElementList = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004840 Pair<VariableExpr, List<Pair<Expression, Identifier>>> groupVarWithFieldList = null;
Yingyi Buacc12a92016-03-26 17:25:05 -07004841 VariableExpr groupVar = null;
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004842 List<Pair<Expression, Identifier>> groupFieldList = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004843}
4844{
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004845 {
4846 Scope newScope = extendCurrentScopeNoPush(true);
4847 // extendCurrentScope(true);
4848 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07004849 <GROUP>
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004850 {
4851 startToken = token;
4852 Token hintToken = fetchHint(token, SqlppHint.HASH_GROUP_BY_HINT);
4853 if (hintToken != null) {
4854 gbc.setHashGroupByHint(true);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004855 }
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004856 }
4857 <BY> groupingElementList = GroupingElementList()
4858 (
4859 <GROUP> <AS> groupVarWithFieldList = VariableWithFieldMap()
Dmitry Lychaginf2c18aa2018-12-28 21:20:23 -08004860 {
4861 groupVar = groupVarWithFieldList.first;
4862 groupFieldList = groupVarWithFieldList.second;
4863 }
Yingyi Buacc12a92016-03-26 17:25:05 -07004864 )?
Yingyi Bu391f09e2015-10-29 13:49:39 -07004865 {
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004866 if (groupingSetsParser == null) {
4867 groupingSetsParser = new SqlppGroupingSetsParser();
4868 }
4869 SourceLocation sourceLoc = getSourceLocation(startToken);
4870 try {
4871 gbyList = groupingSetsParser.parse(groupingElementList, sourceLoc);
4872 } catch (CompilationException e) {
4873 throw new SqlppParseException(sourceLoc, e.getMessage());
4874 }
4875 gbc.setGbyPairList(gbyList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004876 gbc.setDecorPairList(new ArrayList<GbyVariableExpressionPair>());
Yingyi Bu8671ddf2016-08-14 23:58:43 -07004877 gbc.setWithVarMap(new HashMap<Expression, VariableExpr>());
Yingyi Buacc12a92016-03-26 17:25:05 -07004878 gbc.setGroupVar(groupVar);
4879 gbc.setGroupFieldList(groupFieldList);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004880 replaceCurrentScope(newScope);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07004881 return addSourceLocation(gbc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07004882 }
4883}
4884
Dmitry Lychaginac98f482020-03-31 12:26:40 -07004885List<GroupingElement> GroupingElementList() throws ParseException:
4886{
4887 List<GroupingElement> groupingElementList = new ArrayList<GroupingElement>();
4888 GroupingElement groupingElement = null;
4889}
4890{
4891 groupingElement = GroupingElement() { groupingElementList.add(groupingElement); }
4892 ( LOOKAHEAD(1) <COMMA> groupingElement = GroupingElement() { groupingElementList.add(groupingElement); } )*
4893 {
4894 return groupingElementList;
4895 }
4896}
4897
4898GroupingElement GroupingElement() throws ParseException:
4899{
4900 GroupingElement groupingElement = null;
4901 List<GroupingSet> groupingSets = null;
4902 List<GroupingElement> groupingElements = null;
4903}
4904{
4905 (
4906 LOOKAHEAD(2)
4907 <LEFTPAREN> <RIGHTPAREN>
4908 {
4909 groupingElement = GroupingSet.EMPTY;
4910 }
4911 |
4912 LOOKAHEAD({ laIdentifier(ROLLUP) && laToken(2, LEFTPAREN) })
4913 <IDENTIFIER> { expectToken(ROLLUP); }
4914 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4915 {
4916 groupingElement = new RollupCube(groupingSets, false);
4917 }
4918 |
4919 LOOKAHEAD({ laIdentifier(CUBE) && laToken(2, LEFTPAREN) })
4920 <IDENTIFIER> { expectToken(CUBE); }
4921 <LEFTPAREN> groupingSets = OrdinaryGroupingSetList() <RIGHTPAREN>
4922 {
4923 groupingElement = new RollupCube(groupingSets, true);
4924 }
4925 |
4926 LOOKAHEAD({ laIdentifier(GROUPING) && laIdentifier(2, SETS) && laToken(3, LEFTPAREN) })
4927 <IDENTIFIER> { expectToken(GROUPING); } <IDENTIFIER> { expectToken(SETS); }
4928 <LEFTPAREN> groupingElements = GroupingElementList() <RIGHTPAREN>
4929 {
4930 groupingElement = new GroupingSets(groupingElements);
4931 }
4932 |
4933 groupingElement = OrdinaryGroupingSet()
4934 )
4935 {
4936 return groupingElement;
4937 }
4938}
4939
4940GroupingSet OrdinaryGroupingSet() throws ParseException:
4941{
4942 GbyVariableExpressionPair gbyExprPair = null;
4943 List<GbyVariableExpressionPair> items = null;
4944}
4945{
4946 (
4947 LOOKAHEAD(1) <LEFTPAREN> items = GbyVariableExpressionPairList() <RIGHTPAREN>
4948 | gbyExprPair = GbyVariableExpressionPair() { items = Collections.singletonList(gbyExprPair); }
4949 )
4950 {
4951 return new GroupingSet(items);
4952 }
4953}
4954
4955List<GroupingSet> OrdinaryGroupingSetList() throws ParseException:
4956{
4957 GroupingSet groupingSet = null;
4958 List<GroupingSet> items = new ArrayList<GroupingSet>();
4959}
4960{
4961 groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); }
4962 ( LOOKAHEAD(1) <COMMA> groupingSet = OrdinaryGroupingSet() { items.add(groupingSet); } )*
4963 {
4964 return items;
4965 }
4966}
4967
4968List<GbyVariableExpressionPair> GbyVariableExpressionPairList() throws ParseException:
4969{
4970 GbyVariableExpressionPair gbyExprPair = null;
4971 List<GbyVariableExpressionPair> items = new ArrayList<GbyVariableExpressionPair>();
4972}
4973{
4974 gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); }
4975 ( LOOKAHEAD(1) <COMMA> gbyExprPair = GbyVariableExpressionPair() { items.add(gbyExprPair); } )*
4976 {
4977 return items;
4978 }
4979}
4980
4981GbyVariableExpressionPair GbyVariableExpressionPair() throws ParseException:
4982{
4983 Expression expr = null;
4984 VariableExpr var = null;
4985}
4986{
4987 expr = Expression() ( (<AS>)? var = Variable() )?
4988 {
4989 return new GbyVariableExpressionPair(var, expr);
4990 }
4991}
4992
Yingyi Bu391f09e2015-10-29 13:49:39 -07004993HavingClause HavingClause() throws ParseException:
4994{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004995 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07004996 Expression filterExpr = null;
4997}
4998{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07004999 <HAVING> { startToken = token; } filterExpr = Expression()
Yingyi Bu391f09e2015-10-29 13:49:39 -07005000 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005001 HavingClause havingClause = new HavingClause(filterExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005002 return addSourceLocation(havingClause, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005003 }
5004}
5005
5006LimitClause LimitClause() throws ParseException:
5007{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005008 Token startToken = null;
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005009 Expression limitExpr = null, offsetExpr = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005010}
5011{
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005012 (
5013 (
5014 <LIMIT> { startToken = token; pushForbiddenScope(getCurrentScope()); } limitExpr = Expression()
5015 ( <OFFSET> offsetExpr = Expression() )?
5016 { popForbiddenScope(); }
5017 )
5018 |
5019 (
5020 <OFFSET> { startToken = token; pushForbiddenScope(getCurrentScope()); } offsetExpr = Expression()
5021 { popForbiddenScope(); }
5022 )
5023 )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005024 {
Dmitry Lychagin761c9a62020-11-19 10:22:11 -08005025 LimitClause lc = new LimitClause(limitExpr, offsetExpr);
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005026 return addSourceLocation(lc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005027 }
5028}
5029
5030QuantifiedExpression QuantifiedExpression()throws ParseException:
5031{
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005032 Token startToken = null;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005033 QuantifiedExpression qc = new QuantifiedExpression();
5034 List<QuantifiedPair> quantifiedList = new ArrayList<QuantifiedPair>();
5035 Expression satisfiesExpr;
5036 VariableExpr var;
5037 Expression inExpr;
5038 QuantifiedPair pair;
5039}
5040{
5041 {
5042 createNewScope();
5043 }
5044
ggalvizo21e52822021-07-27 10:26:31 -10005045 ( LOOKAHEAD(2)
5046 (<ANY>|<SOME>)<AND><EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME_AND_EVERY); }
5047 | (<ANY>|<SOME>) { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.SOME); }
5048 | <EVERY> { startToken = token; qc.setQuantifier(QuantifiedExpression.Quantifier.EVERY); } )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005049 var = Variable() <IN> inExpr = Expression()
5050 {
5051 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005052 quantifiedList.add(pair);
5053 }
5054 (
5055 <COMMA> var = Variable() <IN> inExpr = Expression()
5056 {
5057 pair = new QuantifiedPair(var, inExpr);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005058 quantifiedList.add(pair);
5059 }
5060 )*
Yingyi Bu858efae2016-10-13 17:31:57 -07005061 <SATISFIES> satisfiesExpr = Expression() (<END>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005062 {
5063 qc.setSatisfiesExpr(satisfiesExpr);
5064 qc.setQuantifiedList(quantifiedList);
5065 removeCurrentScope();
Dmitry Lychagin5c26b422018-06-01 10:22:16 -07005066 return addSourceLocation(qc, startToken);
Yingyi Bu391f09e2015-10-29 13:49:39 -07005067 }
5068}
5069
5070LetClause LetElement() throws ParseException:
5071{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005072 VariableExpr varExp;
5073 Expression beExp;
5074 extendCurrentScope();
5075}
5076{
5077 varExp = Variable() <EQ> beExp = Expression()
5078 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005079 LetClause lc = new LetClause(varExp, beExp);
5080 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005081 return lc;
5082 }
5083}
5084
5085LetClause WithElement() throws ParseException:
5086{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005087 VariableExpr varExp;
5088 Expression beExp;
5089 extendCurrentScope();
5090}
5091{
5092 varExp = Variable() <AS> beExp = Expression()
5093 {
Dmitry Lychaginee54cc02018-05-25 19:26:18 -07005094 LetClause lc = new LetClause(varExp, beExp);
5095 lc.setSourceLocation(varExp.getSourceLocation());
Yingyi Bu391f09e2015-10-29 13:49:39 -07005096 return lc;
5097 }
5098}
5099
5100TOKEN_MGR_DECLS:
5101{
5102 public int commentDepth = 0;
Till Westmanne9b2adf2016-10-15 12:39:01 -07005103 public ArrayDeque<Integer> lexerStateStack = new ArrayDeque<Integer>();
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005104 public Map<SourceLocation, String> hintCollector;
Yingyi Bu391f09e2015-10-29 13:49:39 -07005105
5106 public void pushState() {
5107 lexerStateStack.push( curLexState );
5108 }
5109
5110 public void popState(String token) {
5111 if (lexerStateStack.size() > 0) {
5112 SwitchTo( lexerStateStack.pop() );
5113 } else {
5114 int errorLine = input_stream.getEndLine();
5115 int errorColumn = input_stream.getEndColumn();
5116 String msg = "Lexical error at line " + errorLine + ", column " + errorColumn + ". Encountered \"" + token
5117 + "\" but state stack is empty.";
5118 throw new TokenMgrError(msg, -1);
5119 }
5120 }
Dmitry Lychagine77cdac2019-08-05 11:37:47 -07005121
5122 void CommonTokenAction(Token token) {
5123 Token hintToken = token.specialToken;
5124 if (hintToken != null) {
5125 hintToken.sourceLocation = new SourceLocation(hintToken.beginLine, hintToken.beginColumn);
5126 String text = hintToken.image.substring(1).trim();
5127 boolean hintFound = hintToken.parseHint(text);
5128 hintCollector.put(hintToken.sourceLocation, hintFound ? hintToken.hint.getIdentifier() : hintToken.hintParams);
5129 }
5130 }
Yingyi Bu391f09e2015-10-29 13:49:39 -07005131}
5132
5133<DEFAULT,IN_DBL_BRACE>
5134TOKEN [IGNORE_CASE]:
5135{
Ian Maxon38fe9402020-01-29 19:27:40 -08005136 <ADAPTER: "adapter">
5137 | <ALL : "all">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005138 | <AND : "and">
Yingyi Bu8aac7242016-09-13 23:14:09 -07005139 | <ANY : "any">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005140 | <APPLY : "apply">
5141 | <AS : "as">
5142 | <ASC : "asc">
5143 | <AT : "at">
5144 | <AUTOGENERATED : "autogenerated">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005145 | <BETWEEN : "between">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005146 | <BTREE : "btree">
5147 | <BY : "by">
5148 | <CASE : "case">
5149 | <CLOSED : "closed">
5150 | <CREATE : "create">
Dmitry Lychagind3eccd42021-08-19 12:18:17 -07005151 | <CROSS : "cross">
Dmitry Lychagin0ebc4202021-01-25 13:12:41 -08005152 | <COMPACTION : "compaction"> // no longer used
Yingyi Bu391f09e2015-10-29 13:49:39 -07005153 | <COMPACT : "compact">
5154 | <CONNECT : "connect">
5155 | <CORRELATE : "correlate">
Yingyi Bud56ff032016-08-01 10:26:57 -07005156 | <DATASET : "dataset">
Yingyi Bu1c0fff52016-03-25 20:23:30 -07005157 | <COLLECTION : "collection">
Yingyi Bud56ff032016-08-01 10:26:57 -07005158 | <DATAVERSE : "dataverse">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005159 | <DECLARE : "declare">
5160 | <DEFINITION : "definition">
5161 | <DELETE : "delete">
5162 | <DESC : "desc">
5163 | <DISCONNECT : "disconnect">
5164 | <DISTINCT : "distinct">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005165 | <DIV : "div">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005166 | <DROP : "drop">
5167 | <ELEMENT : "element">
Till Westmann516d1a82016-08-02 14:45:53 -07005168 | <EXPLAIN : "explain">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005169 | <ELSE : "else">
5170 | <ENFORCED : "enforced">
Yingyi Buc8c067c2016-07-25 23:37:19 -07005171 | <END : "end">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005172 | <EVERY : "every">
5173 | <EXCEPT : "except">
5174 | <EXISTS : "exists">
5175 | <EXTERNAL : "external">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005176 | <FALSE : "false">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005177 | <FEED : "feed">
5178 | <FILTER : "filter">
5179 | <FLATTEN : "flatten">
5180 | <FOR : "for">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005181 | <FROM : "from">
5182 | <FULL : "full">
Taewoo Kimc49405a2017-01-04 00:30:43 -08005183 | <FULLTEXT : "fulltext">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005184 | <FUNCTION : "function">
5185 | <GROUP : "group">
5186 | <HAVING : "having">
5187 | <HINTS : "hints">
5188 | <IF : "if">
5189 | <INTO : "into">
5190 | <IN : "in">
5191 | <INDEX : "index">
5192 | <INGESTION : "ingestion">
5193 | <INNER : "inner">
5194 | <INSERT : "insert">
5195 | <INTERNAL : "internal">
5196 | <INTERSECT : "intersect">
Yingyi Budaa549c2016-06-28 22:30:52 -07005197 | <IS : "is">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005198 | <JOIN : "join">
5199 | <KEYWORD : "keyword">
5200 | <KEY : "key">
Dmitry Lychagin1aa0dd42018-04-17 14:35:24 -07005201 | <KNOWN : "known">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005202 | <LEFT : "left">
5203 | <LETTING : "letting">
5204 | <LET : "let">
Yingyi Bua8baf6d2016-07-05 21:40:44 -07005205 | <LIKE : "like">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005206 | <LIMIT : "limit">
5207 | <LOAD : "load">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005208 | <MISSING : "missing">
Murtadha Hubail29f63912019-04-21 14:23:57 +03005209 | <MOD : "mod">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005210 | <NODEGROUP : "nodegroup">
5211 | <NGRAM : "ngram">
Yingyi Budaa549c2016-06-28 22:30:52 -07005212 | <NOT : "not">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005213 | <NULL : "null">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005214 | <OFFSET : "offset">
5215 | <ON : "on">
5216 | <OPEN : "open">
5217 | <OR : "or">
5218 | <ORDER : "order">
5219 | <OUTER : "outer">
5220 | <OUTPUT : "output">
Dmitry Lychaginfdedf622018-10-30 18:12:40 -07005221 | <OVER: "over">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005222 | <PATH : "path">
5223 | <POLICY : "policy">
5224 | <PRESORTED : "pre-sorted">
5225 | <PRIMARY : "primary">
5226 | <RAW : "raw">
5227 | <REFRESH : "refresh">
5228 | <RETURN : "return">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005229 | <RETURNING : "returning">
Dmitry Lychagind4ce84c2020-10-13 16:24:37 -07005230 | <RIGHT : "right">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005231 | <RTREE : "rtree">
5232 | <RUN : "run">
5233 | <SATISFIES : "satisfies">
5234 | <SECONDARY : "secondary">
5235 | <SELECT : "select">
5236 | <SET : "set">
5237 | <SOME : "some">
Abdullah Alamoudifff200c2017-02-18 20:32:14 -08005238 | <START : "start">
5239 | <STOP : "stop">
Dmitry Lychagin2a0848e2020-01-08 13:57:22 -08005240 | <SYNONYM : "synonym">
Murtadha Hubail2c04ae02017-11-21 15:58:01 +03005241 | <TEMPORARY : "temporary"> // intentionally not used but reserved for future usage
Yingyi Bu391f09e2015-10-29 13:49:39 -07005242 | <THEN : "then">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005243 | <TO : "to">
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005244 | <TRUE : "true">
5245 | <TYPE : "type">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005246 | <UNION : "union">
Dmitry Lychagin8b6578a2017-11-08 15:04:35 -08005247 | <UNKNOWN : "unknown">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005248 | <UNNEST : "unnest">
Yingyi Budaa549c2016-06-28 22:30:52 -07005249 | <UPDATE : "update">
Yingyi Bucb5bf332017-01-02 22:19:50 -08005250 | <UPSERT : "upsert">
Yingyi Budaa549c2016-06-28 22:30:52 -07005251 | <USE : "use">
5252 | <USING : "using">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005253 | <VALUE : "value">
Dmitry Lychagin7d594a32018-01-15 14:31:03 -08005254 | <VALUED : "valued">
Dmitry Lychagin793c39e2021-04-21 16:52:10 -07005255 | <VIEW : "view">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005256 | <WHEN : "when">
5257 | <WHERE : "where">
5258 | <WITH : "with">
5259 | <WRITE : "write">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005260}
5261
5262<DEFAULT,IN_DBL_BRACE>
5263TOKEN :
5264{
5265 <CARET : "^">
Yingyi Bufdc71eb2016-08-24 22:41:57 -07005266 | <CONCAT : "||">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005267 | <DIVIDE : "/">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005268 | <MINUS : "-">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005269 | <MUL : "*">
5270 | <PLUS : "+">
5271
5272 | <LEFTPAREN : "(">
5273 | <RIGHTPAREN : ")">
5274 | <LEFTBRACKET : "[">
5275 | <RIGHTBRACKET : "]">
5276
5277 | <ATT : "@">
5278 | <COLON : ":">
5279 | <COMMA : ",">
5280 | <DOT : ".">
Dmitry Lychaginbf48c492018-05-01 18:30:27 -07005281 | <PERCENT: "%">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005282 | <QUES : "?">
5283 | <SEMICOLON : ";">
5284 | <SHARP : "#">
5285
5286 | <LT : "<">
5287 | <GT : ">">
5288 | <LE : "<=">
5289 | <GE : ">=">
5290 | <EQ : "=">
5291 | <NE : "!=">
Yingyi Bu4a4b8962016-09-16 12:09:11 -07005292 | <LG : "<>">
Yingyi Bu391f09e2015-10-29 13:49:39 -07005293 | <SIMILAR : "~=">
5294}
5295
5296<DEFAULT,IN_DBL_BRACE>
5297TOKEN :
5298{
5299 <LEFTBRACE : "{"> { pushState(); } : DEFAULT
5300}
5301
5302<DEFAULT>
5303TOKEN :
5304{
5305 <RIGHTBRACE : "}"> { popState("}"); }
5306}
5307
5308<DEFAULT,IN_DBL_BRACE>
5309TOKEN :
5310{
5311 <LEFTDBLBRACE : "{{"> { pushState(); } : IN_DBL_BRACE
5312}
5313
5314<IN_DBL_BRACE>
5315TOKEN :
5316{
5317 <RIGHTDBLBRACE : "}}"> { popState("}}"); }
5318}
5319
5320<DEFAULT,IN_DBL_BRACE>
5321TOKEN :
5322{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005323 <#DIGIT : ["0" - "9"]>
5324}
5325
5326<DEFAULT,IN_DBL_BRACE>
5327TOKEN:
5328{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005329 <INTEGER_LITERAL : <DIGITS> >
5330 | <DOUBLE_LITERAL: <DIGITS> ( "." <DIGITS> ) (("e"|"E") ("+"|"-")? <DIGITS>)?
Dmitry Lychaginee8526b2018-03-30 14:21:01 -07005331 | <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)
5332 | "." <DIGITS> (("e"|"E") ("+"|"-")? <DIGITS>)?
Yingyi Bu391f09e2015-10-29 13:49:39 -07005333 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005334 | <FLOAT_LITERAL: <DIGITS> ( "f" | "F" )
Yingyi Bue4d919e2016-10-30 10:47:03 -07005335 | <DIGITS> ( "." <DIGITS> ( "f" | "F" ) )?
5336 | "." <DIGITS> ( "f" | "F" )
Yingyi Bu391f09e2015-10-29 13:49:39 -07005337 >
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005338 | <#DIGITS : (<DIGIT>)+ >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005339}
5340
5341<DEFAULT,IN_DBL_BRACE>
5342TOKEN :
5343{
5344 <#LETTER : ["A" - "Z", "a" - "z"]>
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005345 | <#IDENTIFIER_START_SPECIALCHAR : ["_"]>
5346 | <#IDENTIFIER_REST_SPECIALCHAR : ["$"]>
5347 | <#IDENTIFIER_START : <LETTER> | <IDENTIFIER_START_SPECIALCHAR> >
5348 | <#IDENTIFIER_REST : <LETTER> | <DIGIT> | <IDENTIFIER_START_SPECIALCHAR> | <IDENTIFIER_REST_SPECIALCHAR> >
5349 | <IDENTIFIER : <IDENTIFIER_START> (<IDENTIFIER_REST>)* >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005350}
5351
5352<DEFAULT,IN_DBL_BRACE>
5353TOKEN :
5354{
5355 // backslash u + 4 hex digits escapes are handled in the underlying JavaCharStream
Yingyi Bu6d57e492016-06-06 21:24:42 -07005356 <QUOTED_STRING : "`" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005357 <EscapeQuot>
5358 | <EscapeBslash>
5359 | <EscapeSlash>
5360 | <EscapeBspace>
5361 | <EscapeFormf>
5362 | <EscapeNl>
5363 | <EscapeCr>
5364 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005365 | ~["`","\\"])* "`">
5366 | <STRING_LITERAL : ("\"" (
Yingyi Bu391f09e2015-10-29 13:49:39 -07005367 <EscapeQuot>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005368 | <EscapeBslash>
5369 | <EscapeSlash>
5370 | <EscapeBspace>
5371 | <EscapeFormf>
5372 | <EscapeNl>
5373 | <EscapeCr>
5374 | <EscapeTab>
Yingyi Bu6d57e492016-06-06 21:24:42 -07005375 | ~["\"","\\"])* "\"")
5376 | ("\'"(
5377 <EscapeApos>
5378 | <EscapeBslash>
5379 | <EscapeSlash>
5380 | <EscapeBspace>
5381 | <EscapeFormf>
5382 | <EscapeNl>
5383 | <EscapeCr>
5384 | <EscapeTab>
5385 | ~["\'","\\"])* "\'")>
Yingyi Bu391f09e2015-10-29 13:49:39 -07005386 | < #EscapeQuot: "\\\"" >
5387 | < #EscapeApos: "\\\'" >
5388 | < #EscapeBslash: "\\\\" >
5389 | < #EscapeSlash: "\\/" >
5390 | < #EscapeBspace: "\\b" >
5391 | < #EscapeFormf: "\\f" >
5392 | < #EscapeNl: "\\n" >
5393 | < #EscapeCr: "\\r" >
5394 | < #EscapeTab: "\\t" >
5395}
5396
5397<DEFAULT,IN_DBL_BRACE>
5398TOKEN :
5399{
Dmitry Lychagin41b511d2020-12-07 14:48:13 -08005400 <DOLLAR_INTEGER_LITERAL : "$" <INTEGER_LITERAL> >
5401 | <DOLLAR_IDENTIFIER : "$" <IDENTIFIER> >
5402 | <DOLLAR_QUOTED_STRING: "$" <QUOTED_STRING> >
Yingyi Bu391f09e2015-10-29 13:49:39 -07005403}
5404
5405<DEFAULT,IN_DBL_BRACE>
5406SKIP:
5407{
5408 " "
5409 | "\t"
5410 | "\r"
5411 | "\n"
5412}
5413
5414<DEFAULT,IN_DBL_BRACE>
5415SKIP:
5416{
5417 <"//" (~["\n"])* "\n">
5418}
5419
5420<DEFAULT,IN_DBL_BRACE>
5421SKIP:
5422{
5423 <"//" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5424}
5425
5426<DEFAULT,IN_DBL_BRACE>
5427SKIP:
5428{
Yingyi Bu93846a72016-09-13 16:30:39 -07005429 <"--" (~["\n"])* "\n">
5430}
5431
5432
5433<DEFAULT,IN_DBL_BRACE>
5434SKIP:
5435{
5436 <"--" (~["\n","\r"])* ("\n"|"\r"|"\r\n")?>
5437}
5438
5439<DEFAULT,IN_DBL_BRACE>
5440SKIP:
5441{
Yingyi Bu391f09e2015-10-29 13:49:39 -07005442 <"/*"> { pushState(); } : INSIDE_COMMENT
5443}
5444
5445<INSIDE_COMMENT>
5446SPECIAL_TOKEN:
5447{
5448 <"+"(" ")*(~["*"])*>
5449}
5450
5451<INSIDE_COMMENT>
5452SKIP:
5453{
5454 <"/*"> { pushState(); }
5455}
5456
5457<INSIDE_COMMENT>
5458SKIP:
5459{
5460 <"*/"> { popState("*/"); }
5461 | <~[]>
5462}