com.amazonaws.services.dynamodbv2.document.ItemCollection Maven / Gradle / Ivy
Show all versions of aws-java-sdk-dynamodb Show documentation
/*
* Copyright 2014-2019 Amazon.com, Inc. or its affiliates. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License").
* You may not use this file except in compliance with the License.
* A copy of the License is located at
*
* http://aws.amazon.com/apache2.0
*
* or in the "license" file accompanying this file. This file 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.amazonaws.services.dynamodbv2.document;
import java.util.HashMap;
import java.util.Map;
import com.amazonaws.services.dynamodbv2.document.internal.PageBasedCollection;
import com.amazonaws.services.dynamodbv2.document.internal.PageIterable;
import com.amazonaws.services.dynamodbv2.model.Capacity;
import com.amazonaws.services.dynamodbv2.model.ConsumedCapacity;
/**
* A collection of Item
's.
*
* An ItemCollection
object maintains a cursor pointing to its
* current pages of data. Initially the cursor is positioned before the first page.
* The next method moves the cursor to the next row, and because it returns
* false when there are no more rows in the ItemCollection
object,
* it can be used in a while loop to iterate through the collection.
*
* Network calls can be triggered when the collection is iterated across page
* boundaries.
*
* @param low level result type
*/
public abstract class ItemCollection extends PageBasedCollection- {
private int accumulatedItemCount;
private int accumulatedScannedCount;
private ConsumedCapacity accumulatedConsumedCapacity;
protected final void accumulateStats(ConsumedCapacity consumedCapacity,
Integer count, Integer scannedCount) {
if (consumedCapacity != null) {
if (accumulatedConsumedCapacity == null) {
// Create a new consumed capacity by cloning the one passed in
this.accumulatedConsumedCapacity = new ConsumedCapacity();
accumulatedConsumedCapacity.setCapacityUnits(consumedCapacity.getCapacityUnits());
accumulatedConsumedCapacity.setGlobalSecondaryIndexes(
clone(consumedCapacity.getGlobalSecondaryIndexes()));
accumulatedConsumedCapacity.setLocalSecondaryIndexes(
clone(consumedCapacity.getLocalSecondaryIndexes()));
accumulatedConsumedCapacity.setTable(clone(consumedCapacity.getTable()));
accumulatedConsumedCapacity.setTableName(consumedCapacity.getTableName());
} else {
// Accumulate the capacity units
final Double capunit = accumulatedConsumedCapacity.getCapacityUnits();
final Double delta = consumedCapacity.getCapacityUnits();
if (capunit == null) {
accumulatedConsumedCapacity.setCapacityUnits(delta);
} else {
accumulatedConsumedCapacity.setCapacityUnits(capunit.doubleValue()
+ (delta == null ? 0 : delta.doubleValue()));
}
// Accumulate the GSI capacities
final Map
gsi = accumulatedConsumedCapacity.getGlobalSecondaryIndexes();
if (gsi == null) {
accumulatedConsumedCapacity.setGlobalSecondaryIndexes(
clone(consumedCapacity.getGlobalSecondaryIndexes()));
} else {
accumulatedConsumedCapacity.setGlobalSecondaryIndexes(add(
consumedCapacity.getGlobalSecondaryIndexes(),
accumulatedConsumedCapacity.getGlobalSecondaryIndexes()));
}
// Accumulate the LSI capacities
final Map lsi = accumulatedConsumedCapacity.getLocalSecondaryIndexes();
if (lsi == null) {
accumulatedConsumedCapacity.setLocalSecondaryIndexes(
clone(consumedCapacity.getLocalSecondaryIndexes()));
} else {
accumulatedConsumedCapacity.setLocalSecondaryIndexes(add(
consumedCapacity.getLocalSecondaryIndexes(),
accumulatedConsumedCapacity.getLocalSecondaryIndexes()));
}
// Accumulate table capacity
final Capacity tableCapacity = accumulatedConsumedCapacity.getTable();
if (tableCapacity == null) {
accumulatedConsumedCapacity.setTable(clone(consumedCapacity.getTable()));
} else {
accumulatedConsumedCapacity.setTable(add(consumedCapacity.getTable(),
accumulatedConsumedCapacity.getTable()));
}
}
}
if (count != null) {
this.accumulatedItemCount += count.intValue();
}
if (scannedCount != null) {
this.accumulatedScannedCount += scannedCount.intValue();
}
}
private Map add(Map from, Map to) {
if (to == null)
return clone(from);
if (from != null) {
for (Map.Entry entryFrom : from.entrySet()) {
final String key = entryFrom.getKey();
final Capacity tocap = to.get(key);
final Capacity fromcap = entryFrom.getValue();
if (tocap == null) {
to.put(key, clone(fromcap));
} else {
to.put(key, new Capacity().withCapacityUnits(
doubleOf(tocap) + doubleOf(fromcap)));
}
}
}
return to;
}
private Capacity add(final Capacity from, final Capacity to) {
return new Capacity().withCapacityUnits(doubleOf(from) + doubleOf(to));
}
private Map clone(Map capacityMap) {
if (capacityMap == null)
return null;
Map clone =
new HashMap(capacityMap.size());
for (Map.Entry e : capacityMap.entrySet())
clone.put(e.getKey(), clone(e.getValue()));
return clone;
}
private Capacity clone(Capacity capacity) {
return capacity == null
? null
: new Capacity().withCapacityUnits(capacity.getCapacityUnits());
}
private double doubleOf(Capacity cap) {
if (cap == null)
return 0.0;
Double val = cap.getCapacityUnits();
return val == null ? 0.0 : val.doubleValue();
}
/**
* Returns the count of items accumulated so far.
* @deprecated This method returns the accumulated count and not the total count.
* Use {@link #getAccumulatedItemCount} instead.
*/
@Deprecated
public int getTotalCount() {
return getAccumulatedItemCount();
}
/**
* Returns the count of items accumulated so far.
*/
public int getAccumulatedItemCount() {
return accumulatedItemCount;
}
/**
* Returns the scanned count accumulated so far.
* @deprecated This method returns the accumulated count and not the total count.
* Use {@link #getAccumulatedScannedCount} instead.
*/
@Deprecated
public int getTotalScannedCount() {
return getAccumulatedScannedCount();
}
/**
* Returns the scanned count accumulated so far.
*/
public int getAccumulatedScannedCount() {
return accumulatedScannedCount;
}
/**
* Returns the consumed capacity accumulated so far.
* @deprecated This method returns the accumulated consumed capacity and not the total.
* Use {@link #getAccumulatedScannedCount} instead.
*/
@Deprecated
public ConsumedCapacity getTotalConsumedCapacity() {
return getAccumulatedConsumedCapacity();
}
/**
* Returns the consumed capacity accumulated so far.
*/
public ConsumedCapacity getAccumulatedConsumedCapacity() {
return accumulatedConsumedCapacity;
}
// Overriding these just so javadocs will show up.
/**
* Returns an {@code Iterable>} that iterates over pages of
* items from this collection. Each call to {@code Iterator.next} on an
* {@code Iterator} returned from this {@code Iterable} results in exactly
* one call to DynamoDB to retrieve a single page of results.
*
*
* ItemCollection<QueryResult> collection = ...;
* for (Page<Item> page : collection.pages()) {
* processItems(page);
*
* ConsumedCapacity consumedCapacity =
* page.getLowLevelResult().getConsumedCapacity();
*
* Thread.sleep(getBackoff(consumedCapacity.getCapacityUnits()));
* }
*
*
* The use of the internal/undocumented {@code PageIterable} class instead
* of {@code Iterable} in the public interface here is retained for
* backwards compatibility. It doesn't expose any methods beyond those
* of the {@code Iterable} interface. This method will be changed to return
* an {@code Iterable>} directly in a future release of the
* SDK.
*
* @see Page
*/
@Override
public PageIterable- pages() {
return super.pages();
}
/**
* Returns the maximum number of resources to be retrieved in this
* collection; or null if there is no limit.
*/
@Override
public abstract Integer getMaxResultSize();
/**
* Returns the low-level result last retrieved (for the current page) from
* the server side; or null if there has yet no calls to the server.
*/
@Override
public R getLastLowLevelResult() {
return super.getLastLowLevelResult();
}
/**
* Used to register a listener for the event of receiving a low-level result
* from the server side.
*
* @param listener
* listener to be registered. If null, a "none" listener will be
* set.
* @return the previously registered listener. The return value is never
* null.
*/
@Override
public LowLevelResultListener
registerLowLevelResultListener(
LowLevelResultListener listener) {
return super.registerLowLevelResultListener(listener);
}
}