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

io.vertx.redis.client.impl.KeyExtractor Maven / Gradle / Ivy

package io.vertx.redis.client.impl;

import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;

import static io.vertx.redis.client.Command.*;

class KeyExtractor {

  static final byte[] STREAMS_UPPER_CASE = "STREAMS".getBytes(StandardCharsets.UTF_8);
  static final byte[] STREAMS_LOWER_CASE = "streams".getBytes(StandardCharsets.UTF_8);
  static final byte[] KEYS_UPPER_CASE = "KEYS".getBytes(StandardCharsets.UTF_8);
  static final byte[] KEYS_LOWER_CASE = "keys".getBytes(StandardCharsets.UTF_8);
  static final byte[] STORE_UPPER_CASE = "STORE".getBytes(StandardCharsets.UTF_8);
  static final byte[] STORE_LOWER_CASE = "store".getBytes(StandardCharsets.UTF_8);
  static final byte[] STOREDIST_UPPER_CASE = "STOREDIST".getBytes(StandardCharsets.UTF_8);
  static final byte[] STOREDIST_LOWER_CASE = "storedist".getBytes(StandardCharsets.UTF_8);
  private static final byte ZERO = 48;
  private static final byte HYPHEN = 45;
  private static final byte DOLLAR = 36;


  /**
   * Extract movable keys from given command request. Movable means the count of keys and may even their position within
   * command is movable.
   *
   * @return All keys of the command of the given request.
   */
  static byte[][] extractMovableKeys(final RequestImpl req) {
    final List args = Collections.unmodifiableList(req.getArgs());
    final List keys = new ArrayList<>();

    if (req.command().equals(EVAL) || req.command().equals(EVALSHA)) {
      final int keyCount = Integer.parseInt(new String(args.get(1)));
      final int keysStartIndex = 2;
      keys.addAll(args.subList(keysStartIndex, keysStartIndex + keyCount));
    }

    if (req.command().equals(GEORADIUS)
      || req.command().equals(GEORADIUS_RO)
      || req.command().equals(GEORADIUSBYMEMBER)
      || req.command().equals(GEORADIUSBYMEMBER_RO)) {
      keys.add(args.get(0));
      if (!addKeyAfterKeywordIfPresent(args, keys, STORE_LOWER_CASE)) {
        addKeyAfterKeywordIfPresent(args, keys, STORE_UPPER_CASE);
      }
      if (!addKeyAfterKeywordIfPresent(args, keys, STOREDIST_LOWER_CASE)) {
        addKeyAfterKeywordIfPresent(args, keys, STOREDIST_UPPER_CASE);
      }
    }

    if (req.command().equals(SORT)) {
      keys.add(args.get(0));
      if (!addKeyAfterKeywordIfPresent(args, keys, STORE_LOWER_CASE)) {
        addKeyAfterKeywordIfPresent(args, keys, STORE_UPPER_CASE);
      }
    }

    if (req.command().equals(MIGRATE)) {
      int keysKeywordIndex = getIndexOfArgContainsKeyword(args, KEYS_UPPER_CASE);
      if (keysKeywordIndex == -1) {
        keysKeywordIndex = getIndexOfArgContainsKeyword(args, KEYS_LOWER_CASE);
      }
      if (keysKeywordIndex > -1) {
        keys.addAll(args.subList(keysKeywordIndex + 1, args.size()));
      } else {
        keys.add(args.get(2));
      }
    }

    if (req.command().equals(XREAD) || req.command().equals(XREADGROUP)) {
      int streamsKeywordIndex = getIndexOfArgContainsKeyword(args, STREAMS_UPPER_CASE);
      if (streamsKeywordIndex == -1) {
        streamsKeywordIndex = getIndexOfArgContainsKeyword(args, STREAMS_LOWER_CASE);
      }

      final List argsAfterStreamsKeyword = args.subList(streamsKeywordIndex + 1, args.size());

      for (byte[] arg : argsAfterStreamsKeyword) {
        if (!isArgEqualsSign(arg, DOLLAR) && !isArgEqualsSign(arg, ZERO) && !isArgContainsSign(arg, HYPHEN)) {
          keys.add(arg);
        } else if (isArgContainsSign(arg, HYPHEN)) {
          // If hyphen separator, we need to verify both parts, before and after hyphen are parsable to long
          String mayId = new String(arg, StandardCharsets.UTF_8);
          String[] millisAndSequence = mayId.split("-");
          try {
            Long.parseLong(millisAndSequence[0]);
            Long.parseLong(millisAndSequence[1]);
          } catch (NumberFormatException e) {
            keys.add(arg);
          }
        }
      }
    }

    if (req.command().equals(ZINTERSTORE) || req.command().equals(ZUNIONSTORE)) {
      final int keyCount = Integer.parseInt(new String(args.get(1)));
      // destination
      keys.add(args.get(0));
      final int keysStartIndex = 2;
      keys.addAll(args.subList(keysStartIndex, keysStartIndex + keyCount));
    }

    return keys.toArray(new byte[keys.size()][]);
  }

  private static boolean addKeyAfterKeywordIfPresent(List args, List keys, byte[] keyword) {
    int keywordIndex = getIndexOfArgContainsKeyword(args, keyword);
    if (keywordIndex > -1) {
      keys.add(args.get(keywordIndex + 1));
      return true;
    }
    return false;
  }

  private static int getIndexOfArgContainsKeyword(List args, byte[] keyword) {
    int keywordIndex = -1;
    for (int i = 0; i < args.size(); i++) {
      if (Arrays.equals(args.get(i), keyword)) {
        keywordIndex = i;
        break;
      }
    }
    return keywordIndex;
  }

  private static boolean isArgContainsSign(byte[] arg, byte sign) {
    for (byte argByte : arg) {
      if (argByte == sign) {
        return true;
      }
    }
    return false;
  }

  private static boolean isArgEqualsSign(byte[] arg, byte sign) {
    return arg.length == 1 && arg[0] == sign;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy