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

org.knowm.yank.processors.YankBeanProcessor Maven / Gradle / Ivy

package org.knowm.yank.processors;

import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import org.apache.commons.dbutils.BeanProcessor;
import org.knowm.yank.annotations.Column;

/**
 * Combines the override map of BeanProcessor with the snake case mapping of
 * GenererousBeanProcessor. Uses Column annotations to build map
 *
 * @author timmolter
 */
public class YankBeanProcessor extends BeanProcessor {

  /** ResultSet column to bean property name overrides. */
  private final Map columnToFieldOverrides;

  /**
   * Constructor for YankBeanProcessor configured with Bean class type to look for "Column"
   * annotations for column ==> field mapping
   *
   * @param type The Bean type
   */
  public YankBeanProcessor(Class type) {
    super();
    this.columnToFieldOverrides = getMappingFromAnnotations(type);
  }

  private Map getMappingFromAnnotations(Class type) {

    final Map columnToPropertyOverrides = new HashMap();

    for (Field field : type.getDeclaredFields()) {
      if (field.isAnnotationPresent(Column.class)) {
        columnToPropertyOverrides.put(field.getAnnotation(Column.class).value(), field.getName());
      }
    }

    return columnToPropertyOverrides;
  }

  /**
   * The positions in the returned array represent column numbers. The values stored at each
   * position represent the index in the PropertyDescriptor[] for the bean property
   * that matches the column name. Also tried to match snake case column names or overrides. If no
   * bean property was found for a column, the position is set to PROPERTY_NOT_FOUND.
   *
   * @param rsmd The ResultSetMetaData containing column information.
   * @param props The bean property descriptors.
   * @throws SQLException if a database access error occurs
   * @return An int[] with column index to property index mappings. The 0th element is meaningless
   *     because JDBC column indexing starts at 1.
   */
  @Override
  protected int[] mapColumnsToProperties(
      final ResultSetMetaData rsmd, final PropertyDescriptor[] props) throws SQLException {

    final int cols = rsmd.getColumnCount();
    final int[] columnToProperty = new int[cols + 1];
    Arrays.fill(columnToProperty, PROPERTY_NOT_FOUND);

    for (int col = 1; col <= cols; col++) {
      String columnName = rsmd.getColumnLabel(col);

      if (null == columnName || 0 == columnName.length()) {
        columnName = rsmd.getColumnName(col);
      }

      String overrideName = columnToFieldOverrides.get(columnName);
      if (overrideName == null) {
        overrideName = columnName;
      }

      final String generousColumnName = columnName.replace("_", "");

      for (int i = 0; i < props.length; i++) {
        final String propName = props[i].getName();

        // see if either the column name, or the generous one matches
        if (columnName.equalsIgnoreCase(propName)
            || generousColumnName.equalsIgnoreCase(propName)
            || overrideName.equalsIgnoreCase(propName)) {
          columnToProperty[col] = i;
          break;
        }
      }
    }

    return columnToProperty;
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy