functionalj.promise.CombineResult Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of functionalj-core Show documentation
Show all versions of functionalj-core Show documentation
The module for FunctionalJ Core.
// ============================================================================
// Copyright (c) 2017-2021 Nawapunth Manusitthipol (NawaMan - http://nawaman.net).
// ----------------------------------------------------------------------------
// MIT License
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
// ============================================================================
package functionalj.promise;
import java.util.Objects;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.stream.Stream;
import functionalj.function.Func1;
import functionalj.function.NamedExpression;
import functionalj.list.FuncList;
import functionalj.result.Result;
import lombok.val;
@SuppressWarnings("rawtypes")
public class CombineResult {
private final Func1, Result> mergeFunc;
// TODO - Add Else ... which will be called with the current value when unsuccessfull.
private final DeferAction action;
private final int count;
private final Result[] results;
private final SubscriptionRecord[] subscriptions;
private final AtomicBoolean isDone;
private final Promise promise;
CombineResult(FuncList>> hasPromises,
Func1, Result> mergeFunc) {
this.mergeFunc = mergeFunc;
this.count = hasPromises.size();
this.results = new Result[count];
this.subscriptions = new SubscriptionRecord[count];
this.isDone = new AtomicBoolean(false);
val promises = hasPromises
.map(promise -> promise.getExpression())
.map(promise -> promise.getPromise());
this.action = DeferAction.of((Class)null, OnStart.run(()->{
promises.forEach(promise -> promise.start());
}));
promises
.mapWithIndex ((index, promise) -> promise.onComplete(result -> processResult(index, result)))
.forEachWithIndex((index, sub) -> subscriptions[index] = sub);
this.promise = action.getPromise();
this.promise.eavesdrop(result->{
if (result.isCancelled()) {
unsbscribeAll();
}
});
}
DeferAction getDeferAction() {
return action;
}
@SuppressWarnings("unchecked")
private void processResult(int index, Result result) {
if (isDone.get())
return;
if (result.isCancelled())
doneAsCancelled(index);
if (result.isNotReady())
doneAsNotReady(index, result);
if (result.isException())
doneAsException(index, result);
results[index] = result;
if (!(count == Stream.of(results).filter(Objects::nonNull).count()))
return;
if (!isDone.compareAndSet(false, true))
return;
val resultList = (FuncList)FuncList.from(results);
val mergedResult = mergeFunc.apply(resultList);
action.completeWith((Result)mergedResult);
}
private void unsbscribeAll() {
for(val subscription : subscriptions) {
if (subscription != null)
subscription.unsubscribe();
}
}
private void doneAsCancelled(int index) {
if (!isDone.compareAndSet(false, true))
return;
action.abort("Promise#" + index);
unsbscribeAll();
}
private void doneAsNotReady(
int index,
Result result) {
if (!isDone.compareAndSet(false, true))
return;
action.abort(
"Promise#" + index,
new IllegalStateException(
"Result cannot be in 'not ready' at this point: " + result.getStatus(),
result.getException()));
unsbscribeAll();
}
private void doneAsException(
int index,
Result result) {
if (!isDone.compareAndSet(false, true))
return;
action.fail(new PromisePartiallyFailException(index, count, result.getException()));
unsbscribeAll();
}
}