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

weka.core.pmml.Apply Maven / Gradle / Ivy

/*
 *   This program is free software: you can redistribute it and/or modify
 *   it under the terms of the GNU General Public License as published by
 *   the Free Software Foundation, either version 3 of the License, or
 *   (at your option) any later version.
 *
 *   This program is distributed in the hope that it will be useful,
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *   GNU General Public License for more details.
 *
 *   You should have received a copy of the GNU General Public License
 *   along with this program.  If not, see .
 */

/*
 *    Apply.java
 *    Copyright (C) 2008-2012 University of Waikato, Hamilton, New Zealand
 *
 */

package weka.core.pmml;

import java.util.ArrayList;

import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

import weka.core.Attribute;

/**
 * Class encapsulating an Apply Expression.
 * 
 * @author Mark Hall (mhall{[at]}pentaho{[dot]}com)
 * @version $Revision 1.0 $
 */
class Apply extends Expression {
  
  /**
   * For serialization
   */
  private static final long serialVersionUID = -2790648331300695083L;

  /** The list of arguments the function encapsulated in this Apply Expression */
  protected ArrayList m_arguments = new ArrayList();
  
  /** The function to apply (either built-in or a DefineFunction) */
  protected Function m_function = null;
  
  /** The structure of the result of Apply Expression */
  protected Attribute m_outputStructure = null;
  
  /**
   * Constructor. Reads the function name and argument Expressions for
   * this Apply Expression.
   * 
   * @param apply the Element encapsulating this Apply
   * @param opType the optype for this expression (taken from either the 
   * enclosing DefineFunction or DerivedField)
   * @param fieldDefs an ArrayList of Attributes for the fields that this
   * Expression might need to access
   * @param transDict the TransformationDictionary (may be null if there is
   * no dictionary)
   * @throws Exception if there is a problem parsing this Apply Expression
   */
  protected Apply(Element apply, FieldMetaInfo.Optype opType, ArrayList fieldDefs,
      TransformationDictionary transDict) 
    throws Exception {
    super(opType, fieldDefs);
    
    String functionName = apply.getAttribute("function");
    if (functionName == null || functionName.length() == 0) {
      // try the attribute "name" - a sample file produced by MARS
      // uses this attribute name rather than "function" as defined
      // in the PMML spec
      functionName = apply.getAttribute("name");
    }
    
    if (functionName == null || functionName.length() == 0) {
      throw new Exception("[Apply] No function name specified!!");
    }
    //System.err.println(" *** " + functionName);
    m_function = Function.getFunction(functionName, transDict);
    
    // now read the arguments
    NodeList children = apply.getChildNodes();
    for (int i = 0; i < children.getLength(); i++) {
      Node child = children.item(i);
      if (child.getNodeType() == Node.ELEMENT_NODE) {
        String tagName = ((Element)child).getTagName();
        if (!tagName.equals("Extension")) {
          //System.err.println(" ++ " + tagName);
          Expression tempExpression = 
            Expression.getExpression(tagName, child, m_opType, m_fieldDefs, transDict);
          if (tempExpression != null) {
            m_arguments.add(tempExpression);
          }
        }
      }
    }
    
    if (fieldDefs != null) {
      updateDefsForArgumentsAndFunction();
    }
  }
  
  public void setFieldDefs(ArrayList fieldDefs) throws Exception {
    super.setFieldDefs(fieldDefs);
    updateDefsForArgumentsAndFunction();
  }
  
  private void updateDefsForArgumentsAndFunction() throws Exception {
    for (int i = 0; i < m_arguments.size(); i++) {
      m_arguments.get(i).setFieldDefs(m_fieldDefs);
    }
    
    // set the parameter defs for the function here so that we can determine
    // the structure of the output we produce
    ArrayList functionFieldDefs = new ArrayList(m_arguments.size());
    for (int i = 0; i < m_arguments.size(); i++) {     
      functionFieldDefs.add(m_arguments.get(i).getOutputDef());
    }
    m_function.setParameterDefs(functionFieldDefs);
    m_outputStructure = m_function.getOutputDef();
  }

  /**
   * Get the result of evaluating the expression. In the case
   * of a continuous optype, a real number is returned; in
   * the case of a categorical/ordinal optype, the index of the nominal
   * value is returned as a double.
   * 
   * @param incoming the incoming parameter values
   * @return the result of evaluating the expression
   * @throws Exception if there is a problem computing the result
   */
  public double getResult(double[] incoming)
      throws Exception {
    
    // assemble incoming to apply function to by processing each Expression
    // in the list of arguments
    double[] functionIncoming = new double[m_arguments.size()];
    //ArrayList functionParamTypes = m_function.getParameters();
    for (int i = 0; i < m_arguments.size(); i++) {
      functionIncoming[i] = m_arguments.get(i).getResult(incoming);
    }
    
    
    double result = m_function.getResult(functionIncoming);
    
    return result;
  }

  /**
   * Get the result of evaluating the expression for continuous
   * optype. Is the same as calling getResult() when the optype
   * is continuous.
   * 
   * @param incoming the incoming parameter values
   * mining schema
   * @return the result of evaluating the expression.
   * @throws Exception if the optype is not continuous.
   */
  public String getResultCategorical(double[] incoming) throws Exception {
        
    if (m_opType == FieldMetaInfo.Optype.CONTINUOUS) {
      throw new IllegalArgumentException("[Apply] Can't return result as "
          + "categorical/ordinal because optype is continuous!");
    }
    
    double result = getResult(incoming);
    return m_outputStructure.value((int)result);
  }
  
  /**
   * Return the structure of the result of applying this Expression
   * as an Attribute.
   * 
   * @return the structure of the result of applying this Expression as an
   * Attribute.
   */
  public Attribute getOutputDef() {
    if (m_outputStructure == null) {
      // return a "default" output def. This will get replaced
      // by a final one when the final field defs are are set
      // for all expressions after all derived fields are collected
      return (m_opType == FieldMetaInfo.Optype.CATEGORICAL ||
          m_opType == FieldMetaInfo.Optype.ORDINAL)
      ? new Attribute("Placeholder", new ArrayList())
      : new Attribute("Placeholder");
    }
    return m_outputStructure;//.copy(attName);
  }
  
  public String toString(String pad) {
    StringBuffer buff = new StringBuffer();
    
    // Used for DefineFunctions so that we can see which arguments
    // correspond to which parameters
    String[] parameterNames = null;
    
    buff.append(pad + "Apply [" + m_function.toString() +"]:\n");
    buff.append(pad + "args:");
    if (m_function instanceof DefineFunction) {
      parameterNames = m_function.getParameterNames();
    }
    for (int i = 0; i < m_arguments.size(); i++) {
      Expression e = m_arguments.get(i);
      buff.append("\n" + 
          ((parameterNames != null)
              ? pad + parameterNames[i] + " = "
              : "") 
              + e.toString(pad + "  "));
    }
    
    return buff.toString();
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy