package org.infinispan.counter.impl.entries; import java.io.IOException; import java.io.ObjectInput; import java.io.ObjectOutput; import java.util.Collections; import java.util.Set; import org.infinispan.commons.marshall.AdvancedExternalizer; import org.infinispan.commons.marshall.MarshallUtil; import org.infinispan.counter.api.CounterConfiguration; import org.infinispan.counter.api.CounterState; import org.infinispan.counter.api.CounterType; import org.infinispan.counter.impl.externalizers.ExternalizerIds; import org.infinispan.counter.util.Utils; import jdk.nashorn.internal.ir.annotations.Immutable; /** * Stores the counter's value and {@link CounterState}. * <p> * If the counter isn't bounded, the state is always {@link CounterState#VALID}. * * @author Pedro Ruivo * @since 9.0 */ @Immutable public class CounterValue { public static final AdvancedExternalizer<CounterValue> EXTERNALIZER = new Externalizer(); //A valid zero value private static final CounterValue ZERO = new CounterValue(0, CounterState.VALID); private final long value; private final CounterState state; private CounterValue(long value, CounterState state) { this.value = value; this.state = state; } /** * Creates a new valid {@link CounterValue} with the value. * * @param value the counter's value. * @return the {@link CounterValue}. */ public static CounterValue newCounterValue(long value) { return value == 0 ? ZERO : new CounterValue(value, CounterState.VALID); } /** * Creates a new {@link CounterValue} with the value and state based on the boundaries. * * @param value the counter's value. * @param lowerBound the counter's lower bound. * @param upperBound the counter's upper bound. * @return the {@link CounterValue}. */ public static CounterValue newCounterValue(long value, long lowerBound, long upperBound) { return new CounterValue(value, Utils.calculateState(value, lowerBound, upperBound)); } /** * Creates a new {@link CounterValue} with the value and state. * * @param value the counter's value. * @param state the counter's state. * @return the {@link CounterValue}. */ public static CounterValue newCounterValue(long value, CounterState state) { return new CounterValue(value, state); } /** * Creates the initial {@link CounterValue} based on {@link CounterConfiguration}. * * @param configuration the configuration. * @return the {@link CounterValue}. */ public static CounterValue newCounterValue(CounterConfiguration configuration) { return configuration.type() == CounterType.BOUNDED_STRONG ? newCounterValue(configuration.initialValue(), configuration.lowerBound(), configuration.upperBound()) : newCounterValue(configuration.initialValue()); } /** * @return the counter's value. */ public long getValue() { return value; } /** * @return the counter's state. */ public CounterState getState() { return state; } @Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } CounterValue that = (CounterValue) o; return value == that.value && state == that.state; } @Override public int hashCode() { int result = (int) (value ^ (value >>> 32)); result = 31 * result + state.hashCode(); return result; } @Override public String toString() { return "CounterValue{" + "value=" + value + ", state=" + state + '}'; } private static class Externalizer implements AdvancedExternalizer<CounterValue> { @Override public Set<Class<? extends CounterValue>> getTypeClasses() { return Collections.singleton(CounterValue.class); } @Override public Integer getId() { return ExternalizerIds.COUNTER_VALUE; } @Override public void writeObject(ObjectOutput output, CounterValue object) throws IOException { output.writeLong(object.value); MarshallUtil.marshallEnum(object.state, output); } @Override public CounterValue readObject(ObjectInput input) throws IOException, ClassNotFoundException { long value = input.readLong(); CounterState state = MarshallUtil.unmarshallEnum(input, CounterState::valueOf); if (value == 0 && state == CounterState.VALID) { return ZERO; } return new CounterValue(value, state); } } }