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

weka.knowledgeflow.steps.SetPropertiesFromEnvironment 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 .
 */

/*
 *    SetPropertiesFromEnvironment.java
 *    Copyright (C) 2016 University of Waikato, Hamilton, New Zealand
 *
 */

package weka.knowledgeflow.steps;

import weka.core.EnumHelper;
import weka.core.Environment;
import weka.core.OptionHandler;
import weka.core.SelectedTag;
import weka.core.Tag;
import weka.core.Utils;
import weka.core.WekaException;
import weka.gui.knowledgeflow.KFGUIConsts;
import weka.knowledgeflow.JobEnvironment;
import weka.knowledgeflow.StepManager;
import weka.knowledgeflow.StepManagerImpl;

import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.io.File;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;

/**
 * Step that accesses property values stored in the flow environment and
 * attempts to set them on the algorithm-based step that it is connected to.
 * 
 * @author Mark Hall (mhall{[at]}pentaho{[dot]}com)
 * @version $Revision: $
 */
@KFStep(
  name = "SetPropertiesFromEnvironment",
  category = "Flow",
  toolTipText = "Set properties of the connected algorithm-based step (e.g. "
    + "Classifier, Clusterer etc.) using values stored in the flow environment. "
    + "If no property path for a particular setting value is specified, then it is "
    + "assumed that the value provided is scheme name + options in command-line "
    + "form, in which case the underlying scheme of the connected step will be "
    + "constructed and set; otherwise, the property path is used to set a value "
    + "on the existing underlying scheme.",
  iconPath = KFGUIConsts.BASE_ICON_PATH + "SetPropertiesFromEnvironment.gif")
public class SetPropertiesFromEnvironment extends BaseStep {

  private static final long serialVersionUID = -8316084792512232973L;

  /**
   * Initialize the step.
   *
   * @throws WekaException if a problem occurs
   */
  @Override
  public void stepInit() throws WekaException {
    // all the action happens here in the initialization
    Environment env =
      getStepManager().getExecutionEnvironment().getEnvironmentVariables();
    if (env instanceof JobEnvironment
      && getStepManager().numOutgoingConnections() == 1) {
      Map> outgoing =
        getStepManager().getOutgoingConnections();
      StepManagerImpl connectedManager = null;
      for (Map.Entry> e : outgoing.entrySet()) {
        connectedManager = (StepManagerImpl) e.getValue().get(0);
      }

      if (connectedManager != null) {
        Step connectedStep = connectedManager.getManagedStep();
        String stepName = connectedStep.getName();
        Map propertiesToSet =
          ((JobEnvironment) env).getStepProperties(stepName);

        if (propertiesToSet != null && propertiesToSet.size() > 0) {
          if (connectedStep instanceof WekaAlgorithmWrapper) {
            // only handle algorithm wrappers (just about everything
            // else can handle variables
            setProperties((WekaAlgorithmWrapper) connectedStep, propertiesToSet);
          }
        }
      }
    }
  }

  /**
   * Get a list of acceptable incoming connection types
   *
   * @return a list of acceptable incoming connection types
   */
  @Override
  public List getIncomingConnectionTypes() {
    return null;
  }

  @Override
  public List getOutgoingConnectionTypes() {
    if (getStepManager().numOutgoingConnections() == 0) {
      return Arrays.asList(StepManager.CON_INFO);
    }
    return null;
  }

  /**
   * Set properties on the specified target objet
   *
   * @param target the target to set properties on
   * @param propertiesToSet the properties to try and set
   */
  protected void setProperties(WekaAlgorithmWrapper target,
    Map propertiesToSet) {
    for (Map.Entry e : propertiesToSet.entrySet()) {
      String propName = e.getKey();
      String propVal = e.getValue().trim();
      if (propVal.length() == 0) {
        continue;
      }
      try {
        if (propName.length() == 0) {
          // assume the value is scheme + options for specifying the
          // wrapped algorithm

          String[] schemeAndOpts = Utils.splitOptions(propVal);
          if (schemeAndOpts.length > 0) {
            String schemeName = schemeAndOpts[0];
            schemeAndOpts[0] = "";
            Object valToSet = Utils.forName(null, schemeName, schemeAndOpts);
            setValue(target, target.getName(), "wrappedAlgorithm", valToSet);
          }
        } else {
          // single property on the wrapped algorithm
          String[] propPath = propName.split("\\.");
          Object propRoot = target.getWrappedAlgorithm();
          String propToSet = propPath[propPath.length - 1];
          List remainingPath = new ArrayList<>();
          for (int i = 0; i < propPath.length - 1; i++) {
            remainingPath.add(propPath[i]);
          }
          if (remainingPath.size() > 0) {
            propRoot = drillToProperty(propRoot, remainingPath);
          }
          Object valToSet = stringToVal(propVal, propRoot, propToSet);
          setValue(propRoot, propRoot.getClass().getCanonicalName(), propToSet,
            valToSet);
        }
      } catch (Exception ex) {
        String pN = propName.length() == 0 ? "wrapped algorithm" : propName;
        getStepManager().logWarning(
          "Unable to set " + pN + " with value: " + propVal + " on step "
            + target.getName() + ". Reason: " + ex.getMessage());
      }
    }

    // re-initialize (just in case KF environment has called initStep() on
    // the target WekaAlgorithmWrapper before we get to set its properties
    try {
      target.stepInit();
    } catch (WekaException e) {
      getStepManager().logWarning(
        "Was unable to re-initialize step '" + target.getName()
          + "' after setting properties");
    }
  }

  /**
   * Attempts to convert a property value in string form to the object type that
   * the actual property accepts
   *
   * @param propVal the string representation of the property value
   * @param target the target object to receive the property value
   * @param propName the name of the property
   * @return the property value to set as an object
   * @throws WekaException if a problem occurs
   */
  protected Object stringToVal(String propVal, Object target, String propName)
    throws WekaException {

    Object resultVal = null;
    try {
      PropertyDescriptor prop = getPropDescriptor(target, propName);
      if (prop == null) {
        throw new WekaException("Unable to find method '" + propName + "'");
      }

      Method getMethod = prop.getReadMethod();
      Object current = getMethod.invoke(target);
      if (current.getClass().isArray()) {
        resultVal = Utils.forName(null, propVal, null);
      } else if (current instanceof SelectedTag) {
        Tag[] legalTags = ((SelectedTag) current).getTags();
        int tagIndex = Integer.MAX_VALUE;
        // first try and parse as an integer
        try {
          int specifiedID = Integer.parseInt(propVal);
          for (int z = 0; z < legalTags.length; z++) {
            if (legalTags[z].getID() == specifiedID) {
              tagIndex = z;
              break;
            }
          }
        } catch (NumberFormatException e) {
          // try to match tag strings
          for (int z = 0; z < legalTags.length; z++) {
            if (legalTags[z].getReadable().equals(propVal.trim())) {
              tagIndex = z;
              break;
            }
          }
        }
        if (tagIndex != Integer.MAX_VALUE) {
          resultVal = new SelectedTag(tagIndex, legalTags);
        } else {
          throw new WekaException(
            "Unable to set SelectedTag value for property " + "'" + propName
              + "'");
        }
      } else if (current instanceof Enum) {
        EnumHelper helper = new EnumHelper((Enum) current);
        resultVal = EnumHelper.valueFromString(helper.getEnumClass(), propVal);
      } else if (current instanceof OptionHandler) {
        String[] schemeAndOpts = Utils.splitOptions(propVal);
        if (schemeAndOpts.length > 0) {
          String schemeName = schemeAndOpts[0];
          schemeAndOpts[0] = "";
          resultVal = Utils.forName(null, schemeName, schemeAndOpts);
        }
      } else if (current instanceof Number) {
        try {
          if (current instanceof Integer) {
            resultVal = new Integer(propVal);
          } else if (current instanceof Long) {
            resultVal = new Long(propVal);
          } else if (current instanceof Double) {
            resultVal = new Double(propVal);
          } else if (current instanceof Float) {
            resultVal = new Float(propVal);
          }
        } catch (NumberFormatException ex) {
          throw new WekaException("Unable to parse '" + propVal
            + "' as a number");
        }
      } else if (current instanceof Boolean) {
        resultVal =
          propVal.equalsIgnoreCase("true") || propVal.equalsIgnoreCase("yes")
            || propVal.equalsIgnoreCase("Y");
      } else if (current instanceof String) {
        resultVal = propVal;
      } else if (current instanceof File) {
        resultVal = new File(propVal);
      }

      if (resultVal == null) {
        throw new WekaException("Was unable to determine the value to set for "
          + "property '" + propName + "'");
      }
    } catch (Exception ex) {
      throw new WekaException(ex);
    }

    return resultVal;
  }

  /**
   * Sets the value of the property on the target object
   *
   * @param target the target object
   * @param targetName the name of the step owning the target object
   * @param propName the property name
   * @param valToSet the value to set
   * @throws WekaException if a problem occurs
   */
  protected void setValue(Object target, String targetName, String propName,
    Object valToSet) throws WekaException {

    try {
      getStepManager().logDebug(
        "Attempting to set property '" + propName + "' "
          + "with value of type '" + valToSet.getClass().getCanonicalName()
          + " '(" + valToSet + ") on '" + targetName + "'");
      PropertyDescriptor prop = getPropDescriptor(target, propName);

      if (prop == null) {
        throw new WekaException("Unable to find method '" + propName + "'");
      }
      Method setMethod = prop.getWriteMethod();
      setMethod.invoke(target, valToSet);
    } catch (Exception e) {
      throw new WekaException(e);
    }
  }

  /**
   * Gets the property descriptor for the supplied property name on the
   * supplied target object
   *
   * @param target the target object
   * @param propName the property name
   * @return the property descriptor, or null if no such property exists
   * @throws IntrospectionException if a problem occurs
   */
  protected PropertyDescriptor
    getPropDescriptor(Object target, String propName)
      throws IntrospectionException {
    PropertyDescriptor result = null;
    BeanInfo bi = Introspector.getBeanInfo(target.getClass());
    PropertyDescriptor[] properties = bi.getPropertyDescriptors();
    for (PropertyDescriptor p : properties) {
      if (p.getName().equals(propName)) {
        result = p;
        break;
      }
    }

    return result;
  }

  /**
   * Drill down to the last element in the supplied property path list on
   * the given base object
   *
   * @param baseObject the base object to drill down
   * @param propertyPath the property path to traverse
   * @return the Object corresponding to the path
   * @throws WekaException if a problem occurs
   */
  protected Object
    drillToProperty(Object baseObject, List propertyPath)
      throws WekaException {
    Object objectBeingConfigured = baseObject;
    if (propertyPath != null) {
      for (String methodName : propertyPath) {
        try {
          boolean isArray = methodName.endsWith("]");
          int arrayIndex = -1;
          if (isArray) {
            String arrayPart =
              methodName.substring(methodName.indexOf('[') + 1,
                methodName.lastIndexOf(']'));
            arrayIndex = Integer.parseInt(arrayPart.trim());
            methodName = methodName.substring(0, methodName.indexOf('['));
          }

          BeanInfo bi =
            Introspector.getBeanInfo(objectBeingConfigured.getClass());
          PropertyDescriptor[] properties = bi.getPropertyDescriptors();
          PropertyDescriptor targetProperty = null;
          for (PropertyDescriptor p : properties) {
            if (p.getName().equals(methodName)) {
              targetProperty = p;
              break;
            }
          }
          if (targetProperty == null) {
            throw new WekaException(
              "Unable to find accessor method for property path part: "
                + methodName + " on object "
                + objectBeingConfigured.getClass().getName());
          }
          Method getMethod = targetProperty.getReadMethod();
          Object[] args = {};
          objectBeingConfigured = getMethod.invoke(objectBeingConfigured, args);
          if (isArray) {
            // get the indexed element
            if (!objectBeingConfigured.getClass().isArray()) {
              throw new WekaException("Property path element '" + methodName
                + "' was specified as an array type, but the "
                + "resulting object retrieved "
                + "from this property is not an array!");
            }
            objectBeingConfigured =
              ((Object[]) objectBeingConfigured)[arrayIndex];
          }
        } catch (IntrospectionException ex) {
          throw new WekaException("GOEManager: couldn't introspect", ex);
        } catch (InvocationTargetException e) {
          throw new WekaException("Invocation target exception when invoking "
            + methodName + " on " + objectBeingConfigured.getClass().getName(),
            e);
        } catch (IllegalAccessException e) {
          throw new WekaException(e);
        }
      }
    }
    return objectBeingConfigured;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy