io.rsocket.keepalive.KeepAliveConnection 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.keepalive;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufAllocator;
import io.rsocket.DuplexConnection;
import io.rsocket.exceptions.ConnectionErrorException;
import io.rsocket.frame.FrameHeaderFlyweight;
import io.rsocket.frame.FrameType;
import io.rsocket.internal.KeepAliveData;
import io.rsocket.resume.ResumePositionsConnection;
import io.rsocket.resume.ResumeStateHolder;
import io.rsocket.util.DuplexConnectionProxy;
import io.rsocket.util.Function3;
import java.time.Duration;
import java.util.function.Consumer;
import java.util.function.Function;
import javax.annotation.Nullable;
import org.reactivestreams.Publisher;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.core.publisher.MonoProcessor;
public class KeepAliveConnection extends DuplexConnectionProxy
implements ResumePositionsConnection {
private final MonoProcessor keepAliveHandlerReady = MonoProcessor.create();
private final ByteBufAllocator allocator;
private final Function keepAliveData;
private final Function3
keepAliveHandlerFactory;
private final Consumer errorConsumer;
private volatile KeepAliveHandler keepAliveHandler;
private volatile ResumeStateHolder resumeStateHolder;
private volatile boolean keepAliveHandlerStarted;
public static KeepAliveConnection ofClient(
ByteBufAllocator allocator,
DuplexConnection duplexConnection,
Function keepAliveData,
Consumer errorConsumer) {
return new KeepAliveConnection(
allocator, duplexConnection, keepAliveData, KeepAliveHandler::ofClient, errorConsumer);
}
public static KeepAliveConnection ofServer(
ByteBufAllocator allocator,
DuplexConnection duplexConnection,
Function keepAliveData,
Consumer errorConsumer) {
return new KeepAliveConnection(
allocator, duplexConnection, keepAliveData, KeepAliveHandler::ofServer, errorConsumer);
}
private KeepAliveConnection(
ByteBufAllocator allocator,
DuplexConnection duplexConnection,
Function keepAliveData,
Function3 keepAliveHandlerFactory,
Consumer errorConsumer) {
super(duplexConnection);
this.allocator = allocator;
this.keepAliveData = keepAliveData;
this.keepAliveHandlerFactory = keepAliveHandlerFactory;
this.errorConsumer = errorConsumer;
keepAliveHandlerReady.subscribe(this::startKeepAlives);
}
private void startKeepAlives(KeepAliveHandler keepAliveHandler) {
this.keepAliveHandler = keepAliveHandler;
send(keepAliveHandler.send()).subscribe(null, err -> keepAliveHandler.dispose());
keepAliveHandler
.timeout()
.subscribe(
keepAlive -> {
String message =
String.format("No keep-alive acks for %d ms", keepAlive.getTimeoutMillis());
ConnectionErrorException err = new ConnectionErrorException(message);
errorConsumer.accept(err);
dispose();
});
keepAliveHandler.start();
}
@Override
public Mono send(Publisher frames) {
return super.send(Flux.from(frames).doOnNext(this::startKeepAliveHandlerOnce));
}
@Override
public Flux receive() {
return super.receive()
.doOnNext(
f -> {
if (isKeepAliveFrame(f)) {
long receivedPos = keepAliveHandler.receive(f);
if (receivedPos > 0) {
ResumeStateHolder h = this.resumeStateHolder;
if (h != null) {
h.onImpliedPosition(receivedPos);
}
}
} else {
startKeepAliveHandlerOnce(f);
}
});
}
@Override
public Mono onClose() {
return super.onClose()
.doFinally(
s -> {
KeepAliveHandler keepAliveHandler = keepAliveHandlerReady.peek();
if (keepAliveHandler != null) {
keepAliveHandler.dispose();
}
});
}
@Override
public void acceptResumeState(ResumeStateHolder resumeStateHolder) {
this.resumeStateHolder = resumeStateHolder;
keepAliveHandlerReady.subscribe(h -> h.resumeState(resumeStateHolder));
}
private void startKeepAliveHandlerOnce(ByteBuf f) {
if (!keepAliveHandlerStarted && isStartFrame(f)) {
keepAliveHandlerStarted = true;
startKeepAliveHandler(keepAliveData.apply(f));
}
}
private static boolean isStartFrame(ByteBuf frame) {
FrameType frameType = FrameHeaderFlyweight.frameType(frame);
return frameType == FrameType.SETUP || frameType == FrameType.RESUME;
}
private static boolean isKeepAliveFrame(ByteBuf frame) {
return FrameHeaderFlyweight.frameType(frame) == FrameType.KEEPALIVE;
}
private void startKeepAliveHandler(@Nullable KeepAliveData kad) {
if (kad != null) {
KeepAliveHandler handler =
keepAliveHandlerFactory.apply(allocator, kad.getTickPeriod(), kad.getTimeout());
keepAliveHandlerReady.onNext(handler);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy