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

org.hbase.async.BufferedMultiColumnIncrement Maven / Gradle / Ivy

Go to download

An alternative HBase client library for applications requiring fully asynchronous, non-blocking and thread-safe HBase connectivity.

The newest version!
/*
 * Copyright (C) 2016  The Async HBase Authors.  All rights reserved.
 * This file is part of Async HBase.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *   - Redistributions of source code must retain the above copyright notice,
 *     this list of conditions and the following disclaimer.
 *   - Redistributions in binary form must reproduce the above copyright notice,
 *     this list of conditions and the following disclaimer in the documentation
 *     and/or other materials provided with the distribution.
 *   - Neither the name of the StumbleUpon nor the names of its contributors
 *     may be used to endorse or promote products derived from this software
 *     without specific prior written permission.
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */
package org.hbase.async;

import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.CacheStats;
import com.google.common.cache.LoadingCache;
import com.google.common.cache.RemovalListener;
import com.google.common.cache.RemovalNotification;
import com.stumbleupon.async.Deferred;

import java.util.Arrays;
import java.util.Map;
import java.util.concurrent.atomic.AtomicLong;

/**
 * Package-private class to uniquely identify a buffered multi-column atomic increment.
 * @since 1.8
 */
final class BufferedMultiColumnIncrement {

  private static boolean byteArrayEquals(byte[][] self, byte[][] other) {
    if (self.length != other.length) {
      return false;
    }
    for (int i = 0; i < self.length; i++) {
      if (Bytes.memcmp(self[i], other[i]) != 0) {
        return false;
      }
    }
    return true;
  }

  private static int byteArrayHashCode(byte[][] self) {
    int hashCode = 1;
    for (byte[] aSelf : self) {
      hashCode = Arrays.hashCode(aSelf) + 41 * hashCode;
    }
    return hashCode;
  }

  private static int byteArrayLength(byte[][] self) {
    int len = 0;
    for (byte[] aSelf : self) {
      len += aSelf.length;
    }
    return len;
  }

  private static void byteArrayToString(StringBuilder sb, byte[][] self) {
    for (byte[] aSelf : self) {
      sb.append(Bytes.pretty(aSelf));
    }
  }

  private final byte[] table;
  private final byte[] key;
  private final byte[] family;
  private final byte[][] qualifiers;

  BufferedMultiColumnIncrement(final byte[] table, final byte[] key,
                               final byte[] family, final byte[][] qualifiers) {
    this.table = table;
    this.key = key;
    this.family = family;
    this.qualifiers = qualifiers;
  }

  public boolean equals(final Object other) {
    if (other == null || !(other instanceof BufferedMultiColumnIncrement)) {
      return false;
    }
    final BufferedMultiColumnIncrement incr = (BufferedMultiColumnIncrement) other;
    // Compare fields most likely to be different first.
    return Bytes.equals(key, incr.key)
      && byteArrayEquals(qualifiers, incr.qualifiers)
      && Bytes.equals(family, incr.family)
      && Bytes.equals(table, incr.table);
  }

  public int hashCode() {
    return
      Arrays.hashCode(table) + 41 * (
        Arrays.hashCode(key) + 41 * (
          Arrays.hashCode(family) + 41 * (
              byteArrayHashCode(qualifiers) + 41
          )
        )
      );
  }

  public String toString() {
    final StringBuilder buf =
      new StringBuilder(52 + table.length + key.length * 2 + family.length
                        + byteArrayLength(qualifiers));
    buf.append("BufferedIncrement(table=");
    Bytes.pretty(buf, table);
    buf.append(", key=");
    Bytes.pretty(buf, key);
    buf.append(", family=");
    Bytes.pretty(buf, family);
    buf.append(", qualifier=");
    byteArrayToString(buf, qualifiers);
    buf.append(')');
    return buf.toString();
  }

  /**
   * Atomic increment amounts.
   * 

* This behaves like a signed 49 bit atomic integer that * can only be incremented/decremented a specific number * of times. The {@link #update} method must be used * for all increment/decrement operations, as the underlying * raw value of the {@code long} from {@link AtomicLong} is * not the value of the counter, as some of its bits are * reserved to keep track of how many times the value * got changed. *

* Implementation details:
* The first 49 most significant bits are the value of * the amount (including 1 bit for the sign), the last * 15 least significant bits are the number of times * left that {@link #update} can be called. *

* Again, this class opportunistically inherits from * {@link AtomicLong}, but don't call methods * from the parent class directly. */ static final class Amounts { /** Number of least-significant bits (LSB) we reserve to track updates. */ private final static int UPDATE_BITS = 15; /** Mask used to retrieve number of updates left (0x0000000000007FFFL). */ private final static long UPDATE_MASK = (1L << UPDATE_BITS) - 1; /** Mask used to get value bits we can't store (0xFFFF000000000000L). */ private final static long OVERFLOW_MASK = (UPDATE_MASK << (64 - UPDATE_BITS) >> 1); // Reserve the sign bit. final AtomicLong[] values; /** Everyone waiting for this increment is queued up here. */ final Deferred> deferred = new Deferred>(); /** * Creates a new atomic amount. * @param max_updates The maximum number of times {@link #update} * can be called. Beyond this number of calls, the method will return * {@code false}. */ Amounts(final short max_updates, final int numColumns) { assert max_updates > 0 : "WTF: max_updates=" + max_updates; values = new AtomicLong[numColumns]; for(int i = 0; i < numColumns; i++) { values[i] = new AtomicLong(max_updates); } } /** * Atomically updates this amount. * @param delta The delta by which to increment the value. * Of course, if the delta value is negative, the value will be * decremented instead. * @return {@code true} if the update could be done, {@code false} if * it couldn't due to an overflow/underflow or due to reaching the * maximum number of times this Amount could be incremented. */ final boolean update(final long[] delta) { assert (delta.length == values.length); boolean okToUpdate = true; for(int i = 0; i < delta.length && okToUpdate; i++) { while (true) { final long current = values[i].get(); final int updates = numUpdatesLeft(current); if (updates == 0) { okToUpdate = false; break; // Already too many increments. } final long new_amount = amount(current) + delta[i]; if (!checkOverflow(new_amount)) { okToUpdate = false; break; // Overflow, new amount doesn't fit on 49 bits. } final long next = (new_amount << UPDATE_BITS) | (updates - 1); if (values[i].compareAndSet(current, next)) { // we need to reset it to 0. In case it is a partial failure, the value that has been // successfully applied won't be inc'ed again delta[i] = 0L; break; } // else: CAS failed, loop again. } } return okToUpdate; } /** * Atomically sets the number of updates left to 0 and return the value. * @return The raw value, not the amount by which to increment. * To get the raw value, use {@link #amount} on the value returned. */ final long[] getRawAndInvalidate() { long[] currents = new long[values.length]; for(int i = 0; i < currents.length; i++) { while (true) { final long current = values[i].get(); // Technically, we could leave the whole value set to 0 here, but // to help when debugging with toString(), we restore the amount. final long next = amount(current) << UPDATE_BITS; if (values[i].compareAndSet(current, next)) { // => sets updates left to 0. currents[i] = current; break; } // else: CAS failed, loop again. } } return currents; } /** The amount by which we're going to increment the value in HBase. */ static final long amount(final long n) { return n >> UPDATE_BITS; } /** The number of times left that this amount can be updated. */ static final int numUpdatesLeft(final long n) { return (int) (n & UPDATE_MASK); } /** * Returns {@code true} if the given value can fit without overflowing. */ static final boolean checkOverflow(final long value) { // If the amount was positive, then the MSBs must remain 0. Any 1 in // the MSBs would indicate that the value has become too big or has // become negative due to an overflow. Similarly, if the amount was // negative, then MSBs must remain 1. final long masked = value & OVERFLOW_MASK; return masked == 0 || masked == OVERFLOW_MASK; } public String toString() { long[] currents = new long[values.length]; StringBuilder sb = new StringBuilder(); sb.append("Amounts: "); for(int i = 0; i < values.length; i++) { currents[i] = values[i].get(); sb.append(amount(currents[i])); sb.append("|"); sb.append(numUpdatesLeft(currents[i])); sb.append(","); } sb.append("Deferred: "); sb.append(deferred); return sb.toString(); } private static final long serialVersionUID = 1333868962; } /** * Creates a new cache for buffered increments. * @param client The client to work with. * @param size Max number of entries of the cache. */ static LoadingCache newCache(final HBaseClient client, final int size) { final int ncpu = Runtime.getRuntime().availableProcessors(); return CacheBuilder.newBuilder() // Beef up the concurrency level as this is the number of internal // segments used by the hash map. The default is 4, which is not enough // for us as we typically have more than that many threads concurrently // accessing the map. Because Guava's LocalCache maintains a // per-segment buffer of access operations not yet committed, having a // few more segments than we actually need helps increase the number of // read operations we can do on a segment of the map with no interleaved // writes before the segment has to acquire the lock to flush the buffer. // We can't control this otherwise, because it's a hard-coded constant // in LocalCache.DRAIN_THRESHOLD = 0x3F = 63; // In addition, through benchmarking on a heavily contended cache, I saw // that increasing the number of segments (4x to 8x number of hardware // threads) can help significantly boost overall throughput. .concurrencyLevel(ncpu * 4) .maximumSize(size) .recordStats() // As of Guava 12, stats are disabled by default. .removalListener(new EvictionHandler(client)) .build(LOADER); } /** Creates new zero-Amount for new BufferedIncrements. */ static final class Loader extends CacheLoader { /** * Max number of increments/decrements per counter before we force-flush. * This limit is comes from the max callback chain length on a Deferred. */ private static final short MAX_UPDATES = 16383; @Override public Amounts load(final BufferedMultiColumnIncrement key) { return new Amounts(MAX_UPDATES, key.qualifiers.length); } } /** Singleton instance. */ private static final Loader LOADER = new Loader(); /** Handles cache evictions (also called on flushes). */ private static final class EvictionHandler implements RemovalListener { private final HBaseClient client; EvictionHandler(final HBaseClient client) { this.client = client; } @Override public void onRemoval(final RemovalNotification entry) { final Amounts amounts = entry.getValue(); assert(amounts != null); final long[] raw = amounts.getRawAndInvalidate(); final long[] delta = new long[raw.length]; boolean hasUpdates = false; for(int i = 0; i < raw.length; i++) { delta[i] = Amounts.amount(raw[i]); if (Amounts.numUpdatesLeft(raw[i]) < Loader.MAX_UPDATES) { // This amount was never incremented, because the number of updates // left is still the original number. Therefore this is an Amount // that has been evicted before anyone could attach any update to // it, so the delta must be 0, and we don't need to send this RPC. hasUpdates = true; } } if (hasUpdates) { final BufferedMultiColumnIncrement incr = entry.getKey(); final MultiColumnAtomicIncrementRequest req = new MultiColumnAtomicIncrementRequest(incr.table, incr.key, incr.family, incr.qualifiers, delta); client.atomicIncrement(req).chain(amounts.deferred); } } } static final CacheStats ZERO_STATS = new CacheStats(0, 0, 0, 0, 0, 0); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy