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

com.couchbase.client.java.util.NodeLocatorHelper Maven / Gradle / Ivy

/*
 * Copyright (c) 2021 Couchbase, 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.couchbase.client.java.util;

import com.couchbase.client.core.annotation.Stability;
import com.couchbase.client.core.config.BucketConfig;
import com.couchbase.client.core.config.ConfigurationProvider;
import com.couchbase.client.core.config.CouchbaseBucketConfig;
import com.couchbase.client.core.config.MemcachedBucketConfig;
import com.couchbase.client.core.config.NodeInfo;
import com.couchbase.client.core.error.CouchbaseException;
import com.couchbase.client.java.Bucket;

import java.time.Duration;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;
import java.util.stream.Collectors;
import java.util.zip.CRC32;

import static java.nio.charset.StandardCharsets.UTF_8;

/**
 * Helper class to provide direct access on how document IDs are mapped onto nodes.
 */
@Stability.Uncommitted
public class NodeLocatorHelper {

  private final AtomicReference bucketConfig = new AtomicReference<>();

  private NodeLocatorHelper(final Bucket bucket, final Duration waitUntilReadyDuration) {
    if (waitUntilReadyDuration.getSeconds() > 0) {
      bucket.waitUntilReady(waitUntilReadyDuration);
    }

    ConfigurationProvider configurationProvider = bucket.core().configurationProvider();

    BucketConfig bc = configurationProvider.config().bucketConfig(bucket.name());
    if (bc == null) {
      throw new CouchbaseException("Bucket configuration not found, if waitUntilReadyDuration is set to 0 the call" +
        "must be executed before initializing the NodeLocatorHelper!");
    }
    bucketConfig.set(configurationProvider.config().bucketConfig(bucket.name()));

    configurationProvider
      .configs()
      .subscribe(cc -> {
        BucketConfig newConfig = cc.bucketConfig(bucket.name());
        if (newConfig != null) {
          bucketConfig.set(newConfig);
        }
      });
  }

  /**
   * Creates a new {@link NodeLocatorHelper}, mapped on to the given {@link Bucket}.
   *
   * To make sure that the helper has a bucket config to work with in the beginning, it will call
   * {@link Bucket#waitUntilReady(Duration)} with the duration provided as an argument. If you already
   * did call waitUntilReady before initializing the helper, you can pass a duration of 0 in which case
   * it will be omitted.
   *
   * @param bucket the scoped bucket.
   * @param waitUntilReadyDuration the duration used to call waitUntilReady (if 0 ignored).
   * @return the created locator.
   */
  public static NodeLocatorHelper create(final Bucket bucket, final Duration waitUntilReadyDuration) {
    return new NodeLocatorHelper(bucket, waitUntilReadyDuration);
  }

  /**
   * Returns the target active node address for a given document ID on the bucket.
   *
   * @param id the document id to convert.
   * @return the node for the given document id.
   */
  public String activeNodeForId(final String id) {
    BucketConfig config = bucketConfig.get();

    if (config instanceof CouchbaseBucketConfig) {
      return nodeForIdOnCouchbaseBucket(id, (CouchbaseBucketConfig) config);
    } else if (config instanceof MemcachedBucketConfig) {
      return nodeForIdOnMemcachedBucket(id, (MemcachedBucketConfig) config);
    } else {
      throw new UnsupportedOperationException("Bucket type not supported: " + config.getClass().getName());
    }
  }

  /**
   * Returns all target replica nodes which are currently available on the bucket.
   *
   * @param id the document ID to check.
   * @return the list of nodes for the given document ID.
   */
  public List availableReplicaNodesForId(final String id) {
    BucketConfig config = bucketConfig.get();

    if (config instanceof CouchbaseBucketConfig) {
      CouchbaseBucketConfig cbc = (CouchbaseBucketConfig) config;
      List replicas = new ArrayList<>();
      for (int i = 1; i <= cbc.numberOfReplicas(); i++) {
        String foundReplica = replicaNodeForId(id, i, false);
        if (foundReplica != null) {
          replicas.add(foundReplica);
        }
      }
      return replicas;
    } else {
      throw new UnsupportedOperationException("Bucket type not supported: " + config.getClass().getName());
    }
  }

  /**
   * Returns all target replica nodes addresses for a given document ID on the bucket.
   *
   * @param id the document id to convert.
   * @return the node for the given document id.
   */
  public List replicaNodesForId(final String id) {
    BucketConfig config = bucketConfig.get();

    if (config instanceof CouchbaseBucketConfig) {
      CouchbaseBucketConfig cbc = (CouchbaseBucketConfig) config;
      List replicas = new ArrayList<>();
      for (int i = 1; i <= cbc.numberOfReplicas(); i++) {
        replicas.add(replicaNodeForId(id, i));
      }
      return replicas;
    } else {
      throw new UnsupportedOperationException("Bucket type not supported: " + config.getClass().getName());
    }
  }

  /**
   * Returns the target replica node address for a given document ID and replica number on the bucket.
   *
   * @param id the document id to convert.
   * @param replicaNum the replica number.
   * @return the node for the given document id.
   */
  public String replicaNodeForId(final String id, final int replicaNum) {
    return replicaNodeForId(id, replicaNum, true);
  }

  /**
   * Returns the target replica node address for a given document ID and replica number on the bucket.
   *
   * @param id the document id to convert.
   * @param replicaNum the replica number.
   * @param throwOnNotAvailable if on -1 and -2 an exception should be thrown.
   * @return the node for the given document id.
   */
  private String replicaNodeForId(final String id, final int replicaNum, final boolean throwOnNotAvailable) {
    if (replicaNum < 1 || replicaNum > 3) {
      throw new IllegalArgumentException("Replica number must be between 1 and 3.");
    }

    BucketConfig config = bucketConfig.get();

    if (config instanceof CouchbaseBucketConfig) {
      CouchbaseBucketConfig cbc = (CouchbaseBucketConfig) config;
      int partitionId = (int) hashId(id) & cbc.numberOfPartitions() - 1;
      int nodeId = cbc.nodeIndexForReplica(partitionId, replicaNum - 1, false);
      if (nodeId == -1) {
        if (throwOnNotAvailable) {
          throw new IllegalStateException("No partition assigned to node for Document ID: " + id);
        } else {
          return null;
        }
      }
      if (nodeId == -2) {
        if (throwOnNotAvailable) {
          throw new IllegalStateException("Replica not configured for this bucket.");
        } else {
          return null;
        }
      }
      return cbc.nodeAtIndex(nodeId).hostname();
    }  else {
      throw new UnsupportedOperationException("Bucket type not supported: " + config.getClass().getName());
    }
  }

  /**
   * Returns all nodes known in the current config.
   *
   * @return all currently known nodes.
   */
  public List nodes() {
    return bucketConfig.get().nodes().stream().map(NodeInfo::hostname).collect(Collectors.toList());
  }

  private static String nodeForIdOnCouchbaseBucket(final String id, final CouchbaseBucketConfig config) {
    int partitionId = (int) hashId(id) & config.numberOfPartitions() - 1;
    int nodeId = config.nodeIndexForActive(partitionId, false);
    if (nodeId == -1) {
      throw new IllegalStateException("No partition assigned to node for Document ID: " + id);
    }
    return config.nodeAtIndex(nodeId).hostname();
  }

  private static String nodeForIdOnMemcachedBucket(final String id, final MemcachedBucketConfig config) {
    return config.nodeForKey(id.getBytes(UTF_8)).hostname();
  }

  private static long hashId(final String id) {
    CRC32 crc32 = new CRC32();
    crc32.update(id.getBytes(UTF_8));
    return (crc32.getValue() >> 16) & 0x7fff;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy