io.rsocket.keepalive.KeepAliveSupport 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.netty.buffer.Unpooled;
import io.rsocket.frame.KeepAliveFrameFlyweight;
import io.rsocket.resume.ResumeStateHolder;
import java.time.Duration;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.Consumer;
import reactor.core.Disposable;
import reactor.core.publisher.Flux;
public abstract class KeepAliveSupport implements KeepAliveFramesAcceptor {
final ByteBufAllocator allocator;
private final Duration keepAliveInterval;
private final Duration keepAliveTimeout;
private final long keepAliveTimeoutMillis;
private volatile Consumer onTimeout;
private volatile Consumer onFrameSent;
private volatile Disposable ticksDisposable;
private final AtomicBoolean started = new AtomicBoolean();
private volatile ResumeStateHolder resumeStateHolder;
private volatile long lastReceivedMillis;
private KeepAliveSupport(
ByteBufAllocator allocator, int keepAliveInterval, int keepAliveTimeout) {
this.allocator = allocator;
this.keepAliveInterval = Duration.ofMillis(keepAliveInterval);
this.keepAliveTimeout = Duration.ofMillis(keepAliveTimeout);
this.keepAliveTimeoutMillis = keepAliveTimeout;
}
public KeepAliveSupport start() {
this.lastReceivedMillis = System.currentTimeMillis();
if (started.compareAndSet(false, true)) {
ticksDisposable = Flux.interval(keepAliveInterval).subscribe(v -> onIntervalTick());
}
return this;
}
public void stop() {
if (started.compareAndSet(true, false)) {
ticksDisposable.dispose();
}
}
@Override
public void receive(ByteBuf keepAliveFrame) {
this.lastReceivedMillis = System.currentTimeMillis();
if (resumeStateHolder != null) {
long remoteLastReceivedPos = remoteLastReceivedPosition(keepAliveFrame);
resumeStateHolder.onImpliedPosition(remoteLastReceivedPos);
}
if (KeepAliveFrameFlyweight.respondFlag(keepAliveFrame)) {
long localLastReceivedPos = localLastReceivedPosition();
send(
KeepAliveFrameFlyweight.encode(
allocator,
false,
localLastReceivedPos,
KeepAliveFrameFlyweight.data(keepAliveFrame).retain()));
}
}
public KeepAliveSupport resumeState(ResumeStateHolder resumeStateHolder) {
this.resumeStateHolder = resumeStateHolder;
return this;
}
public KeepAliveSupport onSendKeepAliveFrame(Consumer onFrameSent) {
this.onFrameSent = onFrameSent;
return this;
}
public KeepAliveSupport onTimeout(Consumer onTimeout) {
this.onTimeout = onTimeout;
return this;
}
abstract void onIntervalTick();
void send(ByteBuf frame) {
if (onFrameSent != null) {
onFrameSent.accept(frame);
}
}
void tryTimeout() {
long now = System.currentTimeMillis();
if (now - lastReceivedMillis >= keepAliveTimeoutMillis) {
if (onTimeout != null) {
onTimeout.accept(new KeepAlive(keepAliveInterval, keepAliveTimeout));
}
stop();
}
}
long localLastReceivedPosition() {
return resumeStateHolder != null ? resumeStateHolder.impliedPosition() : 0;
}
long remoteLastReceivedPosition(ByteBuf keepAliveFrame) {
return KeepAliveFrameFlyweight.lastPosition(keepAliveFrame);
}
public static final class ServerKeepAliveSupport extends KeepAliveSupport {
public ServerKeepAliveSupport(
ByteBufAllocator allocator, int keepAlivePeriod, int keepAliveTimeout) {
super(allocator, keepAlivePeriod, keepAliveTimeout);
}
@Override
void onIntervalTick() {
tryTimeout();
}
}
public static final class ClientKeepAliveSupport extends KeepAliveSupport {
public ClientKeepAliveSupport(
ByteBufAllocator allocator, int keepAliveInterval, int keepAliveTimeout) {
super(allocator, keepAliveInterval, keepAliveTimeout);
}
@Override
void onIntervalTick() {
tryTimeout();
send(
KeepAliveFrameFlyweight.encode(
allocator, true, localLastReceivedPosition(), Unpooled.EMPTY_BUFFER));
}
}
public static final class KeepAlive {
private final Duration tickPeriod;
private final Duration timeoutMillis;
public KeepAlive(Duration tickPeriod, Duration timeoutMillis) {
this.tickPeriod = tickPeriod;
this.timeoutMillis = timeoutMillis;
}
public Duration getTickPeriod() {
return tickPeriod;
}
public Duration getTimeout() {
return timeoutMillis;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy