All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.hazelcast.org.apache.calcite.sql.parser.SqlParser Maven / Gradle / Ivy

There is a newer version: 5.5.0
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to you under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.hazelcast.org.apache.calcite.sql.parser;

import com.hazelcast.org.apache.calcite.avatica.util.Casing;
import com.hazelcast.org.apache.calcite.avatica.util.Quoting;
import com.hazelcast.org.apache.calcite.config.Lex;
import com.hazelcast.org.apache.calcite.runtime.CalciteContextException;
import com.hazelcast.org.apache.calcite.sql.SqlNode;
import com.hazelcast.org.apache.calcite.sql.SqlNodeList;
import com.hazelcast.org.apache.calcite.sql.parser.impl.SqlParserImpl;
import com.hazelcast.org.apache.calcite.sql.validate.SqlConformance;
import com.hazelcast.org.apache.calcite.sql.validate.SqlConformanceEnum;
import com.hazelcast.org.apache.calcite.sql.validate.SqlDelegatingConformance;
import com.hazelcast.org.apache.calcite.util.SourceStringReader;

import java.io.Reader;
import java.io.StringReader;
import java.util.List;
import java.util.Objects;

/**
 * A SqlParser parses a SQL statement.
 */
public class SqlParser {
  public static final int DEFAULT_IDENTIFIER_MAX_LENGTH = 128;
  @Deprecated // to be removed before 2.0
  public static final boolean DEFAULT_ALLOW_BANG_EQUAL =
      SqlConformanceEnum.DEFAULT.isBangEqualAllowed();

  //~ Instance fields --------------------------------------------------------
  private final SqlAbstractParserImpl parser;

  //~ Constructors -----------------------------------------------------------
  private SqlParser(SqlAbstractParserImpl parser,
      Config config) {
    this.parser = parser;
    parser.setTabSize(1);
    parser.setQuotedCasing(config.quotedCasing());
    parser.setUnquotedCasing(config.unquotedCasing());
    parser.setIdentifierMaxLength(config.identifierMaxLength());
    parser.setConformance(config.conformance());
    switch (config.quoting()) {
    case DOUBLE_QUOTE:
      parser.switchTo("DQID");
      break;
    case BACK_TICK:
      parser.switchTo("BTID");
      break;
    case BRACKET:
      parser.switchTo("DEFAULT");
      break;
    }
  }

  //~ Methods ----------------------------------------------------------------

