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

com.datastax.driver.core.IndexMetadata Maven / Gradle / Ivy

/*
 * Copyright DataStax, Inc.
 *
 * Licensed 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 com.datastax.driver.core;

import com.datastax.driver.core.utils.MoreObjects;
import com.google.common.base.Predicate;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Iterables;
import java.util.Iterator;
import java.util.Map;

/** An immutable representation of secondary index metadata. */
public class IndexMetadata {

  public enum Kind {
    KEYS,
    CUSTOM,
    COMPOSITES
  }

  static final String NAME = "index_name";

  static final String KIND = "kind";

  static final String OPTIONS = "options";

  /** The name of the option used to specify the index target (Cassandra 3.0 onwards). */
  public static final String TARGET_OPTION_NAME = "target";

  /** The name of the option used to specify a custom index class name. */
  public static final String CUSTOM_INDEX_OPTION_NAME = "class_name";

  /** The name of the option used to specify that the index is on the collection (map) keys. */
  public static final String INDEX_KEYS_OPTION_NAME = "index_keys";

  /** The name of the option used to specify that the index is on the collection (map) entries. */
  public static final String INDEX_ENTRIES_OPTION_NAME = "index_keys_and_values";

  private final TableMetadata table;
  private final String name;
  private final Kind kind;
  private final String target;
  private final Map options;

  private IndexMetadata(
      TableMetadata table, String name, Kind kind, String target, Map options) {
    this.table = table;
    this.name = name;
    this.kind = kind;
    this.target = target;
    this.options = options;
  }

  /** Build an IndexMetadata from a system_schema.indexes row. */
  static IndexMetadata fromRow(TableMetadata table, Row indexRow) {
    String name = indexRow.getString(NAME);
    Kind kind = Kind.valueOf(indexRow.getString(KIND));
    Map options = indexRow.getMap(OPTIONS, String.class, String.class);
    String target = options.get(TARGET_OPTION_NAME);
    return new IndexMetadata(table, name, kind, target, options);
  }

  /**
   * Build an IndexMetadata from a legacy layout (index information is stored along with indexed
   * column).
   */
  static IndexMetadata fromLegacy(ColumnMetadata column, ColumnMetadata.Raw raw) {
    Map indexColumns = raw.indexColumns;
    if (indexColumns.isEmpty()) return null;
    String type = indexColumns.get(ColumnMetadata.INDEX_TYPE);
    if (type == null) return null;
    String indexName = indexColumns.get(ColumnMetadata.INDEX_NAME);
    String kindStr = indexColumns.get(ColumnMetadata.INDEX_TYPE);
    Kind kind = kindStr == null ? null : Kind.valueOf(kindStr);
    // Special case check for the value of the index_options column being a string with value 'null'
    // as this
    // column appears to be set this way (JAVA-834).
    String indexOptionsCol = indexColumns.get(ColumnMetadata.INDEX_OPTIONS);
    Map options;
    if (indexOptionsCol == null || indexOptionsCol.isEmpty() || indexOptionsCol.equals("null")) {
      options = ImmutableMap.of();
    } else {
      options = SimpleJSONParser.parseStringMap(indexOptionsCol);
    }
    String target = targetFromLegacyOptions(column, options);
    return new IndexMetadata((TableMetadata) column.getParent(), indexName, kind, target, options);
  }

  private static String targetFromLegacyOptions(
      ColumnMetadata column, Map options) {
    String columnName = Metadata.quoteIfNecessary(column.getName());
    if (options.containsKey(INDEX_KEYS_OPTION_NAME)) return String.format("keys(%s)", columnName);
    if (options.containsKey(INDEX_ENTRIES_OPTION_NAME))
      return String.format("entries(%s)", columnName);
    if (column.getType() instanceof DataType.CollectionType && column.getType().isFrozen())
      return String.format("full(%s)", columnName);
    // Note: the keyword 'values' is not accepted as a valid index target function until 3.0
    return columnName;
  }

  /**
   * Returns the metadata of the table this index is part of.
   *
   * @return the table this index is part of.
   */
  public TableMetadata getTable() {
    return table;
  }

  /**
   * Returns the index name.
   *
   * @return the index name.
   */
  public String getName() {
    return name;
  }

  /**
   * Returns the index kind.
   *
   * @return the index kind.
   */
  public Kind getKind() {
    return kind;
  }

  /**
   * Returns the index target.
   *
   * @return the index target.
   */
  public String getTarget() {
    return target;
  }

  /**
   * Returns whether this index is a custom one.
   *
   * 

If it is indeed a custom index, {@link #getIndexClassName} will return the name of the class * used in Cassandra to implement that index. * * @return {@code true} if this metadata represents a custom index. */ public boolean isCustomIndex() { return getIndexClassName() != null; } /** * The name of the class used to implement the custom index, if it is one. * * @return the name of the class used Cassandra side to implement this custom index if {@code * isCustomIndex() == true}, {@code null} otherwise. */ public String getIndexClassName() { return getOption(CUSTOM_INDEX_OPTION_NAME); } /** * Return the value for the given option name. * * @param name Option name * @return Option value */ public String getOption(String name) { return options != null ? options.get(name) : null; } /** * Returns a CQL query representing this index. * *

This method returns a single 'CREATE INDEX' query corresponding to this index definition. * * @return the 'CREATE INDEX' query corresponding to this index. */ public String asCQLQuery() { String keyspaceName = Metadata.quoteIfNecessary(table.getKeyspace().getName()); String tableName = Metadata.quoteIfNecessary(table.getName()); String indexName = Metadata.quoteIfNecessary(this.name); return isCustomIndex() ? String.format( "CREATE CUSTOM INDEX %s ON %s.%s (%s) USING '%s' %s;", indexName, keyspaceName, tableName, getTarget(), getIndexClassName(), getOptionsAsCql()) : String.format( "CREATE INDEX %s ON %s.%s (%s);", indexName, keyspaceName, tableName, getTarget()); } /** * Builds a string representation of the custom index options. * * @return String representation of the custom index options, similar to what Cassandra stores in * the 'index_options' column of the 'schema_columns' table in the 'system' keyspace. */ private String getOptionsAsCql() { Iterable> filtered = Iterables.filter( options.entrySet(), new Predicate>() { @Override public boolean apply(Map.Entry input) { return !input.getKey().equals(TARGET_OPTION_NAME) && !input.getKey().equals(CUSTOM_INDEX_OPTION_NAME); } }); if (Iterables.isEmpty(filtered)) return ""; StringBuilder builder = new StringBuilder(); builder.append("WITH OPTIONS = {"); Iterator> it = filtered.iterator(); while (it.hasNext()) { Map.Entry option = it.next(); builder.append(String.format("'%s' : '%s'", option.getKey(), option.getValue())); if (it.hasNext()) builder.append(", "); } builder.append("}"); return builder.toString(); } public int hashCode() { return MoreObjects.hashCode(name, kind, target, options); } public boolean equals(Object obj) { if (obj == this) return true; if (!(obj instanceof IndexMetadata)) return false; IndexMetadata other = (IndexMetadata) obj; return MoreObjects.equal(name, other.name) && MoreObjects.equal(kind, other.kind) && MoreObjects.equal(target, other.target) && MoreObjects.equal(options, other.options); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy