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

io.stargate.graphql.schema.graphqlfirst.processor.EntityModel Maven / Gradle / Ivy

There is a newer version: 2.0.31
Show newest version
/*
 * Copyright The Stargate Authors
 *
 * 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 io.stargate.graphql.schema.graphqlfirst.processor;

import com.datastax.oss.driver.shaded.guava.common.collect.ImmutableList;
import io.stargate.db.query.Predicate;
import io.stargate.db.schema.Table;
import io.stargate.db.schema.UserDefinedType;
import java.util.Collection;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;

public class EntityModel {

  public enum Target {
    TABLE,
    UDT,
  }

  private final String graphqlName;
  private final String keyspaceName;
  private final String cqlName;
  private final Target target;
  private final List partitionKey;
  private final List clusteringColumns;
  private final List primaryKey;
  private final List primaryKeyWhereConditions;
  private final List regularColumns;
  private final List allColumns;
  private final Table tableCqlSchema;
  private final UserDefinedType udtCqlSchema;
  private final boolean isFederated;
  private final Optional inputTypeName;

  @SuppressWarnings("PMD.ExcessiveParameterList")
  EntityModel(
      String graphqlName,
      String keyspaceName,
      String cqlName,
      Target target,
      List partitionKey,
      List clusteringColumns,
      List regularColumns,
      Table tableCqlSchema,
      UserDefinedType udtCqlSchema,
      boolean isFederated,
      Optional inputTypeName) {

    assert (target == Target.TABLE && tableCqlSchema != null && udtCqlSchema == null)
        || (target == Target.UDT && tableCqlSchema == null && udtCqlSchema != null);

    this.graphqlName = graphqlName;
    this.keyspaceName = keyspaceName;
    this.cqlName = cqlName;
    this.target = target;
    this.partitionKey = ImmutableList.copyOf(partitionKey);
    this.clusteringColumns = ImmutableList.copyOf(clusteringColumns);
    this.primaryKey =
        ImmutableList.builder().addAll(partitionKey).addAll(clusteringColumns).build();
    this.primaryKeyWhereConditions =
        primaryKey.stream()
            .map(field -> new ConditionModel(field, Predicate.EQ, field.getGraphqlName()))
            .collect(Collectors.toList());
    this.regularColumns = ImmutableList.copyOf(regularColumns);
    this.allColumns =
        ImmutableList.builder().addAll(primaryKey).addAll(regularColumns).build();
    this.tableCqlSchema = tableCqlSchema;
    this.udtCqlSchema = udtCqlSchema;
    this.isFederated = isFederated;
    this.inputTypeName = inputTypeName;
  }

  public String getGraphqlName() {
    return graphqlName;
  }

  public String getKeyspaceName() {
    return keyspaceName;
  }

  public String getCqlName() {
    return cqlName;
  }

  public Target getTarget() {
    return target;
  }

  public List getPartitionKey() {
    return partitionKey;
  }

  public List getClusteringColumns() {
    return clusteringColumns;
  }

  /**
   * The full primary key (partition key + clustering columns), that uniquely identifies a CQL row.
   */
  public List getPrimaryKey() {
    return primaryKey;
  }

  /**
   * Returns a default list of WHERE conditions that select all primary key columns with {@code =}
   * relations.
   */
  public List getPrimaryKeyWhereConditions() {
    return primaryKeyWhereConditions;
  }

  public List getRegularColumns() {
    return regularColumns;
  }

  public List getAllColumns() {
    return allColumns;
  }

  public Table getTableCqlSchema() {
    if (target != Target.TABLE) {
      throw new UnsupportedOperationException("Can't call this method when target = " + target);
    }
    return tableCqlSchema;
  }

  public UserDefinedType getUdtCqlSchema() {
    if (target != Target.UDT) {
      throw new UnsupportedOperationException("Can't call this method when target = " + target);
    }
    return udtCqlSchema;
  }

  public boolean isFederated() {
    return isFederated;
  }

  public Optional getInputTypeName() {
    return inputTypeName;
  }

  /**
   * Validates a set of conditions to ensure that they will produce a valid CQL query that doesn't
   * require ALLOW FILTERING for this entity.
   *
   * @return an optional error message if the validation failed, empty otherwise.
   */
  public Optional validateNoFiltering(Collection conditions) {

    long indexCount = conditions.stream().filter(c -> !c.getField().isPrimaryKey()).count();

    if (indexCount == 0) {
      return validateNoFilteringWhenNoIndex(conditions);
    }
    if (indexCount == 1) {
      return validateNoFilteringWhenOneIndex(conditions);
    }
    return Optional.of("cannot use more than one indexed field per operation");
  }

  private Optional validateNoFilteringWhenNoIndex(Collection conditions) {
    // All partition key fields must be present
    for (FieldModel field : this.getPartitionKey()) {
      if (conditions.stream().noneMatch(c -> c.getField().equals(field))) {
        return Optional.of(
            String.format(
                "every partition key field of type %s must be present (expected: %s).",
                this.getGraphqlName(), describePartitionKey()));
      }
    }

    // When a clustering field is NOT restricted by an equality relation, then no other clustering
    // field after it can be restricted by any condition.
    String firstNonEqField = null;
    for (FieldModel field : this.getClusteringColumns()) {
      Optional maybeCondition =
          conditions.stream().filter(c -> c.getField().equals(field)).findFirst();
      if (!maybeCondition.isPresent()) {
        if (firstNonEqField == null) {
          firstNonEqField = field.getGraphqlName();
        }
      } else {
        if (firstNonEqField != null) {
          return Optional.of(
              String.format(
                  "clustering field %s is not restricted by EQ or IN, "
                      + "so no other clustering field after it can be restricted (offending: %s).",
                  firstNonEqField, field.getGraphqlName()));
        }
        Predicate predicate = maybeCondition.get().getPredicate();
        if (predicate != Predicate.EQ && predicate != Predicate.IN) {
          firstNonEqField = field.getGraphqlName();
        }
      }
    }
    return Optional.empty();
  }

  private Optional validateNoFilteringWhenOneIndex(Collection conditions) {

    // No clustering fields
    if (conditions.stream().anyMatch(c -> c.getField().isClusteringColumn())) {
      return Optional.of("when an indexed field is present, no clustering field can be used");
    }

    // Either no partition key fields, or all of them with equality restrictions
    long partitionKeyCount =
        this.getPartitionKey().stream()
            .filter(field -> conditions.stream().anyMatch(c -> c.getField().equals(field)))
            .count();
    if (partitionKeyCount > 0 && partitionKeyCount != this.getPartitionKey().size()) {
      return Optional.of(
          String.format(
              "when an indexed field is present, either none or all of the partition key "
                  + "fields must be present (expected %s).",
              describePartitionKey()));
    }

    return Optional.empty();
  }

  private String describePartitionKey() {
    return this.getPartitionKey().stream()
        .map(FieldModel::getGraphqlName)
        .collect(Collectors.joining(", "));
  }

  /**
   * Validates a set of conditions to ensure that they form a valid WHERE clause for an UPDATE.
   *
   * @return an optional error message if the validation failed, empty otherwise.
   */
  public Optional validateForUpdate(Collection conditions) {
    long primaryKeyCount =
        getPrimaryKey().stream()
            .filter(
                field ->
                    conditions.stream()
                        .anyMatch(
                            (c ->
                                c.getField().equals(field)
                                    && (c.getPredicate() == Predicate.EQ
                                        || c.getPredicate() == Predicate.IN))))
            .count();
    if (primaryKeyCount != getPrimaryKey().size()) {
      return Optional.of(
          String.format(
              "all of the primary key fields must be restricted by EQ or IN predicates (expected %s).",
              describePrimaryKey()));
    }
    return Optional.empty();
  }

  private String describePrimaryKey() {
    return this.getPrimaryKey().stream()
        .map(FieldModel::getGraphqlName)
        .collect(Collectors.joining(", "));
  }

  @Override
  public String toString() {
    return "EntityMappingModel{"
        + "graphqlName='"
        + graphqlName
        + '\''
        + ", keyspaceName='"
        + keyspaceName
        + '\''
        + ", cqlName='"
        + cqlName
        + '\''
        + ", target="
        + target
        + ", partitionKey="
        + partitionKey
        + ", clusteringColumns="
        + clusteringColumns
        + ", primaryKey="
        + primaryKey
        + ", regularColumns="
        + regularColumns
        + ", allColumns="
        + allColumns
        + ", tableCqlSchema="
        + tableCqlSchema
        + ", udtCqlSchema="
        + udtCqlSchema
        + ", isFederated="
        + isFederated
        + ", inputTypeName="
        + inputTypeName
        + '}';
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy