io.rsocket.loadbalance.FluxDeferredResolution Maven / Gradle / Ivy
/*
* Copyright 2015-2020 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.loadbalance;
import io.netty.util.ReferenceCountUtil;
import io.rsocket.Payload;
import io.rsocket.frame.FrameType;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
import java.util.function.BiConsumer;
import org.reactivestreams.Subscription;
import reactor.core.CoreSubscriber;
import reactor.core.Scannable;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Operators;
import reactor.util.annotation.Nullable;
import reactor.util.context.Context;
abstract class FluxDeferredResolution extends Flux
implements CoreSubscriber, Subscription, BiConsumer, Scannable {
final ResolvingOperator parent;
final INPUT fluxOrPayload;
final FrameType requestType;
volatile long requested;
@SuppressWarnings("rawtypes")
static final AtomicLongFieldUpdater REQUESTED =
AtomicLongFieldUpdater.newUpdater(FluxDeferredResolution.class, "requested");
static final long STATE_UNSUBSCRIBED = -1;
static final long STATE_SUBSCRIBER_SET = 0;
static final long STATE_SUBSCRIBED = -2;
static final long STATE_TERMINATED = Long.MIN_VALUE;
Subscription s;
CoreSubscriber super Payload> actual;
boolean done;
FluxDeferredResolution(ResolvingOperator parent, INPUT fluxOrPayload, FrameType requestType) {
this.parent = parent;
this.fluxOrPayload = fluxOrPayload;
this.requestType = requestType;
REQUESTED.lazySet(this, STATE_UNSUBSCRIBED);
}
@Override
public final void subscribe(CoreSubscriber super Payload> actual) {
if (this.requested == STATE_UNSUBSCRIBED
&& REQUESTED.compareAndSet(this, STATE_UNSUBSCRIBED, STATE_SUBSCRIBER_SET)) {
actual.onSubscribe(this);
if (this.requested == STATE_TERMINATED) {
return;
}
this.actual = actual;
this.parent.observe(this);
} else {
Operators.error(actual, new IllegalStateException("Only a single Subscriber allowed"));
}
}
@Override
public final Context currentContext() {
return this.actual.currentContext();
}
@Nullable
@Override
public final Object scanUnsafe(Attr key) {
long state = this.requested;
if (key == Attr.PARENT) {
return this.s;
}
if (key == Attr.ACTUAL) {
return this.parent;
}
if (key == Attr.TERMINATED) {
return this.done;
}
if (key == Attr.CANCELLED) {
return state == STATE_TERMINATED;
}
return null;
}
@Override
public final void onSubscribe(Subscription s) {
final long state = this.requested;
Subscription a = this.s;
if (state == STATE_TERMINATED) {
s.cancel();
return;
}
if (a != null) {
s.cancel();
return;
}
long r;
long accumulated = 0;
for (; ; ) {
r = this.requested;
if (r == STATE_TERMINATED || r == STATE_SUBSCRIBED) {
s.cancel();
return;
}
this.s = s;
long toRequest = r - accumulated;
if (toRequest > 0) { // if there is something,
s.request(toRequest); // then we do a request on the given subscription
}
accumulated = r;
if (REQUESTED.compareAndSet(this, r, STATE_SUBSCRIBED)) {
return;
}
}
}
@Override
public final void onNext(Payload payload) {
this.actual.onNext(payload);
}
@Override
public final void onError(Throwable t) {
if (this.done) {
Operators.onErrorDropped(t, this.actual.currentContext());
return;
}
this.done = true;
this.actual.onError(t);
}
@Override
public final void onComplete() {
if (this.done) {
return;
}
this.done = true;
this.actual.onComplete();
}
@Override
public final void request(long n) {
if (Operators.validate(n)) {
long r = this.requested; // volatile read beforehand
if (r > STATE_SUBSCRIBED) { // works only in case onSubscribe has not happened
long u;
for (; ; ) { // normal CAS loop with overflow protection
if (r == Long.MAX_VALUE) {
// if r == Long.MAX_VALUE then we dont care and we can loose this
// request just in case of racing
return;
}
u = Operators.addCap(r, n);
if (REQUESTED.compareAndSet(this, r, u)) {
// Means increment happened before onSubscribe
return;
} else {
// Means increment happened after onSubscribe
// update new state to see what exactly happened (onSubscribe |cancel | requestN)
r = this.requested;
// check state (expect -1 | -2 to exit, otherwise repeat)
if (r < 0) {
break;
}
}
}
}
if (r == STATE_TERMINATED) { // if canceled, just exit
return;
}
// if onSubscribe -> subscription exists (and we sure of that because volatile read
// after volatile write) so we can execute requestN on the subscription
this.s.request(n);
}
}
public final void cancel() {
long state = REQUESTED.getAndSet(this, STATE_TERMINATED);
if (state == STATE_TERMINATED) {
return;
}
if (state == STATE_SUBSCRIBED) {
this.s.cancel();
} else {
this.parent.remove(this);
if (requestType == FrameType.REQUEST_STREAM) {
ReferenceCountUtil.safeRelease(this.fluxOrPayload);
}
}
}
boolean isTerminated() {
return this.requested == STATE_TERMINATED;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy