/** * 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<String> colNames; private List<TypeInfo> colTypes; private transient ObjectInspectorConverters.Converter converters[]; private transient ArrayList<Object> evaluatedArguments; public GenericUDFMacro(String macroName, ExprNodeDesc bodyDesc, List<String> colNames, List<TypeInfo> 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<ObjectInspector> colObjectInspectors = new ArrayList<ObjectInspector>(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<Object>(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<String> colNames) { this.colNames = colNames; } public List<String> getColNames() { return colNames; } public void setColTypes(List<TypeInfo> colTypes) { this.colTypes = colTypes; } public List<TypeInfo> getColTypes() { return colTypes; } }