ratpack.exec.internal.CachingUpstream Maven / Gradle / Ivy
/*
* Copyright 2014 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 ratpack.exec.internal;
import ratpack.exec.*;
import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference;
public class CachingUpstream implements Upstream {
private Upstream extends T> upstream;
private final AtomicBoolean fired = new AtomicBoolean();
private final Queue>> waiting = new ConcurrentLinkedQueue<>();
private final AtomicBoolean draining = new AtomicBoolean();
private final AtomicReference> result = new AtomicReference<>();
public CachingUpstream(Upstream extends T> upstream) {
this.upstream = upstream;
}
private void tryDrain() {
if (draining.compareAndSet(false, true)) {
try {
ExecResult extends T> result = this.result.get();
Downstream super ExecResult extends T>> downstream = waiting.poll();
while (downstream != null) {
downstream.success(result);
downstream = waiting.poll();
}
} finally {
draining.set(false);
}
}
if (!draining.get() && !waiting.isEmpty()) {
tryDrain();
}
}
@Override
public void connect(Downstream super T> downstream) throws Exception {
if (fired.compareAndSet(false, true)) {
upstream.connect(new Downstream() {
@Override
public void error(Throwable throwable) {
result.set(new ResultBackedExecResult<>(Result.error(throwable), Execution.current()));
doDrainInNewSegment();
downstream.error(throwable);
}
@Override
public void success(T value) {
result.set(new ResultBackedExecResult<>(Result.success(value), Execution.current()));
doDrainInNewSegment();
downstream.success(value);
}
@Override
public void complete() {
result.set(new CompleteExecResult<>(Execution.current()));
doDrainInNewSegment();
downstream.complete();
}
});
} else {
Promise.>of(innerDownstream -> {
ExecResult extends T> result = this.result.get();
if (result == null) {
waiting.add(innerDownstream);
} else {
innerDownstream.success(result);
}
}).then(result -> downstream.accept(result));
}
}
private void doDrainInNewSegment() {
this.upstream = null; // release
DefaultExecution.require().getEventLoop().execute(this::tryDrain);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy