package reactor.core.publisher;
import io.micrometer.core.instrument.Clock;
import io.micrometer.core.instrument.MeterRegistry;
import io.micrometer.core.instrument.Tags;
import io.micrometer.core.instrument.Timer;
import org.reactivestreams.Subscription;
import reactor.core.CoreSubscriber;
import reactor.core.Fuseable;
import reactor.util.Metrics;
import reactor.util.annotation.Nullable;
import static reactor.core.publisher.FluxMetrics.resolveName;
import static reactor.core.publisher.FluxMetrics.resolveTags;
final class MonoMetricsFuseable<T> extends InternalMonoOperator<T, T> implements Fuseable {
final String name;
final Tags tags;
final MeterRegistry registryCandidate;
MonoMetricsFuseable(Mono<? extends T> mono) {
super(mono);
this.name = resolveName(mono);
this.tags = resolveTags(mono, FluxMetrics.DEFAULT_TAGS_MONO);
this.registryCandidate = Metrics.MicrometerConfiguration.getRegistry();;
}
@Override
public CoreSubscriber<? super T> subscribeOrReturn(CoreSubscriber<? super T> actual) {
return new MetricsFuseableSubscriber<>(actual, registryCandidate, Clock.SYSTEM, this.name, this.tags);
}
@Override
public Object scanUnsafe(Attr key) {
if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC;
return super.scanUnsafe(key);
}
static final class MetricsFuseableSubscriber<T> extends MonoMetrics.MetricsSubscriber<T>
implements Fuseable, QueueSubscription<T> {
int mode;
@Nullable
Fuseable.QueueSubscription<T> qs;
MetricsFuseableSubscriber(CoreSubscriber<? super T> actual,
MeterRegistry registry,
Clock clock,
String sequenceName,
Tags sequenceTags) {
super(actual, registry, clock, sequenceName, sequenceTags);
}
@Override
public void clear() {
if (qs != null) {
qs.clear();
}
}
@Override
public void onComplete() {
if (mode == ASYNC) {
if (!done) {
FluxMetrics.recordOnCompleteEmpty(sequenceName, commonTags, registry, subscribeToTerminateSample);
}
actual.onComplete();
}
else {
if (done) {
return;
}
done = true;
FluxMetrics.recordOnCompleteEmpty(sequenceName, commonTags, registry, subscribeToTerminateSample);
actual.onComplete();
}
}
@Override
public void onNext(T t) {
if (mode == ASYNC) {
actual.onNext(null);
}
else {
if (done) {
FluxMetrics.recordMalformed(sequenceName, commonTags, registry);
Operators.onNextDropped(t, actual.currentContext());
return;
}
done = true;
FluxMetrics.recordOnComplete(sequenceName,
commonTags,
registry,
subscribeToTerminateSample);
actual.onNext(t);
actual.onComplete();
}
}
@Override
public boolean isEmpty() {
return qs == null || qs.isEmpty();
}
@Override
public void onSubscribe(Subscription s) {
if (Operators.validate(this.s, s)) {
FluxMetrics.recordOnSubscribe(sequenceName, commonTags, registry);
this.subscribeToTerminateSample = Timer.start(clock);
this.qs = Operators.as(s);
this.s = s;
actual.onSubscribe(this);
}
}
@Override
@Nullable
public T poll() {
if (qs == null) {
return null;
}
try {
T v = qs.poll();
if (!done) {
if (v == null && mode == SYNC) {
FluxMetrics.recordOnCompleteEmpty(sequenceName, commonTags, registry, subscribeToTerminateSample);
}
else if (v != null) {
FluxMetrics.recordOnComplete(sequenceName, commonTags, registry, subscribeToTerminateSample);
}
}
done = true;
return v;
}
catch (Throwable e) {
FluxMetrics.recordOnError(sequenceName, commonTags, registry, subscribeToTerminateSample, e);
throw e;
}
}
@Override
public int requestFusion(int mode) {
if (qs != null) {
this.mode = qs.requestFusion(mode);
return this.mode;
}
return Fuseable.NONE;
}
@Override
public int size() {
return qs == null ? 0 : qs.size();
}
@Override
public Object scanUnsafe(Attr key) {
if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC;
return super.scanUnsafe(key);
}
}
}