io.rsocket.resume.InMemoryResumableFramesStore Maven / Gradle / Ivy
/*
* Copyright 2015-2019 the original author or authors.
*
* 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 io.rsocket.resume;
import io.netty.buffer.ByteBuf;
import java.util.Queue;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.core.publisher.MonoProcessor;
import reactor.util.concurrent.Queues;
public class InMemoryResumableFramesStore implements ResumableFramesStore {
private static final Logger logger = LoggerFactory.getLogger(InMemoryResumableFramesStore.class);
private static final long SAVE_REQUEST_SIZE = Long.MAX_VALUE;
private final MonoProcessor disposed = MonoProcessor.create();
volatile long position;
volatile long impliedPosition;
volatile int cacheSize;
final Queue cachedFrames;
private final String tag;
private final int cacheLimit;
private volatile int upstreamFrameRefCnt;
public InMemoryResumableFramesStore(String tag, int cacheSizeBytes) {
this.tag = tag;
this.cacheLimit = cacheSizeBytes;
this.cachedFrames = cachedFramesQueue(cacheSizeBytes);
}
public Mono saveFrames(Flux frames) {
MonoProcessor completed = MonoProcessor.create();
frames
.doFinally(s -> completed.onComplete())
.subscribe(new FramesSubscriber(SAVE_REQUEST_SIZE));
return completed;
}
@Override
public void releaseFrames(long remoteImpliedPos) {
long pos = position;
logger.debug(
"{} Removing frames for local: {}, remote implied: {}", tag, pos, remoteImpliedPos);
long removeSize = Math.max(0, remoteImpliedPos - pos);
while (removeSize > 0) {
ByteBuf cachedFrame = cachedFrames.poll();
if (cachedFrame != null) {
removeSize -= releaseTailFrame(cachedFrame);
} else {
break;
}
}
if (removeSize > 0) {
throw new IllegalStateException(
String.format(
"Local and remote state disagreement: "
+ "need to remove additional %d bytes, but cache is empty",
removeSize));
} else if (removeSize < 0) {
throw new IllegalStateException(
"Local and remote state disagreement: " + "local and remote frame sizes are not equal");
} else {
logger.debug("{} Removed frames. Current cache size: {}", tag, cacheSize);
}
}
@Override
public Flux resumeStream() {
return Flux.generate(
() -> new ResumeStreamState(cachedFrames.size(), upstreamFrameRefCnt),
(state, sink) -> {
if (state.next()) {
/*spsc queue has no iterator - iterating by consuming*/
ByteBuf frame = cachedFrames.poll();
if (state.shouldRetain(frame)) {
frame.retain();
}
cachedFrames.offer(frame);
sink.next(frame);
} else {
sink.complete();
logger.debug("{} Resuming stream completed", tag);
}
return state;
});
}
@Override
public long framePosition() {
return position;
}
@Override
public long frameImpliedPosition() {
return impliedPosition;
}
@Override
public void resumableFrameReceived(ByteBuf frame) {
/*called on transport thread so non-atomic on volatile is safe*/
impliedPosition += frame.readableBytes();
}
@Override
public Mono onClose() {
return disposed;
}
@Override
public void dispose() {
cacheSize = 0;
ByteBuf frame = cachedFrames.poll();
while (frame != null) {
frame.release();
frame = cachedFrames.poll();
}
disposed.onComplete();
}
@Override
public boolean isDisposed() {
return disposed.isTerminated();
}
/* this method and saveFrame() won't be called concurrently,
* so non-atomic on volatile is safe*/
private int releaseTailFrame(ByteBuf content) {
int frameSize = content.readableBytes();
cacheSize -= frameSize;
position += frameSize;
content.release();
return frameSize;
}
/*this method and releaseTailFrame() won't be called concurrently,
* so non-atomic on volatile is safe*/
void saveFrame(ByteBuf frame) {
if (upstreamFrameRefCnt == 0) {
upstreamFrameRefCnt = frame.refCnt();
}
int frameSize = frame.readableBytes();
long availableSize = cacheLimit - cacheSize;
while (availableSize < frameSize) {
ByteBuf cachedFrame = cachedFrames.poll();
if (cachedFrame != null) {
availableSize += releaseTailFrame(cachedFrame);
} else {
break;
}
}
if (availableSize >= frameSize) {
cachedFrames.offer(frame.retain());
cacheSize += frameSize;
} else {
position += frameSize;
}
}
static class ResumeStreamState {
private final int cacheSize;
private final int expectedRefCnt;
private int cacheCounter;
public ResumeStreamState(int cacheSize, int expectedRefCnt) {
this.cacheSize = cacheSize;
this.expectedRefCnt = expectedRefCnt;
}
public boolean next() {
if (cacheCounter < cacheSize) {
cacheCounter++;
return true;
} else {
return false;
}
}
public boolean shouldRetain(ByteBuf frame) {
return frame.refCnt() == expectedRefCnt;
}
}
static Queue cachedFramesQueue(int size) {
return Queues.get(size).get();
}
class FramesSubscriber implements Subscriber {
private final long firstRequestSize;
private final long refillSize;
private int received;
private Subscription s;
public FramesSubscriber(long requestSize) {
this.firstRequestSize = requestSize;
this.refillSize = firstRequestSize / 2;
}
@Override
public void onSubscribe(Subscription s) {
this.s = s;
s.request(firstRequestSize);
}
@Override
public void onNext(ByteBuf byteBuf) {
saveFrame(byteBuf);
if (firstRequestSize != Long.MAX_VALUE && ++received == refillSize) {
received = 0;
s.request(refillSize);
}
}
@Override
public void onError(Throwable t) {
logger.info("unexpected onError signal: {}, {}", t.getClass(), t.getMessage());
}
@Override
public void onComplete() {}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy