io.rsocket.keepalive.KeepAliveSupport Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rsocket-core Show documentation
Show all versions of rsocket-core Show documentation
Core functionality for the RSocket library
The newest version!
/*
* 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.KeepAliveFrameCodec;
import io.rsocket.resume.ResumeStateHolder;
import java.time.Duration;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.function.Consumer;
import reactor.core.Disposable;
import reactor.core.publisher.Flux;
import reactor.core.scheduler.Scheduler;
import reactor.core.scheduler.Schedulers;
public abstract class KeepAliveSupport implements KeepAliveFramesAcceptor {
final ByteBufAllocator allocator;
final Scheduler scheduler;
final Duration keepAliveInterval;
final Duration keepAliveTimeout;
final long keepAliveTimeoutMillis;
volatile int state;
static final AtomicIntegerFieldUpdater STATE =
AtomicIntegerFieldUpdater.newUpdater(KeepAliveSupport.class, "state");
static final int STOPPED_STATE = 0;
static final int STARTING_STATE = 1;
static final int STARTED_STATE = 2;
static final int DISPOSED_STATE = -1;
volatile Consumer onTimeout;
volatile Consumer onFrameSent;
Disposable ticksDisposable;
volatile ResumeStateHolder resumeStateHolder;
volatile long lastReceivedMillis;
private KeepAliveSupport(
ByteBufAllocator allocator, int keepAliveInterval, int keepAliveTimeout) {
this.allocator = allocator;
this.scheduler = Schedulers.parallel();
this.keepAliveInterval = Duration.ofMillis(keepAliveInterval);
this.keepAliveTimeout = Duration.ofMillis(keepAliveTimeout);
this.keepAliveTimeoutMillis = keepAliveTimeout;
}
public KeepAliveSupport start() {
if (this.state == STOPPED_STATE && STATE.compareAndSet(this, STOPPED_STATE, STARTING_STATE)) {
this.lastReceivedMillis = scheduler.now(TimeUnit.MILLISECONDS);
final Disposable disposable =
Flux.interval(keepAliveInterval, scheduler).subscribe(v -> onIntervalTick());
this.ticksDisposable = disposable;
if (this.state != STARTING_STATE
|| !STATE.compareAndSet(this, STARTING_STATE, STARTED_STATE)) {
disposable.dispose();
}
}
return this;
}
public void stop() {
terminate(STOPPED_STATE);
}
@Override
public void receive(ByteBuf keepAliveFrame) {
this.lastReceivedMillis = scheduler.now(TimeUnit.MILLISECONDS);
if (resumeStateHolder != null) {
final long remoteLastReceivedPos = KeepAliveFrameCodec.lastPosition(keepAliveFrame);
resumeStateHolder.onImpliedPosition(remoteLastReceivedPos);
}
if (KeepAliveFrameCodec.respondFlag(keepAliveFrame)) {
long localLastReceivedPos = localLastReceivedPosition();
send(
KeepAliveFrameCodec.encode(
allocator,
false,
localLastReceivedPos,
KeepAliveFrameCodec.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;
}
@Override
public void dispose() {
terminate(DISPOSED_STATE);
}
@Override
public boolean isDisposed() {
return ticksDisposable.isDisposed();
}
abstract void onIntervalTick();
void send(ByteBuf frame) {
if (onFrameSent != null) {
onFrameSent.accept(frame);
}
}
void tryTimeout() {
long now = scheduler.now(TimeUnit.MILLISECONDS);
if (now - lastReceivedMillis >= keepAliveTimeoutMillis) {
if (onTimeout != null) {
onTimeout.accept(new KeepAlive(keepAliveInterval, keepAliveTimeout));
}
stop();
}
}
void terminate(int terminationState) {
for (; ; ) {
final int state = this.state;
if (state == STOPPED_STATE || state == DISPOSED_STATE) {
return;
}
final Disposable disposable = this.ticksDisposable;
if (STATE.compareAndSet(this, state, terminationState)) {
disposable.dispose();
return;
}
}
}
long localLastReceivedPosition() {
return resumeStateHolder != null ? resumeStateHolder.impliedPosition() : 0;
}
public static final class ClientKeepAliveSupport extends KeepAliveSupport {
public ClientKeepAliveSupport(
ByteBufAllocator allocator, int keepAliveInterval, int keepAliveTimeout) {
super(allocator, keepAliveInterval, keepAliveTimeout);
}
@Override
void onIntervalTick() {
tryTimeout();
send(
KeepAliveFrameCodec.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