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

functionalj.promise.CombineResult Maven / Gradle / Ivy

There is a newer version: 1.0.17
Show newest version
// ============================================================================
// Copyright (c) 2017-2019 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();
    }
    
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy