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

org.apache.hadoop.hive.accumulo.serde.AccumuloSerDeParameters 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.accumulo.serde;

import java.util.Collections;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.Properties;

import org.apache.accumulo.core.security.Authorizations;
import org.apache.accumulo.core.security.ColumnVisibility;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hive.accumulo.AccumuloConnectionParameters;
import org.apache.hadoop.hive.accumulo.columns.ColumnMapper;
import org.apache.hadoop.hive.accumulo.columns.ColumnMapping;
import org.apache.hadoop.hive.accumulo.columns.HiveAccumuloRowIdColumnMapping;
import org.apache.hadoop.hive.common.JavaUtils;
import org.apache.hadoop.hive.serde.serdeConstants;
import org.apache.hadoop.hive.serde2.SerDeException;
import org.apache.hadoop.hive.serde2.lazy.LazySerDeParameters;
import org.apache.hadoop.hive.serde2.typeinfo.TypeInfo;
import org.apache.hadoop.util.ReflectionUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.google.common.base.Preconditions;

/**
 *
 */
public class AccumuloSerDeParameters extends AccumuloConnectionParameters {
  private static final Logger log = LoggerFactory.getLogger(AccumuloSerDeParameters.class);

  public static final String COLUMN_MAPPINGS = "accumulo.columns.mapping";
  public static final String ITERATOR_PUSHDOWN_KEY = "accumulo.iterator.pushdown";
  public static final boolean ITERATOR_PUSHDOWN_DEFAULT = true;

  public static final String DEFAULT_STORAGE_TYPE = "accumulo.default.storage";

  public static final String VISIBILITY_LABEL_KEY = "accumulo.visibility.label";
  public static final ColumnVisibility DEFAULT_VISIBILITY_LABEL = new ColumnVisibility();

  public static final String AUTHORIZATIONS_KEY = "accumulo.authorizations";

  public static final String COMPOSITE_ROWID_FACTORY = "accumulo.composite.rowid.factory";
  public static final String COMPOSITE_ROWID_CLASS = "accumulo.composite.rowid";

  protected final ColumnMapper columnMapper;

  private Properties tableProperties;
  private String serdeName;
  private LazySerDeParameters lazySerDeParameters;
  private AccumuloRowIdFactory rowIdFactory;

  public AccumuloSerDeParameters(Configuration conf, Properties tableProperties, String serdeName)
      throws SerDeException {
    super(conf);
    this.tableProperties = tableProperties;
    this.serdeName = serdeName;

    lazySerDeParameters = new LazySerDeParameters(conf, tableProperties, serdeName);

    // The default encoding for this table when not otherwise specified
    String defaultStorage = tableProperties.getProperty(DEFAULT_STORAGE_TYPE);

    columnMapper = new ColumnMapper(getColumnMappingValue(), defaultStorage,
        lazySerDeParameters.getColumnNames(), lazySerDeParameters.getColumnTypes());

    log.info("Constructed column mapping " + columnMapper);

    // Generate types for column mapping
    if (null == getColumnTypeValue()) {
      tableProperties.setProperty(serdeConstants.LIST_COLUMN_TYPES, columnMapper.getTypesString());
    }

    if (columnMapper.size() < lazySerDeParameters.getColumnNames().size()) {
      throw new TooManyHiveColumnsException("You have more " + COLUMN_MAPPINGS
          + " fields than hive columns");
    } else if (columnMapper.size() > lazySerDeParameters.getColumnNames().size()) {
      throw new TooManyAccumuloColumnsException(
          "You have more hive columns than fields mapped with " + COLUMN_MAPPINGS);
    }

    this.rowIdFactory = initRowIdFactory(conf, tableProperties);
  }

  protected AccumuloRowIdFactory initRowIdFactory(Configuration conf, Properties tbl)
      throws SerDeException {
    try {
      AccumuloRowIdFactory keyFactory = createRowIdFactory(conf, tbl);
      if (keyFactory != null) {
        keyFactory.init(this, tbl);
      }
      return keyFactory;
    } catch (Exception e) {
      throw new SerDeException(e);
    }
  }

  @SuppressWarnings({"rawtypes", "unchecked"})
  protected AccumuloRowIdFactory createRowIdFactory(Configuration job, Properties tbl)
      throws Exception {
    // Try to load the composite factory if one was provided
    String factoryClassName = tbl.getProperty(COMPOSITE_ROWID_FACTORY);
    if (factoryClassName != null) {
      log.info("Loading CompositeRowIdFactory class " + factoryClassName);
      Class factoryClazz = JavaUtils.loadClass(factoryClassName);
      return (AccumuloRowIdFactory) ReflectionUtils.newInstance(factoryClazz, job);
    }

    // See if a custom CompositeKey class was provided
    String keyClassName = tbl.getProperty(COMPOSITE_ROWID_CLASS);
    if (keyClassName != null) {
      log.info("Loading CompositeRowId class " + keyClassName);
      Class keyClass = JavaUtils.loadClass(keyClassName);
      Class compositeRowIdClass = keyClass
          .asSubclass(AccumuloCompositeRowId.class);
      return new CompositeAccumuloRowIdFactory(compositeRowIdClass);
    }

    return new DefaultAccumuloRowIdFactory();
  }

  public LazySerDeParameters getSerDeParameters() {
    return lazySerDeParameters;
  }

  public Properties getTableProperties() {
    return tableProperties;
  }

  public String getColumnTypeValue() {
    return tableProperties.getProperty(serdeConstants.LIST_COLUMN_TYPES);
  }

  public String getSerDeName() {
    return serdeName;
  }

  public String getColumnMappingValue() {
    return tableProperties.getProperty(COLUMN_MAPPINGS);
  }

  public HiveAccumuloRowIdColumnMapping getRowIdColumnMapping() {
    return columnMapper.getRowIdMapping();
  }

  public boolean getIteratorPushdown() {
    return conf.getBoolean(ITERATOR_PUSHDOWN_KEY, ITERATOR_PUSHDOWN_DEFAULT);
  }

  public List getHiveColumnNames() {
    return Collections.unmodifiableList(lazySerDeParameters.getColumnNames());
  }

  public List getHiveColumnTypes() {
    return Collections.unmodifiableList(lazySerDeParameters.getColumnTypes());
  }

  public ColumnMapper getColumnMapper() {
    return columnMapper;
  }

  public int getRowIdOffset() {
    return columnMapper.getRowIdOffset();
  }

  public List getColumnMappings() {
    return columnMapper.getColumnMappings();
  }

  public AccumuloRowIdFactory getRowIdFactory() {
    return rowIdFactory;
  }

  public String getRowIdHiveColumnName() {
    int rowIdOffset = columnMapper.getRowIdOffset();
    if (-1 == rowIdOffset) {
      return null;
    }

    List hiveColumnNames = lazySerDeParameters.getColumnNames();
    if (0 > rowIdOffset || hiveColumnNames.size() <= rowIdOffset) {
      throw new IllegalStateException("Tried to find rowID offset at position " + rowIdOffset
          + " from Hive columns " + hiveColumnNames);
    }

    return hiveColumnNames.get(rowIdOffset);
  }

  public ColumnMapping getColumnMappingForHiveColumn(String hiveColumn) {
    List hiveColumnNames = lazySerDeParameters.getColumnNames();

    for (int offset = 0; offset < hiveColumnNames.size() && offset < columnMapper.size(); offset++) {
      String hiveColumnName = hiveColumnNames.get(offset);
      if (hiveColumn.equals(hiveColumnName)) {
        return columnMapper.get(offset);
      }
    }

    throw new NoSuchElementException("Could not find column mapping for Hive column " + hiveColumn);
  }

  public TypeInfo getTypeForHiveColumn(String hiveColumn) {
    List hiveColumnNames = lazySerDeParameters.getColumnNames();
    List hiveColumnTypes = lazySerDeParameters.getColumnTypes();

    for (int i = 0; i < hiveColumnNames.size() && i < hiveColumnTypes.size(); i++) {
      String columnName = hiveColumnNames.get(i);
      if (hiveColumn.equals(columnName)) {
        return hiveColumnTypes.get(i);
      }
    }

    throw new NoSuchElementException("Could not find Hive column type for " + hiveColumn);
  }

  /**
   * Extracts the table property to allow a custom ColumnVisibility label to be set on updates to be
   * written to an Accumulo table. The value in the table property must be a properly formatted
   * {@link ColumnVisibility}. If not value is present in the table properties, an empty
   * ColumnVisibility is returned.
   *
   * @return The ColumnVisibility to be applied to all updates sent to Accumulo
   */
  public ColumnVisibility getTableVisibilityLabel() {
    String visibilityLabel = tableProperties.getProperty(VISIBILITY_LABEL_KEY, null);
    if (null == visibilityLabel || visibilityLabel.isEmpty()) {
      return DEFAULT_VISIBILITY_LABEL;
    }

    return new ColumnVisibility(visibilityLabel);
  }

  /**
   * Extracts the table property to allow dynamic Accumulo Authorizations to be used when reading
   * data from an Accumulo table. If no Authorizations are provided in the table properties, null is
   * returned to preserve the functionality to read all data that the current user has access to.
   *
   * @return The Authorizations that should be used to read data from Accumulo, null if no
   *         configuration is supplied.
   */
  public Authorizations getAuthorizations() {
    String authorizationStr = tableProperties.getProperty(AUTHORIZATIONS_KEY, null);

    return getAuthorizationsFromValue(authorizationStr);
  }

  /**
   * Create an Authorizations object when the provided value is not null. Will return null,
   * otherwise.
   *
   * @param authorizationStr
   *          Configuration value to parse
   * @return Authorization object or null
   */
  protected static Authorizations getAuthorizationsFromValue(String authorizationStr) {
    if (null == authorizationStr) {
      return null;
    }

    return new Authorizations(authorizationStr);
  }

  /**
   * Extract any configuration on Authorizations to be used from the provided Configuration. If a
   * non-null value is not present in the configuration, a null object is returned
   *
   * @return Authorization built from configuration value, null if no value is present in conf
   */
  public static Authorizations getAuthorizationsFromConf(Configuration conf) {
    Preconditions.checkNotNull(conf);

    String authorizationStr = conf.get(AUTHORIZATIONS_KEY, null);

    return getAuthorizationsFromValue(authorizationStr);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy