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

org.verdictdb.sqlreader.ExpressionGen Maven / Gradle / Ivy

/*
 *    Copyright 2018 University of Michigan
 *
 *    Licensed 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 org.verdictdb.sqlreader;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

import org.antlr.v4.runtime.tree.ParseTree;
import org.verdictdb.core.sqlobject.AsteriskColumn;
import org.verdictdb.core.sqlobject.BaseColumn;
import org.verdictdb.core.sqlobject.ColumnOp;
import org.verdictdb.core.sqlobject.ConstantColumn;
import org.verdictdb.core.sqlobject.SelectQuery;
import org.verdictdb.core.sqlobject.SubqueryColumn;
import org.verdictdb.core.sqlobject.UnnamedColumn;
import org.verdictdb.parser.VerdictSQLParser;
import org.verdictdb.parser.VerdictSQLParser.Column_nameContext;
import org.verdictdb.parser.VerdictSQLParser.Full_column_nameContext;
import org.verdictdb.parser.VerdictSQLParserBaseVisitor;

import com.google.common.base.Joiner;

public class ExpressionGen extends VerdictSQLParserBaseVisitor {

  //    private MetaData meta;

  //    public ExpressionGen(MetaData meta) {
  //        this.meta = meta;
  //    }

  public ExpressionGen() {}

  @Override
  public ColumnOp visitInterval(VerdictSQLParser.IntervalContext ctx) {
    String unit = "day";
    if (ctx.DAY() != null || ctx.DAYS() != null) {
      unit = "day";
    } else if (ctx.MONTH() != null || ctx.MONTHS() != null) {
      unit = "month";
    } else if (ctx.YEAR() != null || ctx.YEARS() != null) {
      unit = "year";
    }
    return new ColumnOp(
        "interval",
        Arrays.asList(
            ConstantColumn.valueOf(ctx.constant_expression().getText()),
            ConstantColumn.valueOf(unit)));
  }

  @Override
  public ColumnOp visitDate(VerdictSQLParser.DateContext ctx) {
    return new ColumnOp(
        "date",
        Arrays.asList(ConstantColumn.valueOf(ctx.constant_expression().getText())));
  }

  @Override
  public ConstantColumn visitPrimitive_expression(
      VerdictSQLParser.Primitive_expressionContext ctx) {
    return ConstantColumn.valueOf(ctx.getText());
  }

  @Override
  public ConstantColumn visitTime_unit(VerdictSQLParser.Time_unitContext ctx) {
    return ConstantColumn.valueOf(ctx.getText());
  }

  @Override
  public BaseColumn visitColumn_ref_expression(VerdictSQLParser.Column_ref_expressionContext ctx) {
    Full_column_nameContext fullColName = ctx.full_column_name();
    if (fullColName == null) {
      return null;
    }

    Column_nameContext columnName = fullColName.column_name();
    String colName = stripQuote(columnName.getText());
    if (fullColName.table_name() == null) {
      return new BaseColumn(colName);
    }
    String tableName = stripQuote(fullColName.table_name().table.getText());
    if (fullColName.table_name().schema == null) {
      return new BaseColumn(tableName, colName);
    } else {
      return new BaseColumn(
          stripQuote(fullColName.table_name().schema.getText()), tableName, colName);
    }
  }

  private String stripQuote(String expr) {
    return expr.replace("\"", "").replace("`", "");
  }

  @Override
  public ColumnOp visitBinary_operator_expression(
      VerdictSQLParser.Binary_operator_expressionContext ctx) {
    String opType = null;
    if (ctx.op.getText().equals("+")) {
      opType = "add";
    } else if (ctx.op.getText().equals("-")) {
      opType = "subtract";
    } else if (ctx.op.getText().equals("*")) {
      opType = "multiply";
    } else if (ctx.op.getText().equals("/")) {
      opType = "divide";
    } else {
      opType = ctx.op.getText();
    }
    return new ColumnOp(opType, Arrays.asList(visit(ctx.expression(0)), visit(ctx.expression(1))));
  }

  @Override
  public UnnamedColumn visitIs_null_expression(VerdictSQLParser.Is_null_expressionContext ctx) {
    UnnamedColumn left = visit(ctx.expression());

    if (ctx.null_notnull().NOT() == null) {
      return ColumnOp.rightisnull(left); // ?? is null
    } else {
      return ColumnOp.rightisnotnull(left); // ?? is not null
    }
  }

  @Override
  public UnnamedColumn visitNot_expression(VerdictSQLParser.Not_expressionContext ctx) {
    UnnamedColumn col = visit(ctx.expression());

    return ColumnOp.not(col);
  }

  @Override
  public ColumnOp visitFunction_call_expression(
      VerdictSQLParser.Function_call_expressionContext ctx) {
    VerdictSQLParserBaseVisitor v =
        new VerdictSQLParserBaseVisitor() {

          @Override
          public ColumnOp visitAggregate_windowed_function(
              VerdictSQLParser.Aggregate_windowed_functionContext ctx) {
            String fname;
            UnnamedColumn col = null;
            if (ctx.all_distinct_expression() != null) {
              ExpressionGen g = new ExpressionGen();
              col = g.visit(ctx.all_distinct_expression());
            }

            // OverClause overClause = null;

            if (ctx.AVG() != null) {
              fname = "avg";
            } else if (ctx.SUM() != null) {
              fname = "sum";
            } else if (ctx.COUNT() != null) {
              if (ctx.all_distinct_expression() != null
                  && ctx.all_distinct_expression().DISTINCT() != null) {
                fname = "countdistinct";
              } else {
                fname = "count";
                if (ctx.all_distinct_expression() != null) {
                  ExpressionGen g = new ExpressionGen();
                  col = g.visit(ctx.all_distinct_expression());
                } else {
                  col = new AsteriskColumn();
                }
              }
            } // else if (ctx.NDV() != null) {
            //  fname = FuncName.IMPALA_APPROX_COUNT_DISTINCT;}
            else if (ctx.MIN() != null) {
              fname = "min";
            } else if (ctx.MAX() != null) {
              fname = "max";
            } // else if (ctx.STDDEV_SAMP() != null) {
            //  fname = FuncName.STDDEV_SAMP; }
            else {
              fname = "UNKNOWN";
            }

            //  if (ctx.over_clause() != null) {
            //      overClause = OverClause.from(vc, ctx.over_clause());
            //  }

            return new ColumnOp(fname, col);
          }

          @Override
          public ColumnOp visitUnary_function(VerdictSQLParser.Unary_functionContext ctx) {
            ExpressionGen g = new ExpressionGen();
            //        if (ctx.predicate_function()!=null) {
            //          String fname =
            // ctx.predicate_function().function_name.getText().toLowerCase();
            //          if (ctx.NOT()!=null) {
            //            return new ColumnOp("not " + fname,
            // g.visit(ctx.predicate_function().expression()));
            //          }
            //          else return new ColumnOp(fname, g.visit(ctx.expression()));
            //        }
            String fname = ctx.function_name.getText().toLowerCase();
            if (fname.equals("cast")) {
              List dataTypeStr = new ArrayList<>();
              for (ParseTree child : ctx.cast_as_expression().data_type().children) {
                dataTypeStr.add(child.getText());
              }
              return new ColumnOp(
                  fname,
                  Arrays.asList(
                      g.visit(ctx.cast_as_expression().expression()),
                      ConstantColumn.valueOf(Joiner.on(" ").join(dataTypeStr))));
              // ConstantColumn.valueOf(ctx.cast_as_expression().data_type().getText())));
            } else {
              return new ColumnOp(fname, g.visit(ctx.expression()));
            }
          }

          @Override
          public ColumnOp visitNoparam_function(VerdictSQLParser.Noparam_functionContext ctx) {
            String fname = ctx.function_name.getText().toLowerCase();
            return new ColumnOp(fname);
          }

          @Override
          public ColumnOp visitBinary_function(VerdictSQLParser.Binary_functionContext ctx) {
            String fname = ctx.function_name.getText().toLowerCase();
            ExpressionGen g = new ExpressionGen();
            return new ColumnOp(
                fname,
                Arrays.asList(
                    g.visit(ctx.expression(0)), g.visit(ctx.expression(1))));
          }

          @Override
          public ColumnOp visitTernary_function(VerdictSQLParser.Ternary_functionContext ctx) {
            String fname = ctx.function_name.getText().toLowerCase();
            ExpressionGen g = new ExpressionGen();
            return new ColumnOp(
                fname,
                Arrays.asList(
                    g.visit(ctx.expression(0)),
                    g.visit(ctx.expression(1)),
                    g.visit(ctx.expression(2))));
          }

          @Override
          public ColumnOp visitNary_function(VerdictSQLParser.Nary_functionContext ctx) {
            String fname = ctx.function_name.getText().toLowerCase();
            ExpressionGen g = new ExpressionGen();
            List columns = new ArrayList<>();
            for (VerdictSQLParser.ExpressionContext expressionContext : ctx.expression()) {
              columns.add(g.visit(expressionContext));
            }
            return new ColumnOp(fname, columns);
          }

          @Override
          public ColumnOp visitExtract_time_function(
              VerdictSQLParser.Extract_time_functionContext ctx) {
            String fname = "extract";
            ExpressionGen g = new ExpressionGen();
            return new ColumnOp(
                fname,
                Arrays.asList(
                    ConstantColumn.valueOf(ctx.extract_unit().getText()),
                    g.visit(ctx.expression())));
          }

          @Override
          public ColumnOp visitOverlay_string_function(
              VerdictSQLParser.Overlay_string_functionContext ctx) {
            String fname = "overlay";
            ExpressionGen g = new ExpressionGen();
            List operands = new ArrayList<>();
            operands.add(g.visit(ctx.expression(0)));
            operands.add(g.visit(ctx.expression(1)));
            operands.add(g.visit(ctx.expression(2)));
            if (ctx.expression().size() == 4) {
              operands.add(g.visit(ctx.expression(3)));
            }
            return new ColumnOp(fname, operands);
          }

          @Override
          public ColumnOp visitSubstring_string_function(
              VerdictSQLParser.Substring_string_functionContext ctx) {
            String fname = "substring";
            ExpressionGen g = new ExpressionGen();
            List operands = new ArrayList<>();
            operands.add(g.visit(ctx.expression(0)));
            operands.add(g.visit(ctx.expression(1)));
            if (ctx.expression().size() == 3) {
              operands.add(g.visit(ctx.expression(2)));
            }
            return new ColumnOp(fname, operands);
          }

          @Override
          public ColumnOp visitTimestamp_function(VerdictSQLParser.Timestamp_functionContext ctx) {
            ExpressionGen g = new ExpressionGen();
            return new ColumnOp("timestampwithoutparentheses", g.visit(ctx.expression()));
          }

          @Override
          public ColumnOp visitDateadd_function(VerdictSQLParser.Dateadd_functionContext ctx) {
            String fname = "dateadd";
            ExpressionGen g = new ExpressionGen();
            List operands = new ArrayList<>();
            operands.add(g.visit(ctx.time_unit()));
            operands.add(g.visit(ctx.expression(0)));
            operands.add(g.visit(ctx.expression(1)));
            return new ColumnOp(fname, operands);
          }
        };
    return v.visit(ctx);
  }

  @Override
  public ColumnOp visitCase_expr(VerdictSQLParser.Case_exprContext ctx) {
    if (ctx.search_condition() != null) {
      List operands = new ArrayList<>();
      CondGen gen = new CondGen();
      for (VerdictSQLParser.ExpressionContext expressionContext : ctx.expression()) {
        int index = ctx.expression().indexOf(expressionContext);
        if (index != ctx.expression().size() - 1)
          operands.add(gen.visit(ctx.search_condition(index)));
        operands.add(visit(expressionContext));
      }
      return new ColumnOp("casewhen", operands);
    } else {
      List operands = new ArrayList<>();
      ExpressionGen gen = new ExpressionGen();
      for (VerdictSQLParser.ExpressionContext expressionContext : ctx.expression()) {
        operands.add(gen.visit(expressionContext));
      }
      return new ColumnOp("caseexprwhen", operands);
    }
  }

  @Override
  public UnnamedColumn visitBracket_expression(VerdictSQLParser.Bracket_expressionContext ctx) {
    return visit(ctx.expression());
  }

  @Override
  public SubqueryColumn visitSubquery_expression(VerdictSQLParser.Subquery_expressionContext ctx) {
    RelationGen g = new RelationGen();
    return SubqueryColumn.getSubqueryColumn(
        (SelectQuery) g.visit(ctx.subquery().select_statement()));
  }

  public UnnamedColumn getSearch_condition(List ctx) {
    CondGen g = new CondGen();
    if (ctx.size() == 1) {
      return g.visit(ctx.get(0));
    } else {
      UnnamedColumn col = visit(ctx.get(0));
      for (int i = 0; i < ctx.size(); i++) {
        col = new ColumnOp("and", Arrays.asList(col, g.visit(ctx.get(i))));
      }
      return col;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy