All Downloads are FREE. Search and download functionalities are using the official Maven repository.

reactor.core.publisher.FluxOnAssembly Maven / Gradle / Ivy

Go to download

Easy Redis Java client and Real-Time Data Platform. Valkey compatible. Sync/Async/RxJava3/Reactive API. Client side caching. Over 50 Redis based Java objects and services: JCache API, Apache Tomcat, Hibernate, Spring, Set, Multimap, SortedSet, Map, List, Queue, Deque, Semaphore, Lock, AtomicLong, Map Reduce, Bloom filter, Scheduler, RPC

There is a newer version: 3.43.0
Show newest version
/*
 * Copyright (c) 2011-2017 Pivotal Software Inc, All Rights Reserved.
 *
 * 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
 *
 *       https://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 reactor.core.publisher;

import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;
import java.util.function.Supplier;

import org.reactivestreams.Publisher;
import org.reactivestreams.Subscription;
import reactor.core.CoreSubscriber;
import reactor.core.Exceptions;
import reactor.core.Fuseable;
import reactor.core.Scannable;
import reactor.util.annotation.Nullable;
import reactor.util.function.Tuple4;
import reactor.util.function.Tuples;

/**
 * Captures the current stacktrace when this publisher is created and
 * makes it available/visible for debugging purposes from
 * the inner Subscriber.
 * 

* Note that getting a stacktrace is a costly operation. *

* The operator sanitizes the stacktrace and removes noisy entries such as: *

    *
  • java.lang.Thread entries
  • *
  • method references with source line of 1 (bridge methods)
  • *
  • Tomcat worker thread entries
  • *
  • JUnit setup
  • *
* * @param the value type passing through * @see https://github.com/reactor/reactive-streams-commons */ final class FluxOnAssembly extends InternalFluxOperator implements Fuseable, AssemblyOp { final AssemblySnapshot snapshotStack; /** * Create an assembly trace decorated as a {@link Flux}. */ FluxOnAssembly(Flux source, AssemblySnapshot snapshotStack) { super(source); this.snapshotStack = snapshotStack; } @Override public String stepName() { return snapshotStack.operatorAssemblyInformation(); } @Override public Object scanUnsafe(Attr key) { if (key == Attr.ACTUAL_METADATA) return !snapshotStack.checkpointed; if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC; return super.scanUnsafe(key); } @Override public String toString() { return snapshotStack.operatorAssemblyInformation(); } static void fillStacktraceHeader(StringBuilder sb, Class sourceClass, @Nullable String description) { sb.append("\nAssembly trace from producer [") .append(sourceClass.getName()) .append("]"); if (description != null) { sb.append(", described as [") .append(description) .append("]"); } sb.append(" :\n"); } @SuppressWarnings("unchecked") static CoreSubscriber wrapSubscriber(CoreSubscriber actual, Flux source, @Nullable AssemblySnapshot snapshotStack) { if(snapshotStack != null) { if (actual instanceof ConditionalSubscriber) { ConditionalSubscriber cs = (ConditionalSubscriber) actual; return new OnAssemblyConditionalSubscriber<>(cs, snapshotStack, source); } else { return new OnAssemblySubscriber<>(actual, snapshotStack, source); } } else { return actual; } } @Override @SuppressWarnings("unchecked") public CoreSubscriber subscribeOrReturn(CoreSubscriber actual) { return wrapSubscriber(actual, source, snapshotStack); } /** * A snapshot of current assembly traceback, possibly with a user-readable * description or a wider correlation ID. */ static class AssemblySnapshot { final boolean checkpointed; @Nullable final String description; final Supplier assemblyInformationSupplier; String cached; /** * @param description a description for the assembly traceback. * @param assemblyInformationSupplier a callSite supplier. */ AssemblySnapshot(@Nullable String description, Supplier assemblyInformationSupplier) { this(description != null, description, assemblyInformationSupplier); } AssemblySnapshot(String assemblyInformation) { this.checkpointed = false; this.description = null; this.assemblyInformationSupplier = null; this.cached = assemblyInformation; } private AssemblySnapshot(boolean checkpointed, @Nullable String description, Supplier assemblyInformationSupplier) { this.checkpointed = checkpointed; this.description = description; this.assemblyInformationSupplier = assemblyInformationSupplier; } @Nullable public String getDescription() { return description; } public boolean isLight() { return false; } public String lightPrefix() { return ""; } String toAssemblyInformation() { if(cached == null) { cached = assemblyInformationSupplier.get(); } return cached; } String operatorAssemblyInformation() { return Traces.extractOperatorAssemblyInformation(toAssemblyInformation()); } } static final class AssemblyLightSnapshot extends AssemblySnapshot { AssemblyLightSnapshot(@Nullable String description) { super(true, description, null); cached = "checkpoint(\"" + description + "\")"; } @Override public boolean isLight() { return true; } @Override public String lightPrefix() { return "checkpoint"; } @Override String operatorAssemblyInformation() { return cached; } } static final class MethodReturnSnapshot extends AssemblySnapshot { MethodReturnSnapshot(String method) { super(true, method, null); cached = method; } @Override public boolean isLight() { return true; } @Override String operatorAssemblyInformation() { return cached; } } /** * The holder for the assembly stacktrace (as its message). * * @implNote this package-private exception needs access to package-private enclosing class and methods, * but it is also detected by {@link Exceptions#isTraceback(Throwable)} via {@link Class#getCanonicalName() reflection}. * Be sure to update said method in case of a refactoring. * * @see Exceptions#isTraceback(Throwable) */ static final class OnAssemblyException extends RuntimeException { final List> chainOrder = new LinkedList<>(); /** */ private static final long serialVersionUID = 5278398300974016773L; OnAssemblyException(String message) { super(message); } @Override public Throwable fillInStackTrace() { return this; } void add(Publisher parent, AssemblySnapshot snapshot) { if (snapshot.isLight()) { add(parent, snapshot.lightPrefix(), snapshot.getDescription()); } else { String assemblyInformation = snapshot.toAssemblyInformation(); String[] parts = Traces.extractOperatorAssemblyInformationParts(assemblyInformation); if (parts.length > 0) { String prefix = parts.length > 1 ? parts[0] : ""; String line = parts[parts.length - 1]; add(parent, prefix, line); } } } private void add(Publisher parent, String prefix, String line) { //noinspection ConstantConditions int key = getParentOrThis(Scannable.from(parent)); synchronized (chainOrder) { int i = 0; int n = chainOrder.size(); int j = n - 1; Tuple4 tmp; while(j >= 0){ tmp = chainOrder.get(j); //noinspection ConstantConditions if(tmp.getT1() == key){ //noinspection ConstantConditions i = tmp.getT4(); break; } j--; } for(;;){ Tuple4 t = Tuples.of(parent.hashCode(), prefix, line, i); if(!chainOrder.contains(t)){ chainOrder.add(t); break; } i++; } } } @Override public String getMessage() { //skip the "error has been observed" traceback if mapped traceback is empty synchronized (chainOrder) { if (chainOrder.isEmpty()) { return super.getMessage(); } int maxWidth = 0; for (Tuple4 t : chainOrder) { int length = t.getT2().length(); if (length > maxWidth) { maxWidth = length; } } StringBuilder sb = new StringBuilder(super.getMessage()) .append("\nError has been observed at the following site(s):\n"); for(Tuple4 t : chainOrder) { Integer indent = t.getT4(); String operator = t.getT2(); String message = t.getT3(); sb.append("\t|_"); for (int i = 0; i < indent; i++) { sb.append("____"); } for (int i = operator.length(); i < maxWidth + 1; i++) { sb.append(' '); } sb.append(operator); sb.append(Traces.CALL_SITE_GLUE); sb.append(message); sb.append("\n"); } sb.append("Stack trace:"); return sb.toString(); } } } static int getParentOrThis(Scannable parent) { return parent.parents() .filter(s -> !(s instanceof AssemblyOp)) .findFirst() .map(Object::hashCode) .orElse(parent.hashCode()); } static class OnAssemblySubscriber implements InnerOperator, QueueSubscription { final AssemblySnapshot snapshotStack; final Publisher parent; final CoreSubscriber actual; QueueSubscription qs; Subscription s; int fusionMode; OnAssemblySubscriber(CoreSubscriber actual, AssemblySnapshot snapshotStack, Publisher parent) { this.actual = actual; this.snapshotStack = snapshotStack; this.parent = parent; } @Override public final CoreSubscriber actual() { return actual; } @Override @Nullable public Object scanUnsafe(Attr key) { if (key == Attr.PARENT) return s; if (key == Attr.ACTUAL_METADATA) return !snapshotStack.checkpointed; if (key == Attr.RUN_STYLE) return Attr.RunStyle.SYNC; return InnerOperator.super.scanUnsafe(key); } @Override public String toString() { return snapshotStack.operatorAssemblyInformation(); } @Override public String stepName() { return toString(); } @Override final public void onNext(T t) { actual.onNext(t); } @Override final public void onError(Throwable t) { actual.onError(fail(t)); } @Override final public void onComplete() { actual.onComplete(); } @Override final public int requestFusion(int requestedMode) { QueueSubscription qs = this.qs; if (qs != null) { int m = qs.requestFusion(requestedMode); if (m != Fuseable.NONE) { fusionMode = m; } return m; } return Fuseable.NONE; } final Throwable fail(Throwable t) { boolean lightCheckpoint = snapshotStack.isLight(); OnAssemblyException onAssemblyException = null; for (Throwable e : t.getSuppressed()) { if (e instanceof OnAssemblyException) { onAssemblyException = (OnAssemblyException) e; break; } } if (onAssemblyException == null) { if (lightCheckpoint) { onAssemblyException = new OnAssemblyException(""); } else { StringBuilder sb = new StringBuilder(); fillStacktraceHeader(sb, parent.getClass(), snapshotStack.getDescription()); sb.append(snapshotStack.toAssemblyInformation().replaceFirst("\\n$", "")); String description = sb.toString(); onAssemblyException = new OnAssemblyException(description); } t = Exceptions.addSuppressed(t, onAssemblyException); final StackTraceElement[] stackTrace = t.getStackTrace(); if (stackTrace.length > 0) { StackTraceElement[] newStackTrace = new StackTraceElement[stackTrace.length]; int i = 0; for (StackTraceElement stackTraceElement : stackTrace) { String className = stackTraceElement.getClassName(); if (className.startsWith("reactor.core.publisher.") && className.contains("OnAssembly")) { continue; } newStackTrace[i] = stackTraceElement; i++; } newStackTrace = Arrays.copyOf(newStackTrace, i); onAssemblyException.setStackTrace(newStackTrace); t.setStackTrace(new StackTraceElement[] { stackTrace[0] }); } } onAssemblyException.add(parent, snapshotStack); return t; } @Override final public boolean isEmpty() { try { return qs.isEmpty(); } catch (Throwable ex) { Exceptions.throwIfFatal(ex); throw Exceptions.propagate(fail(ex)); } } @Override final public void onSubscribe(Subscription s) { if (Operators.validate(this.s, s)) { this.s = s; this.qs = Operators.as(s); actual.onSubscribe(this); } } @Override final public int size() { return qs.size(); } @Override final public void clear() { qs.clear(); } @Override final public void request(long n) { s.request(n); } @Override final public void cancel() { s.cancel(); } @Override @Nullable final public T poll() { try { return qs.poll(); } catch (final Throwable ex) { Exceptions.throwIfFatal(ex); throw Exceptions.propagate(fail(ex)); } } } static final class OnAssemblyConditionalSubscriber extends OnAssemblySubscriber implements ConditionalSubscriber { final ConditionalSubscriber actualCS; OnAssemblyConditionalSubscriber(ConditionalSubscriber actual, AssemblySnapshot stacktrace, Publisher parent) { super(actual, stacktrace, parent); this.actualCS = actual; } @Override public boolean tryOnNext(T t) { return actualCS.tryOnNext(t); } } } interface AssemblyOp {}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy