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

org.apache.hadoop.hive.ql.udf.generic.GenericUDFMacro Maven / Gradle / Ivy

/**
 * 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 org.apache.hadoop.hive.ql.udf.generic;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;

import org.apache.hadoop.hive.ql.exec.ExprNodeEvaluator;
import org.apache.hadoop.hive.ql.exec.ExprNodeEvaluatorFactory;
import org.apache.hadoop.hive.ql.exec.UDFArgumentException;
import org.apache.hadoop.hive.ql.exec.UDFArgumentLengthException;
import org.apache.hadoop.hive.ql.metadata.HiveException;
import org.apache.hadoop.hive.ql.plan.ExprNodeDesc;
import org.apache.hadoop.hive.serde2.objectinspector.ObjectInspector;
import org.apache.hadoop.hive.serde2.objectinspector.ObjectInspectorConverters;
import org.apache.hadoop.hive.serde2.objectinspector.ObjectInspectorFactory;
import org.apache.hadoop.hive.serde2.typeinfo.TypeInfo;
import org.apache.hadoop.hive.serde2.typeinfo.TypeInfoUtils;

/**
 * GenericUDFMacro wraps a user-defined macro expression into a GenericUDF
 * interface.
 */
public class GenericUDFMacro extends GenericUDF implements Serializable {

  private static final long serialVersionUID = 2829755821687181020L;
  private String macroName;
  private ExprNodeDesc bodyDesc;
  private transient ExprNodeEvaluator body;
  private List colNames;
  private List colTypes;
  private transient ObjectInspectorConverters.Converter converters[];
  private transient ArrayList evaluatedArguments;

  public GenericUDFMacro(String macroName, ExprNodeDesc bodyDesc,
                         List colNames, List colTypes) {

    this.macroName = macroName;
    this.bodyDesc = bodyDesc;
    this.colNames = colNames;
    this.colTypes = colTypes;
    assert(this.bodyDesc != null);
    assert(colNames.size() == colTypes.size());
  }

  // For serialization only.
  public GenericUDFMacro() {

  }

  public boolean isDeterministic() {
    if(body != null) {
      return body.isDeterministic();
    }
    return true;
  }
  
  public boolean isStateful() {
    if(body != null) {
      return body.isStateful();
    }
    return false;
  }
 
  private void checkNotNull(Object object, String msg) {
    if(object == null) {
      throw new NullPointerException(msg);
    }
  }
  @Override
  public ObjectInspector initialize(ObjectInspector[] arguments) throws UDFArgumentException {
    checkNotNull(colTypes, "colTypes");
    checkNotNull(arguments, "arguments");
    checkNotNull(bodyDesc, "bodyDesc");
    if(colTypes.size() != arguments.length) {
      throw new UDFArgumentLengthException(
          "The macro " + macroName + " accepts exactly " + colTypes.size() + " arguments.");
    }
    try {
      body = ExprNodeEvaluatorFactory.get(bodyDesc);
    } catch (HiveException ex) {
      throw new UDFArgumentException(ex);
    }
    converters = new ObjectInspectorConverters.Converter[arguments.length];
    ArrayList colObjectInspectors = new ArrayList(colTypes.size());
    for (int index = 0; index < arguments.length; ++index) {
      ObjectInspector objectInspector = TypeInfoUtils.
          getStandardWritableObjectInspectorFromTypeInfo(colTypes.get(index));
      colObjectInspectors.add(objectInspector);
      converters[index] =
          ObjectInspectorConverters.getConverter(arguments[index], objectInspector);
    }
    evaluatedArguments = new ArrayList(arguments.length);
    ObjectInspector structOI = ObjectInspectorFactory
        .getStandardStructObjectInspector(colNames, colObjectInspectors);
    try {
      return body.initialize(structOI);
    } catch (HiveException ex) {
      throw new UDFArgumentException(ex);
    }
  }

  @Override
  public Object evaluate(DeferredObject[] arguments) throws HiveException {
    evaluatedArguments.clear();
    for (int index = 0; index < arguments.length; ++index) {
      evaluatedArguments.add(converters[index].convert(arguments[index].get()));
    }
    return body.evaluate(evaluatedArguments);
  }

  @Override
  public String getDisplayString(String[] children) {
      return getStandardDisplayString(macroName, children);
  }

  public void setMacroName(String macroName) {
    this.macroName = macroName;
  }
  public String getMacroName() {
    return macroName;
  }

  public void setBody(ExprNodeDesc bodyDesc) {
    this.bodyDesc = bodyDesc;
  }
  public ExprNodeDesc getBody() {
    return bodyDesc;
  }

  public void setColNames(List colNames) {
    this.colNames = colNames;
  }
  public List getColNames() {
    return colNames;
  }

  public void setColTypes(List colTypes) {
    this.colTypes = colTypes;
  }
  public List getColTypes() {
    return colTypes;
  }
}