  /**
   * Creates a SqlParser to parse the given string using
   * Calcite's parser implementation.
   *
   * 

The default lexical policy is similar to Oracle. * * @see Lex#ORACLE * * @param s An SQL statement or expression to parse. * @return A parser */ public static SqlParser create(String s) { return create(s, configBuilder().build()); } /** * Creates a SqlParser to parse the given string using the * parser implementation created from given {@link SqlParserImplFactory} * with given quoting syntax and casing policies for identifiers. * * @param sql A SQL statement or expression to parse * @param config The parser configuration (identifier max length, etc.) * @return A parser */ public static SqlParser create(String sql, Config config) { return create(new SourceStringReader(sql), config); } /** * Creates a SqlParser to parse the given string using the * parser implementation created from given {@link SqlParserImplFactory} * with given quoting syntax and casing policies for identifiers. * *

Unlike * {@link #create(java.lang.String, com.hazelcast.org.apache.calcite.sql.parser.SqlParser.Config)}, * the parser is not able to return the original query string, but will * instead return "?". * * @param reader The source for the SQL statement or expression to parse * @param config The parser configuration (identifier max length, etc.) * @return A parser */ public static SqlParser create(Reader reader, Config config) { SqlAbstractParserImpl parser = config.parserFactory().getParser(reader); return new SqlParser(parser, config); } /** * Parses a SQL expression. * * @throws SqlParseException if there is a parse error */ public SqlNode parseExpression() throws SqlParseException { try { return parser.parseSqlExpressionEof(); } catch (Throwable ex) { if (ex instanceof CalciteContextException) { final String originalSql = parser.getOriginalSql(); if (originalSql != null) { ((CalciteContextException) ex).setOriginalStatement(originalSql); } } throw parser.normalizeException(ex); } } /** Normalizes a SQL exception. */ private SqlParseException handleException(Throwable ex) { if (ex instanceof CalciteContextException) { final String originalSql = parser.getOriginalSql(); if (originalSql != null) { ((CalciteContextException) ex).setOriginalStatement(originalSql); } } return parser.normalizeException(ex); } /** * Parses a SELECT statement. * * @return A {@link com.hazelcast.org.apache.calcite.sql.SqlSelect} for a regular * SELECT statement; a {@link com.hazelcast.org.apache.calcite.sql.SqlBinaryOperator} * for a UNION, INTERSECT, or EXCEPT. * @throws SqlParseException if there is a parse error */ public SqlNode parseQuery() throws SqlParseException { try { return parser.parseSqlStmtEof(); } catch (Throwable ex) { throw handleException(ex); } } /** * Parses a SELECT statement and reuses parser. * * @param sql sql to parse * @return A {@link com.hazelcast.org.apache.calcite.sql.SqlSelect} for a regular * SELECT statement; a {@link com.hazelcast.org.apache.calcite.sql.SqlBinaryOperator} * for a UNION, INTERSECT, or EXCEPT. * @throws SqlParseException if there is a parse error */ public SqlNode parseQuery(String sql) throws SqlParseException { parser.ReInit(new StringReader(sql)); return parseQuery(); } /** * Parses an SQL statement. * * @return top-level SqlNode representing stmt * @throws SqlParseException if there is a parse error */ public SqlNode parseStmt() throws SqlParseException { return parseQuery(); } /** * Parses a list of SQL statements separated by semicolon. * The semicolon is required between statements, but is * optional at the end. * * @return list of SqlNodeList representing the list of SQL statements * @throws SqlParseException if there is a parse error */ public SqlNodeList parseStmtList() throws SqlParseException { try { return parser.parseSqlStmtList(); } catch (Throwable ex) { throw handleException(ex); } } /** * Get the parser metadata. * * @return {@link SqlAbstractParserImpl.Metadata} implementation of * underlying parser. */ public SqlAbstractParserImpl.Metadata getMetadata() { return parser.getMetadata(); } /** * Returns the warnings that were generated by the previous invocation * of the parser. */ public List getWarnings() { return parser.warnings; } /** * Builder for a {@link Config}. */ public static ConfigBuilder configBuilder() { return new ConfigBuilder(); } /** * Builder for a {@link Config} that starts with an existing {@code Config}. */ public static ConfigBuilder configBuilder(Config config) { return new ConfigBuilder().setConfig(config); } /** * Interface to define the configuration for a SQL parser. * * @see ConfigBuilder */ public interface Config { /** Default configuration. */ Config DEFAULT = configBuilder().build(); int identifierMaxLength(); Casing quotedCasing(); Casing unquotedCasing(); Quoting quoting(); boolean caseSensitive(); SqlConformance conformance(); @Deprecated // to be removed before 2.0 boolean allowBangEqual(); SqlParserImplFactory parserFactory(); } /** Builder for a {@link Config}. */ public static class ConfigBuilder { private Casing quotedCasing = Lex.ORACLE.quotedCasing; private Casing unquotedCasing = Lex.ORACLE.unquotedCasing; private Quoting quoting = Lex.ORACLE.quoting; private int identifierMaxLength = DEFAULT_IDENTIFIER_MAX_LENGTH; private boolean caseSensitive = Lex.ORACLE.caseSensitive; private SqlConformance conformance = SqlConformanceEnum.DEFAULT; private SqlParserImplFactory parserFactory = SqlParserImpl.FACTORY; private ConfigBuilder() {} /** Sets configuration identical to a given {@link Config}. */ public ConfigBuilder setConfig(Config config) { this.quotedCasing = config.quotedCasing(); this.unquotedCasing = config.unquotedCasing(); this.quoting = config.quoting(); this.identifierMaxLength = config.identifierMaxLength(); this.conformance = config.conformance(); this.parserFactory = config.parserFactory(); return this; } public ConfigBuilder setQuotedCasing(Casing quotedCasing) { this.quotedCasing = Objects.requireNonNull(quotedCasing); return this; } public ConfigBuilder setUnquotedCasing(Casing unquotedCasing) { this.unquotedCasing = Objects.requireNonNull(unquotedCasing); return this; } public ConfigBuilder setQuoting(Quoting quoting) { this.quoting = Objects.requireNonNull(quoting); return this; } public ConfigBuilder setCaseSensitive(boolean caseSensitive) { this.caseSensitive = caseSensitive; return this; } public ConfigBuilder setIdentifierMaxLength(int identifierMaxLength) { this.identifierMaxLength = identifierMaxLength; return this; } @SuppressWarnings("unused") @Deprecated // to be removed before 2.0 public ConfigBuilder setAllowBangEqual(final boolean allowBangEqual) { if (allowBangEqual != conformance.isBangEqualAllowed()) { setConformance( new SqlDelegatingConformance(conformance) { @Override public boolean isBangEqualAllowed() { return allowBangEqual; } }); } return this; } public ConfigBuilder setConformance(SqlConformance conformance) { this.conformance = conformance; return this; } public ConfigBuilder setParserFactory(SqlParserImplFactory factory) { this.parserFactory = Objects.requireNonNull(factory); return this; } public ConfigBuilder setLex(Lex lex) { setCaseSensitive(lex.caseSensitive); setUnquotedCasing(lex.unquotedCasing); setQuotedCasing(lex.quotedCasing); setQuoting(lex.quoting); return this; } /** Builds a * {@link Config}. */ public Config build() { return new ConfigImpl(identifierMaxLength, quotedCasing, unquotedCasing, quoting, caseSensitive, conformance, parserFactory); } } /** Implementation of * {@link Config}. * Called by builder; all values are in private final fields. */ private static class ConfigImpl implements Config { private final int identifierMaxLength; private final boolean caseSensitive; private final SqlConformance conformance; private final Casing quotedCasing; private final Casing unquotedCasing; private final Quoting quoting; private final SqlParserImplFactory parserFactory; private ConfigImpl(int identifierMaxLength, Casing quotedCasing, Casing unquotedCasing, Quoting quoting, boolean caseSensitive, SqlConformance conformance, SqlParserImplFactory parserFactory) { this.identifierMaxLength = identifierMaxLength; this.caseSensitive = caseSensitive; this.conformance = Objects.requireNonNull(conformance); this.quotedCasing = Objects.requireNonNull(quotedCasing); this.unquotedCasing = Objects.requireNonNull(unquotedCasing); this.quoting = Objects.requireNonNull(quoting); this.parserFactory = Objects.requireNonNull(parserFactory); } public int identifierMaxLength() { return identifierMaxLength; } public Casing quotedCasing() { return quotedCasing; } public Casing unquotedCasing() { return unquotedCasing; } public Quoting quoting() { return quoting; } public boolean caseSensitive() { return caseSensitive; } public SqlConformance conformance() { return conformance; } public boolean allowBangEqual() { return conformance.isBangEqualAllowed(); } public SqlParserImplFactory parserFactory() { return parserFactory; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy