/*
* Copyright 2015 Ben Manes. 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.
* 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.github.benmanes.caffeine.cache;
import static com.github.benmanes.caffeine.cache.RandomSeedEnforcer.ensureRandomSeed;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.lessThanOrEqualTo;
import static org.hamcrest.Matchers.nullValue;
import java.util.concurrent.ThreadLocalRandom;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
/**
* @author ben.manes@gmail.com (Ben Manes)
*/
public final class FrequencySketchTest {
final Integer item = ThreadLocalRandom.current().nextInt();
@Test
public void construc() {
FrequencySketch<Integer> sketch = new FrequencySketch<>();
assertThat(sketch.table, is(nullValue()));
}
@Test(dataProvider = "sketch", expectedExceptions = IllegalArgumentException.class)
public void ensureCapacity_negative(FrequencySketch<Integer> sketch) {
sketch.ensureCapacity(-1);
}
@Test(dataProvider = "sketch")
public void ensureCapacity_smaller(FrequencySketch<Integer> sketch) {
int size = sketch.table.length;
sketch.ensureCapacity(size / 2);
assertThat(sketch.table.length, is(size));
assertThat(sketch.tableMask, is(size - 1));
assertThat(sketch.sampleSize, is(10 * size));
}
@Test(dataProvider = "sketch")
public void ensureCapacity_larger(FrequencySketch<Integer> sketch) {
int size = sketch.table.length;
sketch.ensureCapacity(size * 2);
assertThat(sketch.table.length, is(size * 2));
assertThat(sketch.tableMask, is(2 * size - 1));
assertThat(sketch.sampleSize, is(10 * 2 * size));
}
@Test(dataProvider = "sketch")
public void increment_once(FrequencySketch<Integer> sketch) {
sketch.increment(item);
assertThat(sketch.frequency(item), is(1));
}
@Test(dataProvider = "sketch")
public void increment_max(FrequencySketch<Integer> sketch) {
for (int i = 0; i < 20; i++) {
sketch.increment(item);
}
assertThat(sketch.frequency(item), is(15));
}
@Test(dataProvider = "sketch")
public void increment_distinct(FrequencySketch<Integer> sketch) {
sketch.increment(item);
sketch.increment(item + 1);
assertThat(sketch.frequency(item), is(1));
assertThat(sketch.frequency(item + 1), is(1));
assertThat(sketch.frequency(item + 2), is(0));
}
@Test
public void reset() {
boolean reset = false;
FrequencySketch<Integer> sketch = new FrequencySketch<>();
sketch.ensureCapacity(64);
for (int i = 1; i < 20 * sketch.table.length; i++) {
sketch.increment(i);
if (sketch.size != i) {
reset = true;
break;
}
}
assertThat(reset, is(true));
assertThat(sketch.size, lessThanOrEqualTo(sketch.sampleSize / 2));
}
@Test
public void heavyHitters() {
FrequencySketch<Double> sketch = makeSketch(512);
for (int i = 100; i < 100_000; i++) {
sketch.increment((double) i);
}
for (int i = 0; i < 10; i += 2) {
for (int j = 0; j < i; j++) {
sketch.increment((double) i);
}
}
// A perfect popularity count yields an array [0, 0, 2, 0, 4, 0, 6, 0, 8, 0]
int[] popularity = new int[10];
for (int i = 0; i < 10; i++) {
popularity[i] = sketch.frequency((double) i);
}
for (int i = 0; i < popularity.length; i++) {
if ((i == 0) || (i == 1) || (i == 3) || (i == 5) || (i == 7) || (i == 9)) {
assertThat(popularity[i], lessThanOrEqualTo(popularity[2]));
} else if (i == 2) {
assertThat(popularity[2], lessThanOrEqualTo(popularity[4]));
} else if (i == 4) {
assertThat(popularity[4], lessThanOrEqualTo(popularity[6]));
} else if (i == 6) {
assertThat(popularity[6], lessThanOrEqualTo(popularity[8]));
}
}
}
@DataProvider(name = "sketch")
public Object[][] providesSketch() {
return new Object[][] {{ makeSketch(512) }};
}
private static <E> FrequencySketch<E> makeSketch(long maximumSize) {
FrequencySketch<E> sketch = new FrequencySketch<>();
sketch.ensureCapacity(maximumSize);
ensureRandomSeed(sketch);
return sketch;
}
}