package org.infinispan.container.entries; import static java.lang.Math.min; import java.io.IOException; import java.io.ObjectInput; import java.io.ObjectOutput; import java.util.Set; import org.infinispan.commons.io.UnsignedNumeric; import org.infinispan.commons.marshall.AbstractExternalizer; import org.infinispan.commons.util.Util; import org.infinispan.marshall.core.Ids; /** * A transient, mortal cache value to correspond with {@link org.infinispan.container.entries.TransientMortalCacheEntry} * * @author Manik Surtani * @since 4.0 */ public class TransientMortalCacheValue extends MortalCacheValue { protected long maxIdle = -1; protected long lastUsed; public TransientMortalCacheValue(Object value, long created, long lifespan, long maxIdle, long lastUsed) { this(value, created, lifespan, maxIdle); this.lastUsed = lastUsed; } public TransientMortalCacheValue(Object value, long created, long lifespan, long maxIdle) { super(value, created, lifespan); this.maxIdle = maxIdle; } @Override public long getMaxIdle() { return maxIdle; } public void setMaxIdle(long maxIdle) { this.maxIdle = maxIdle; } @Override public long getLastUsed() { return lastUsed; } public void setLastUsed(long lastUsed) { this.lastUsed = lastUsed; } @Override public boolean isExpired(long now) { return ExpiryHelper.isExpiredTransientMortal(maxIdle, lastUsed, lifespan, created, now); } @Override public boolean isExpired() { return isExpired(System.currentTimeMillis()); } @Override public InternalCacheEntry toInternalCacheEntry(Object key) { return new TransientMortalCacheEntry(key, value, maxIdle, lifespan, lastUsed, created); } @Override public long getExpiryTime() { long lset = lifespan > -1 ? created + lifespan : -1; long muet = maxIdle > -1 ? lastUsed + maxIdle : -1; if (lset == -1) return muet; if (muet == -1) return lset; return min(lset, muet); } @Override public boolean equals(Object o) { if (this == o) return true; if (!(o instanceof TransientMortalCacheValue)) return false; if (!super.equals(o)) return false; TransientMortalCacheValue that = (TransientMortalCacheValue) o; if (lastUsed != that.lastUsed) return false; if (maxIdle != that.maxIdle) return false; return true; } @Override public int hashCode() { int result = super.hashCode(); result = 31 * result + (int) (maxIdle ^ (maxIdle >>> 32)); result = 31 * result + (int) (lastUsed ^ (lastUsed >>> 32)); return result; } @Override public String toString() { return "TransientMortalCacheValue{" + "maxIdle=" + maxIdle + ", lastUsed=" + lastUsed + "} " + super.toString(); } @Override public TransientMortalCacheValue clone() { return (TransientMortalCacheValue) super.clone(); } public static class Externalizer extends AbstractExternalizer<TransientMortalCacheValue> { @Override public void writeObject(ObjectOutput output, TransientMortalCacheValue value) throws IOException { output.writeObject(value.value); UnsignedNumeric.writeUnsignedLong(output, value.created); output.writeLong(value.lifespan); // could be negative so should not use unsigned longs UnsignedNumeric.writeUnsignedLong(output, value.lastUsed); output.writeLong(value.maxIdle); // could be negative so should not use unsigned longs } @Override public TransientMortalCacheValue readObject(ObjectInput input) throws IOException, ClassNotFoundException { Object v = input.readObject(); long created = UnsignedNumeric.readUnsignedLong(input); Long lifespan = input.readLong(); long lastUsed = UnsignedNumeric.readUnsignedLong(input); Long maxIdle = input.readLong(); return new TransientMortalCacheValue(v, created, lifespan, maxIdle, lastUsed); } @Override public Integer getId() { return Ids.TRANSIENT_MORTAL_VALUE; } @Override public Set<Class<? extends TransientMortalCacheValue>> getTypeClasses() { return Util.<Class<? extends TransientMortalCacheValue>>asSet(TransientMortalCacheValue.class); } } }