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

io.stargate.graphql.schema.cqlfirst.dml.NameConversions 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.cqlfirst.dml;

import com.datastax.oss.driver.shaded.guava.common.collect.ImmutableMap;
import java.util.Map;

public class NameConversions {

  enum IdentifierType {
    TABLE,
    UDT,
    COLUMN,
  }

  private NameConversions() {}

  public static String toGraphql(String cqlName, IdentifierType type) {
    if (cqlName == null || cqlName.isEmpty()) {
      throw new IllegalArgumentException("CQL name must be non-null and not empty");
    }
    String graphqlName;
    if (type == IdentifierType.TABLE) {
      graphqlName = RESERVED_GRAPHQL_NAMES.get(cqlName);
      if (graphqlName != null) {
        return graphqlName;
      }
    }
    graphqlName = maybeHexEscape(cqlName);
    if (type == IdentifierType.TABLE) {
      graphqlName = escapeReservedPrefix(graphqlName);
      graphqlName = escapeReservedSuffix(graphqlName);
    } else if (type == IdentifierType.UDT) {
      graphqlName = escapeReservedPrefix(graphqlName);
      // No need to escape suffixes because we append our own:
      graphqlName += "Udt";
    }
    return graphqlName;
  }

  public static String toCql(String graphqlName, IdentifierType type) {
    if (graphqlName == null || graphqlName.isEmpty()) {
      throw new IllegalArgumentException("GraphQL name must be non-null and not empty");
    }
    if (type == IdentifierType.UDT && graphqlName.endsWith("Udt")) {
      graphqlName = graphqlName.substring(0, graphqlName.length() - 3);
    }
    return maybeHexUnescape(graphqlName);
  }

  private static String maybeHexEscape(String source) {
    // Note that we traverse the string twice in the worst case; but that's a fast linear iteration,
    // and not escaping should be the most common case anyway.
    return needsHexEscape(source) ? hexEscape(source) : source;
  }

  private static boolean needsHexEscape(String source) {
    assert source != null && !source.isEmpty();
    // GraphQL does not allow leading double underscores or digits
    if (source.startsWith("__") || isDigit(source.charAt(0))) {
      return true;
    }
    int i = 0, length = source.length();
    while (i < length) {
      int cp = source.codePointAt(i), charCount = Character.charCount(cp);
      if (charCount > 1 || !isValidGraphql(cp) || (cp == 'x' && lengthOfHexEscape(source, i) > 0)) {
        return true;
      }
      i += charCount;
    }
    return false;
  }

  private static String hexEscape(String source) {
    assert source != null && !source.isEmpty();
    StringBuilder result = new StringBuilder();
    int i = 0, length = source.length(), cp;
    if (source.startsWith("__")) {
      result.append("x5f__");
      i = 2;
    } else if (isDigit((cp = source.charAt(0)))) {
      appendHexEscape(result, cp);
      i = 1;
    }
    while (i < length) {
      cp = source.codePointAt(i);
      int charCount = Character.charCount(cp);
      if (charCount > 1 || !isValidGraphql(cp) || (cp == 'x' && lengthOfHexEscape(source, i) > 0)) {
        appendHexEscape(result, cp);
      } else {
        result.append((char) cp);
      }
      i += charCount;
    }
    return result.toString();
  }

  private static String maybeHexUnescape(String source) {
    return needsHexUnescape(source) ? hexUnescape(source) : source;
  }

  private static boolean needsHexUnescape(String source) {
    assert source != null && !source.isEmpty();
    int i = 0, length = source.length();
    while (i < length) {
      if (source.charAt(i) == 'x' && lengthOfHexEscape(source, i) > 0) {
        return true;
      }
      i += 1;
    }
    return false;
  }

  private static String hexUnescape(String source) {
    assert source != null && !source.isEmpty();
    StringBuilder result = new StringBuilder();
    int i = 0, length = source.length();
    while (i < length) {
      char c = source.charAt(i);
      int l;
      if (c == 'x' && (l = lengthOfHexEscape(source, i)) > 0) {
        String hexString = source.substring(i + 1, i + l - 1);
        result.appendCodePoint(Integer.parseInt(hexString, 16));
        i += l;
      } else {
        result.append(c);
        i += 1;
      }
    }
    return result.toString();
  }

  /**
   * @return the length of the hex escape starting at the current index (including the leading 'x'
   *     and trailing '_'), or -1 if there is none.
   */
  private static int lengthOfHexEscape(String name, int i) {
    assert name.charAt(i) == 'x';
    int j = i + 1;
    while (j < name.length()) {
      char c = name.charAt(j);
      if (c == '_') {
        // Must have at least one digit, 'x_' is not a valid hex escape.
        return (j == i + 1) ? -1 : j - i + 1;
      } else if (!isHex(c)) {
        return -1;
      }
      j += 1;
    }
    return -1;
  }

  private static void appendHexEscape(StringBuilder target, int codePoint) {
    target.append('x').append(Integer.toHexString(codePoint)).append('_');
  }

  private static boolean isValidGraphql(int c) {
    return isDigit(c) || (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || c == '_';
  }

  private static boolean isHex(int c) {
    return isDigit(c) || (c >= 'a' && c <= 'f');
  }

  private static boolean isDigit(int c) {
    return c >= '0' && c <= '9';
  }

  private static String escapeReservedPrefix(String result) {
    for (Map.Entry entry : RESERVED_PREFIXES.entrySet()) {
      String prefix = entry.getKey();
      String replacement = entry.getValue();
      if (result.startsWith(prefix)) {
        return replacement + result.substring(prefix.length());
      }
    }
    return result;
  }

  private static String escapeReservedSuffix(String result) {
    for (Map.Entry entry : RESERVED_SUFFIXES.entrySet()) {
      String suffix = entry.getKey();
      String replacement = entry.getValue();
      if (result.endsWith(suffix)) {
        return result.substring(0, result.length() - suffix.length()) + replacement;
      }
    }
    return result;
  }

  private static final ImmutableMap RESERVED_GRAPHQL_NAMES =
      ImmutableMap.builder()
          // We use these names for our own types:
          .put("Mutation", "Mutatiox6e_")
          .put("MutationConsistency", "MutationConsistencx79_")
          .put("MutationOptions", "MutationOptionx73_")
          .put("Query", "Querx79_")
          .put("QueryConsistency", "QueryConsistencx79_")
          .put("QueryOptions", "QueryOptionx73_")
          .put("SerialConsistency", "SerialConsistencx79_")
          // Built-in scalars can clash with a table via "FilterInput" types
          .put("Boolean", "Booleax6e_")
          .put("Float", "Floax74_")
          .put("Int", "Inx74_")
          .put("String", "Strinx67_")
          // Custom scalars can clash with a table
          .put("Uuid", "Uuix64_")
          .put("TimeUuid", "TimeUuix64_")
          .put("Inet", "Inex74_")
          .put("Date", "Datx65_")
          .put("Duration", "Duratiox6e_")
          .put("BigInt", "BigInx74_")
          .put("Counter", "Countex72_")
          .put("Ascii", "Ascix69_")
          .put("Decimal", "Decimax6c_")
          .put("Varint", "Varinx74_")
          .put("Float32", "Float32")
          .put("Blob", "Blox62_")
          .put("SmallInt", "SmallInx74_")
          .put("TinyInt", "TinyInx74_")
          .put("Timestamp", "Timestamx70_")
          .put("Time", "Timx65_")
          .put("conversionWarnings", "conversionWarningx73_")
          .build();

  private static final ImmutableMap RESERVED_PREFIXES =
      ImmutableMap.builder()
          .put("Entry", "Entrx79_")
          .put("Tuple", "Tuplx65_")
          .build();

  private static final ImmutableMap RESERVED_SUFFIXES =
      ImmutableMap.builder()
          .put("Input", "Inpux74_")
          .put("Result", "Resulx74_")
          .put("Order", "Ordex72_")
          .put("Udt", "Udx74_")
          .build();
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy