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

org.glassfish.hk2.runlevel.internal.CurrentTaskFuture Maven / Gradle / Ivy

There is a newer version: 4.0.0-M3
Show newest version
/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright (c) 2013 Oracle and/or its affiliates. All rights reserved.
 *
 * The contents of this file are subject to the terms of either the GNU
 * General Public License Version 2 only ("GPL") or the Common Development
 * and Distribution License("CDDL") (collectively, the "License").  You
 * may not use this file except in compliance with the License.  You can
 * obtain a copy of the License at
 * https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
 * or packager/legal/LICENSE.txt.  See the License for the specific
 * language governing permissions and limitations under the License.
 *
 * When distributing the software, include this License Header Notice in each
 * file and include the License file at packager/legal/LICENSE.txt.
 *
 * GPL Classpath Exception:
 * Oracle designates this particular file as subject to the "Classpath"
 * exception as provided by Oracle in the GPL Version 2 section of the License
 * file that accompanied this code.
 *
 * Modifications:
 * If applicable, add the following below the License Header, with the fields
 * enclosed by brackets [] replaced by your own identifying information:
 * "Portions Copyright [year] [name of copyright owner]"
 *
 * Contributor(s):
 * If you wish your version of this file to be governed by only the CDDL or
 * only the GPL Version 2, indicate your decision by adding "[Contributor]
 * elects to include this software in this distribution under the [CDDL or GPL
 * Version 2] license."  If you don't indicate a single choice of license, a
 * recipient has the option to distribute your version of this file under
 * either the CDDL, the GPL Version 2 or to extend the choice of license to
 * its licensees as provided above.  However, if you add GPL Version 2 code
 * and therefore, elected the GPL Version 2 license, then the option applies
 * only if the new code is made subject to such option by the copyright
 * holder.
 */
package org.glassfish.hk2.runlevel.internal;

import java.util.HashSet;
import java.util.List;
import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

import org.glassfish.hk2.api.ActiveDescriptor;
import org.glassfish.hk2.api.Descriptor;
import org.glassfish.hk2.api.IndexedFilter;
import org.glassfish.hk2.api.Injectee;
import org.glassfish.hk2.api.MultiException;
import org.glassfish.hk2.api.ServiceHandle;
import org.glassfish.hk2.api.ServiceLocator;
import org.glassfish.hk2.runlevel.ChangeableRunLevelFuture;
import org.glassfish.hk2.runlevel.ErrorInformation;
import org.glassfish.hk2.runlevel.RunLevel;
import org.glassfish.hk2.runlevel.RunLevelListener;
import org.glassfish.hk2.runlevel.Sorter;
import org.glassfish.hk2.runlevel.utilities.Utilities;

/**
 * This is the implementation of RunLevelFuture.  There should
 * only be one of these active in the system at any time.  Of
 * course users are given a handle to this object, so they can
 * hold onto references to it for as long as they'd like.
 * 
 * @author jwells
 *
 */
public class CurrentTaskFuture implements ChangeableRunLevelFuture {
    private final AsyncRunLevelContext asyncContext;
    private final Executor executor;
    private final ServiceLocator locator;
    private int proposedLevel;
    private final boolean useThreads;
    private final List> allListenerHandles;
    private final List> allSorterHandles;
    private final int maxThreads;
    private final Timer timer;
    private final long cancelTimeout;
    
    private UpAllTheWay upAllTheWay;
    private DownAllTheWay downAllTheWay;
    
    private boolean done = false;
    private boolean cancelled = false;
    private boolean inCallback = false;
    
    /* package */ CurrentTaskFuture(AsyncRunLevelContext asyncContext,
            Executor executor,
            ServiceLocator locator,
            int proposedLevel,
            int maxThreads,
            boolean useThreads,
            long cancelTimeout,
            Timer timer) {
        this.asyncContext = asyncContext;
        this.executor = executor;
        this.locator = locator;
        this.proposedLevel = proposedLevel;
        this.useThreads = useThreads;
        this.maxThreads = maxThreads;
        this.cancelTimeout = cancelTimeout;
        this.timer = timer;
        
        int currentLevel = asyncContext.getCurrentLevel();
        
        allListenerHandles = locator.getAllServiceHandles(RunLevelListener.class);
        allSorterHandles = locator.getAllServiceHandles(Sorter.class);
        
        if (currentLevel == proposedLevel) {
            done = true;
        }
        else if (currentLevel < proposedLevel) {
            upAllTheWay = new UpAllTheWay(proposedLevel,
                    this,
                    allListenerHandles,
                    allSorterHandles,
                    maxThreads,
                    useThreads,
                    cancelTimeout);
        }
        else {
            downAllTheWay = new DownAllTheWay(proposedLevel, this, allListenerHandles);
        }
    }
    
    /* package */ void go() {
        UpAllTheWay localUpAllTheWay;
        DownAllTheWay localDownAllTheWay;
        
        synchronized (this) {
            localUpAllTheWay = upAllTheWay;
            localDownAllTheWay = downAllTheWay;
        }
        
        if (localUpAllTheWay != null) {
            localUpAllTheWay.go();
        }
        else if (localDownAllTheWay != null) {
            if (useThreads) {
                executor.execute(localDownAllTheWay);
            }
            else {
                localDownAllTheWay.run();
            }
        }
        else {
            asyncContext.jobDone();
        }
    }
    
    @Override
    public boolean isUp() {
        synchronized (this) {
            if (upAllTheWay != null) return true;
            return false;
        }
    }
    
    @Override
    public boolean isDown() {
        synchronized (this) {
            if (downAllTheWay != null) return true;
            return false;
        }
    }

    /* (non-Javadoc)
     * @see java.util.concurrent.Future#cancel(boolean)
     */
    @Override
    public boolean cancel(boolean mayInterruptIfRunning) {
        // Not locking in this order can cause deadlocks
        synchronized (asyncContext) {
            synchronized (this) {
                if (done) return false;
                if (cancelled) return false;
            
                cancelled = true;
            
                if (upAllTheWay != null) {
                    upAllTheWay.cancel();
                }
                else if (downAllTheWay != null) {
                    downAllTheWay.cancel();
                }
            
                return true;
            }
        }
    }

    /* (non-Javadoc)
     * @see java.util.concurrent.Future#isCancelled()
     */
    @Override
    public boolean isCancelled() {
        synchronized (this) {
            return cancelled;
        }
    }

    /* (non-Javadoc)
     * @see java.util.concurrent.Future#isDone()
     */
    @Override
    public boolean isDone() {
        synchronized (this) {
            return done;
        }
    }
    
    @Override
    public int getProposedLevel() {
        synchronized (this) {
            return proposedLevel;
        }
    }
    
    @Override
    public int changeProposedLevel(int proposedLevel) {
        int oldProposedVal;
        boolean needGo = false;
        synchronized (this) {
            if (done) throw new IllegalStateException("Cannot change the proposed level of a future that is already complete");
            if (!inCallback) throw new IllegalStateException(
                    "changeProposedLevel must only be called from inside a RunLevelListener callback method");
            
            oldProposedVal = this.proposedLevel;
            int currentLevel = asyncContext.getCurrentLevel();
            this.proposedLevel = proposedLevel;
            
            if (upAllTheWay != null) {
                if (currentLevel <= proposedLevel) {
                    upAllTheWay.setGoingTo(proposedLevel, false);
                }
                else {
                    // Changing directions to down
                    upAllTheWay.setGoingTo(currentLevel, true); // This will make upAllTheWay stop
                    upAllTheWay = null;
                    
                    downAllTheWay = new DownAllTheWay(proposedLevel, this, allListenerHandles);
                    needGo = true;
                }
            }
            else if (downAllTheWay != null) {
                if (currentLevel >= proposedLevel) {
                    downAllTheWay.setGoingTo(proposedLevel, false);
                }
                else {
                    // Changing directions to up
                    downAllTheWay.setGoingTo(currentLevel, true);  // This will make downAllTheWay stop
                    downAllTheWay = null;
                    
                    upAllTheWay = new UpAllTheWay(proposedLevel,
                            this,
                            allListenerHandles,
                            allSorterHandles,
                            maxThreads,
                            useThreads,
                            cancelTimeout);
                    needGo = true;
                }
            }
            else {
                if (proposedLevel > currentLevel) {
                    upAllTheWay = new UpAllTheWay(proposedLevel,
                            this,
                            allListenerHandles,
                            allSorterHandles,
                            maxThreads,
                            useThreads,
                            cancelTimeout);
                    needGo = true;
                }
                else if (proposedLevel < currentLevel) {
                    downAllTheWay = new DownAllTheWay(proposedLevel, this, allListenerHandles);
                    needGo = true;
                }
            }
        }
        
        if (needGo) {
            go();
        }
        
        return oldProposedVal;
    }
    
    private void setInCallback(boolean inCallback) {
        synchronized (this) {
            this.inCallback = inCallback;
        }
    }

    /* (non-Javadoc)
     * @see java.util.concurrent.Future#get()
     */
    @Override
    public Object get() throws InterruptedException, ExecutionException {
        try {
            return get(Long.MAX_VALUE, TimeUnit.MILLISECONDS);
        }
        catch (TimeoutException te) {
            throw new AssertionError(te);
        }
    }

    /* (non-Javadoc)
     * @see java.util.concurrent.Future#get(long, java.util.concurrent.TimeUnit)
     */
    @Override
    public Object get(long timeout, TimeUnit unit) throws InterruptedException,
            ExecutionException, TimeoutException {
        AllTheWay allTheWay = null;
        synchronized (this) {
            if (upAllTheWay != null) {
                allTheWay = upAllTheWay;
            }
            else if (downAllTheWay != null) {
                allTheWay = downAllTheWay;
            }
        }
        
        if (allTheWay == null) return null;
        
        Boolean result = null;
        for (;;) {
            try {
                result = allTheWay.waitForResult(timeout, unit);
                if (result == null) {
                    synchronized (this) {
                        if (upAllTheWay != null) {
                            allTheWay = upAllTheWay;
                        }
                        else if (downAllTheWay != null) {
                            allTheWay = downAllTheWay;
                        }
                    }
                    
                    continue;
                }
                
                if (!result) {
                    throw new TimeoutException();
                }
                
                synchronized (this) {
                    done = true;
                }
                
                return null;
            }
            catch (MultiException me) {
                synchronized (this) {
                    done = true;
                }
                
                throw new ExecutionException(me);
            }
        }
    }
    
    private void invokeOnProgress(ChangeableRunLevelFuture job, int level,
            List> listeners) {
        setInCallback(true);
        try {
            for (ServiceHandle listener : listeners) {
                try {
                    listener.getService().onProgress(job, level);
                }
                catch (Throwable th) {
                    // TODO:  Need a log message here
               }
            }
        }
        finally {
            setInCallback(false);
        }
    }
    
    private static void invokeOnCancelled(CurrentTaskFuture job, int levelAchieved,
            List> listeners) {
        for (ServiceHandle listener : listeners) {
            try {
                listener.getService().onCancelled(new CurrentTaskFutureWrapper(job), levelAchieved);
            }
            catch (Throwable th) {
                // TODO:  Need a log message here
            }
        }
    }
    
    private static ErrorInformation invokeOnError(CurrentTaskFuture job, Throwable th,
            ErrorInformation.ErrorAction action,
            List> listeners,
            Descriptor descriptor) {
        ErrorInformationImpl errorInfo = new ErrorInformationImpl(th, action, descriptor);
        
        for (ServiceHandle listener : listeners) {
            try {
                listener.getService().onError(new CurrentTaskFutureWrapper(job),
                        errorInfo);
            }
            catch (Throwable th2) {
                 // TODO:  Need a log message here
            }
        }
        
        return errorInfo;
    }
    
    private interface AllTheWay {
        /**
         * The method to call on the internal job
         * 
         * @param timeout The amount of time to wait for a result
         * @param unit The unit of the above time value
         * @return True if the job finished, False if the timeout is up prior to the job
         * finishing, and null if the job was repurposed and the caller may now need to
         * listen on a different job
         * @throws InterruptedException On a thread getting jacked
         * @throws MultiException Other exceptions
         */
        public Boolean waitForResult(long timeout, TimeUnit unit) throws InterruptedException, MultiException;
        
    }
            
    
    private class UpAllTheWay implements AllTheWay {
        private final Object lock = new Object();
        
        private int goingTo;
        private final int maxThreads;
        private final boolean useThreads;
        private final CurrentTaskFuture future;
        private final List> listeners;
        private final List> sorters;
        private final long cancelTimeout;
        
        private int workingOn;
        private UpOneLevel currentJob;
        private boolean cancelled = false;
        private boolean done = false;
        private boolean repurposed = false;
        private MultiException exception = null;
        
        private UpAllTheWay(int goingTo, CurrentTaskFuture future,
                List> listeners,
                List> sorters,
                int maxThreads,
                boolean useThreads,
                long cancelTimeout) {
            this.goingTo = goingTo;
            this.future = future;
            this.listeners = listeners;
            this.maxThreads = maxThreads;
            this.useThreads = useThreads;
            this.sorters = sorters;
            this.cancelTimeout = cancelTimeout;
            
            workingOn = asyncContext.getCurrentLevel();
        }
        
        private void cancel() {
            synchronized (lock) {
                cancelled = true;
                asyncContext.levelCancelled();
                currentJob.cancel();
            }
        }
        
        @Override
        public Boolean waitForResult(long timeout, TimeUnit unit) throws InterruptedException, MultiException {
            long totalWaitTimeMillis = TimeUnit.MILLISECONDS.convert(timeout, unit);
            
            synchronized (lock) {
                while (totalWaitTimeMillis > 0L && !done && !repurposed) {
                    long startTime = System.currentTimeMillis();
                    
                    lock.wait(totalWaitTimeMillis);
                    
                    long elapsedTime = System.currentTimeMillis() - startTime;
                    totalWaitTimeMillis -= elapsedTime;
                }
                
                if (repurposed) return null;
                
                if (done && (exception != null)) {
                    throw exception;
                }
                
                return done;
            }
        }
        
        private void setGoingTo(int goingTo, boolean repurposed) {
            synchronized (lock) {
                this.goingTo = goingTo;
                if (repurposed) {
                    this.repurposed = true;
                }
            }
        }
        
        private void go() {
            if (useThreads) {
                synchronized (lock) {
                    workingOn++;
                    if (workingOn > goingTo) {
                        if (!repurposed) {
                            asyncContext.jobDone();
                    
                            done = true;
                        }
                        
                        lock.notifyAll();
                        return;
                    }
            
                    currentJob = new UpOneLevel(workingOn,
                            this,
                            future,
                            listeners,
                            sorters,
                            maxThreads,
                            cancelTimeout);
            
                    executor.execute(currentJob);
                    return;
                }
            }
                
            workingOn++;
            while (workingOn <= goingTo) {
                synchronized (lock) {
                    if (done) break;
                    
                    currentJob = new UpOneLevel(workingOn,
                            this,
                            future,
                            listeners,
                            sorters,
                            0,
                            cancelTimeout);
                }
                
                currentJob.run();
                
                workingOn++;
            }
             
            synchronized (lock) {
                if (done) return;
                
                if (!repurposed) {
                    asyncContext.jobDone();
                
                    done = true;
                }
                
                lock.notifyAll();
            }
        }
        
        private void currentJobComplete(MultiException accumulatedExceptions) {
            asyncContext.clearErrors();
            
            if (accumulatedExceptions != null) {
                DownAllTheWay downer = new DownAllTheWay(workingOn - 1, null, null);
                
                downer.run();
                
                synchronized (lock) {                    
                    done = true;
                    this.exception = accumulatedExceptions;
                    lock.notifyAll();
                    
                    asyncContext.jobDone();
                }
                
                return;
            }
            
            DownAllTheWay downer = null;
            synchronized (lock) {
                if (cancelled) {
                    downer = new DownAllTheWay(workingOn - 1, null, null);
                }
            }
            
            if (downer != null) {
                downer.run();
                
                invokeOnCancelled(future, workingOn - 1, listeners);
                
                synchronized (lock) {
                    done = true;
                    lock.notifyAll();
                        
                    asyncContext.jobDone();
                        
                    return;
                }
            }
            
            asyncContext.setCurrentLevel(workingOn);
            invokeOnProgress(future, workingOn, listeners);
                
            if (useThreads) {
                go();
            }
        }
    }
    
    private class UpOneLevel implements Runnable {
        private final Object lock = new Object();
        private final Object queueLock = new Object();
        private final int upToThisLevel;
        private final CurrentTaskFuture currentTaskFuture;
        private final List> listeners;
        private final List> sorters;
        private final UpAllTheWay master;
        private final int maxThreads;
        private final long cancelTimeout;
        private int numJobs;
        private int completedJobs;
        private MultiException accumulatedExceptions;
        private boolean cancelled = false;
        private CancelTimer hardCanceller;
        private int numJobsRunning = 0;
        private boolean hardCancelled = false;
        private final HashSet> outstandingHandles = new HashSet>();
        
        private UpOneLevel(int paramUpToThisLevel,
                UpAllTheWay master,
                CurrentTaskFuture currentTaskFuture,
                List> listeners,
                List> sorters,
                int maxThreads,
                long cancelTimeout) {
            this.upToThisLevel = paramUpToThisLevel;
            this.master = master;
            this.maxThreads = maxThreads;
            this.currentTaskFuture = currentTaskFuture;
            this.listeners = listeners;
            this.sorters = sorters;
            this.cancelTimeout = cancelTimeout;
        }
        
        private void cancel() {
            synchronized (lock) {
                cancelled = true;
                hardCanceller = new CancelTimer(this);
                timer.schedule(hardCanceller, cancelTimeout);
            }
        }
        
        private void hardCancel() {
            synchronized (asyncContext) {
                synchronized (lock) {
                    hardCancelled = true;
                }
                
                HashSet> poisonMe;
                synchronized (queueLock) {
                    poisonMe = new HashSet>(outstandingHandles);
                    outstandingHandles.clear();
                }
                
                for (ServiceHandle handle : poisonMe) {
                    asyncContext.hardCancelOne(handle.getActiveDescriptor());
                }
            }
            
            master.currentJobComplete(null);
        }
        
        private void jobRunning(ServiceHandle handle) {
            numJobsRunning++;
            outstandingHandles.add(handle);
        }
        
        private void jobFinished(ServiceHandle handle) {
            outstandingHandles.remove(handle);
            numJobsRunning--;
        }
        
        private int getJobsRunning() {
            return numJobsRunning;
        }
        
        private List> applySorters(List> jobs) {
            List> retVal = jobs;
            
            for (ServiceHandle sorterHandle : sorters) {
                Sorter sorter = sorterHandle.getService();
                if (sorter == null) continue;
                
                List> sortedList = sorter.sort(retVal);
                if (sortedList == null) continue;
                
                retVal = sortedList;
            }
            
            return retVal;
        }

        @Override
        public void run() {
            Object jobsLock = new Object();
            List> jobs = locator.getAllServiceHandles(new IndexedFilter() {

                @Override
                public boolean matches(Descriptor d) {
                    return (upToThisLevel == Utilities.getRunLevelValue(locator, d));
                }

                @Override
                public String getAdvertisedContract() {
                    return RunLevel.class.getName();
                }

                @Override
                public String getName() {
                    return null;
                }
                
            });
            
            jobs = applySorters(jobs);
            
            numJobs = jobs.size();
            if (numJobs <= 0) {
                jobComplete();
                return;
            }
            
            int runnersToCreate = ((numJobs < maxThreads) ? numJobs : maxThreads) - 1;
            if (!useThreads) runnersToCreate = 0;
            
            for (int lcv = 0; lcv < runnersToCreate; lcv++) {
                QueueRunner runner = new QueueRunner(locator, asyncContext, jobsLock, jobs, this, lock, maxThreads);
                
                executor.execute(runner);
            }
            
            QueueRunner myRunner = new QueueRunner(locator, asyncContext, jobsLock, jobs, this, lock, maxThreads);
            myRunner.run();
        }
        
        private void fail(Throwable th, Descriptor descriptor) {
            synchronized (lock) {
                if (hardCancelled) return;
                
                ErrorInformation info = invokeOnError(currentTaskFuture, th,
                        ErrorInformation.ErrorAction.GO_TO_NEXT_LOWER_LEVEL_AND_STOP,
                        listeners,
                        descriptor);
                
                if (ErrorInformation.ErrorAction.IGNORE.equals(info.getAction())) return;
                
                if (accumulatedExceptions == null) {
                    accumulatedExceptions = new MultiException();
                }
                
                accumulatedExceptions.addError(th);
            }
        }
        
        private void jobComplete() {
            boolean complete = false;
            synchronized (lock) {
                if (hardCancelled) return;
                
                completedJobs++;
                if (completedJobs >= numJobs) {
                    complete = true;
                    if (hardCanceller != null) {
                        hardCanceller.cancel();
                        hardCanceller = null;
                    }
                }
            }
            
            if (complete) {
                master.currentJobComplete(accumulatedExceptions);
            }
        }
        
    }
    
    private static class CancelTimer extends TimerTask {
        private final UpOneLevel parent;
        
        private CancelTimer(UpOneLevel parent) {
            this.parent = parent;
        }

        @Override
        public void run() {
            parent.hardCancel();
        }
    }
    
    /**
     * Goes down all the way to the proposed level
     * 
     * @author jwells
     *
     */
    private class DownAllTheWay implements Runnable, AllTheWay {
        private int goingTo;
        private CurrentTaskFuture future;
        private final List> listeners;
        
        private int workingOn;
        
        private boolean cancelled = false;
        private boolean done = false;
        private boolean repurposed = false;
        
        private Throwable lastError = null;
        private ActiveDescriptor lastErrorDescriptor = null;
        
        private boolean queueWaiter = false;
        private List> queue;
        private boolean downHardCancelled = false;
        
        private HardCancelDownTimer hardCancelDownTimer = null;
        
        public DownAllTheWay(int goingTo,
                CurrentTaskFuture future,
                List> listeners) {
            this.goingTo = goingTo;
            this.future = future;
            this.listeners = listeners;
            
            if (future == null) {
                // This is an error or cancelled case, so we are pretending
                // we have gotten higher than we have
                workingOn = asyncContext.getCurrentLevel() + 1;
            }
            else {
                workingOn = asyncContext.getCurrentLevel();
            }
        }
        
        private void cancel() {
            List> localQueue;
            
            synchronized (this) {
                if (cancelled) return; // idempotent
                cancelled = true;
                
                if (queue == null) {
                    queueWaiter = true;
                    
                    while (!done && queue == null) {
                        try {
                            this.wait();
                        }
                        catch (InterruptedException e) {
                            throw new RuntimeException(e);
                        }
                    }
                    
                    if (done) return;
                }
                
                localQueue = queue;
            }
                
            synchronized (localQueue) {
                if (localQueue.isEmpty()) return;
                
                hardCancelDownTimer = new HardCancelDownTimer(this, localQueue);
                timer.schedule(hardCancelDownTimer, cancelTimeout, cancelTimeout);
            }
        }
        
        private void setGoingTo(int goingTo, boolean repurposed) {
            synchronized (this) {
                this.goingTo = goingTo;
                if (repurposed) {
                    this.repurposed = true;
                }
            }
        }
        
        private int getGoingTo() {
            synchronized (this) {
                return goingTo;
            }
        }

        @Override
        public void run() {
            while (workingOn > getGoingTo()) {
                boolean runOnCancelled;
                boolean localCancelled;
                synchronized (this) {
                    localCancelled = cancelled;
                    runOnCancelled = cancelled && (future != null);
                }
                
                if (runOnCancelled) {
                    // Run outside of lock
                    invokeOnCancelled(future, workingOn, listeners);
                }
                
                synchronized (this) {
                    if (localCancelled) {
                        asyncContext.jobDone();
                        
                        done = true;
                        
                        this.notifyAll();
                        
                        return;
                    }
                }
                
                int proceedingTo = workingOn - 1;
                
                // This happens FIRST.  Here the definition of
                // the current level is that level which is guaranteed
                // to have ALL of its known services started.  Once
                // we destroy the first one of them (or are about to)
                // then we are officially at the next level
                asyncContext.setCurrentLevel(proceedingTo);
                
                // But we don't call the proceedTo until all those services are gone
                List> localQueue = asyncContext.getOrderedListOfServicesAtLevel(workingOn);
                synchronized(this) {
                    queue = localQueue;
                    if (queueWaiter) {
                        this.notifyAll();
                    }
                    queueWaiter = false;
                }
                
                ErrorInformation errorInfo = null;
                synchronized (queue) {
                    for (;;) {
                        DownQueueRunner currentRunner = new DownQueueRunner(queue, queue, this, locator);
                        executor.execute(currentRunner);
                    
                        lastError = null;
                        for (;;) {
                            while (!queue.isEmpty() && (lastError == null) && (downHardCancelled == false)) {
                                try {
                                    queue.wait();
                                }
                                catch (InterruptedException ie) {
                                    throw new RuntimeException(ie);
                                }
                            }
                            
                            if (downHardCancelled) {
                                currentRunner.caput = true;
                            }
                        
                            if ((lastError != null) && (future != null)) {
                                errorInfo = invokeOnError(future, lastError, ErrorInformation.ErrorAction.IGNORE, listeners, lastErrorDescriptor);
                            }
                            lastError = null;
                            lastErrorDescriptor = null;
                        
                            if (queue.isEmpty() || downHardCancelled) {
                                downHardCancelled = false;
                                break;
                            }
                        }
                        
                        if (queue.isEmpty()) {
                            if (hardCancelDownTimer != null) {
                                hardCancelDownTimer.cancel();
                            }
                            
                            break;
                        }
                    }
                }
                
                synchronized(this) {
                    queue = null;
                }
                
                if (errorInfo != null && ErrorInformation.ErrorAction.GO_TO_NEXT_LOWER_LEVEL_AND_STOP.equals(errorInfo.getAction())) {
                    synchronized (this) {
                        goingTo = workingOn;
                    }
                }
                
                workingOn--;
                
                if (future != null) {
                    invokeOnProgress(future, proceedingTo, listeners);
                }
            }
            
            if (future == null) {
                // This is done as part of a cancel or error, do no
                // notifying this is special
                return;
            }
            
            synchronized (this) {
                if (!repurposed) {
                    asyncContext.jobDone();
                
                    done = true;
                }
                
                this.notifyAll();
            }
            
        }
        
        @Override
        public Boolean waitForResult(long timeout, TimeUnit unit) throws InterruptedException, MultiException {
            long totalWaitTimeMillis = TimeUnit.MILLISECONDS.convert(timeout, unit);
            
            synchronized (this) {
                while (totalWaitTimeMillis > 0L && !done && !repurposed) {
                    long startTime = System.currentTimeMillis();
                    
                    this.wait(totalWaitTimeMillis);
                    
                    long elapsedTime = System.currentTimeMillis() - startTime;
                    totalWaitTimeMillis -= elapsedTime;
                }
                
                if (repurposed) return null;
                
                return done;
            }
        }
        
    }
    
    private static class HardCancelDownTimer extends TimerTask {
        private final DownAllTheWay parent;
        private final List> queue;
        
        private int lastQueueSize;
        
        private HardCancelDownTimer(DownAllTheWay parent, List> queue) {
            this.parent = parent;
            this.queue = queue;
            lastQueueSize = queue.size();
        }

        @Override
        public void run() {
            synchronized (queue) {
                int currentSize = queue.size();
                if (currentSize == 0) return;
                
                if (currentSize == lastQueueSize) {
                    parent.downHardCancelled = true;
                    queue.notify();
                }
                else {
                    lastQueueSize = currentSize;
                }
            }
        }
    }
    
    private static class QueueRunner implements Runnable {
        private final ServiceLocator locator;
        private final AsyncRunLevelContext asyncContext;
        private final Object queueLock;
        private final List> queue;
        private final UpOneLevel parent;
        private final Object parentLock;
        private final int maxThreads;
        private ServiceHandle wouldHaveBlocked;
        private final HashSet> alreadyTried = new HashSet>();
        
        private QueueRunner(ServiceLocator locator,
                AsyncRunLevelContext asyncContext,
                Object queueLock,
                List> queue,
                UpOneLevel parent,
                Object parentLock,
                int maxThreads) {
            this.locator = locator;
            this.asyncContext = asyncContext;
            this.queueLock = queueLock;
            this.queue = queue;
            this.parent = parent;
            this.parentLock = parentLock;
            this.maxThreads = maxThreads;
        }

        @Override
        public void run() {
            ServiceHandle runningHandle = null;
            for (;;) {
                ServiceHandle job;
                boolean block;
                synchronized(queueLock) {
                    if (runningHandle != null) parent.jobFinished(runningHandle);
                    
                    if (wouldHaveBlocked != null) {
                        alreadyTried.add(wouldHaveBlocked.getActiveDescriptor());
                        
                        queue.add(queue.size(), wouldHaveBlocked);
                        wouldHaveBlocked = null;
                    }
                    
                    if (queue.isEmpty()) return;
                    
                    if (maxThreads <= 0) {
                        block = true;
                    }
                    else {
                        int currentlyEmptyThreads = maxThreads - parent.getJobsRunning();
                        block = (queue.size() <= currentlyEmptyThreads);
                    }
                    
                    if (block) {
                        job = queue.remove(0);
                    }
                    else {
                        job = null;
                        for (int lcv = 0; lcv < queue.size(); lcv++) {
                            ActiveDescriptor candidate = queue.get(lcv).getActiveDescriptor();
                            if (!alreadyTried.contains(candidate)) {
                                job = queue.remove(lcv);
                                break;
                            }
                        }
                        if (job == null) {
                            // Every job in the queue is one I've tried already
                            job = queue.remove(0);
                            
                            block = true;
                        }
                    }
                    
                    parent.jobRunning(job);
                    runningHandle = job;
                }
                
                oneJob(job, block);
            }
            
        }
        
        /**
         * This method does a preliminary check of whether or not the descriptor (or any children) would cause
         * the thread to block.  If this method returns true then we do not try this service, which can save
         * on going down the getService stack and on the throwing and creation of WouldBlockException
         * 
         * @param cycleChecker To ensure we are not caught in a cycle
         * @param checkMe The descriptor to check
         * @return false if as far as we know this descriptor would NOT block, true if we think if we tried
         * this descriptor right now that it would block
         */
        private boolean isWouldBlockRightNow(HashSet> cycleChecker, ActiveDescriptor checkMe) {
            if (checkMe == null) return false;
            
            if (cycleChecker.contains(checkMe)) return false;
            cycleChecker.add(checkMe);
            
            if (asyncContext.wouldBlockRightNow(checkMe)) {
                return true;
            }
            
            if (!checkMe.isReified()) {
                checkMe = locator.reifyDescriptor(checkMe);
            }
            
            for (Injectee ip : checkMe.getInjectees()) {
                ActiveDescriptor childService;
                try {
                    childService = locator.getInjecteeDescriptor(ip);
                }
                catch (MultiException me) {
                    continue;
                }
                
                if (childService == null) continue;
                
                if (!childService.getScope().equals(RunLevel.class.getName())) continue;
                
                if (isWouldBlockRightNow(cycleChecker, childService)) {
                    if (asyncContext.wouldBlockRightNow(checkMe)) {
                        return true;
                    }
                    return true;
                }
            }
            
            return false;
        }
        
        private void oneJob(ServiceHandle fService, boolean block) {
            fService.setServiceData(!block);
            boolean completed = true;
            try {
                boolean ok;
                synchronized (parentLock) {
                    ok = (!parent.cancelled && (parent.accumulatedExceptions == null));
                }
                
                if (!block && isWouldBlockRightNow(new HashSet>(), fService.getActiveDescriptor())) {
                    wouldHaveBlocked = fService;
                    completed = false;
                    ok = false;
                }
                
                if (ok) {
                    fService.getService();
                }
            }
            catch (MultiException me) {
                if (!block && isWouldBlock(me)) {
                    // In this case completed is FALSE, as the job has NOT completed
                    wouldHaveBlocked = fService;
                    completed = false;
                }
                else if (!isWasCancelled(me)) {
                    parent.fail(me, fService.getActiveDescriptor());
                }
            }
            catch (Throwable th) {
                parent.fail(th, fService.getActiveDescriptor());
            }
            finally {
                fService.setServiceData(null);
                if (completed) {
                    parent.jobComplete();
                }
            }
        }
    }
    
    private static class DownQueueRunner implements Runnable {
        private final Object queueLock;
        private final List> queue;
        private final DownAllTheWay parent;
        private final ServiceLocator locator;
        private boolean caput;
        
        private DownQueueRunner(Object queueLock,
                List> queue,
                DownAllTheWay parent,
                ServiceLocator locator) {
            this.queueLock = queue;
            this.queue = queue;
            this.parent = parent;
            this.locator = locator;
        }

        @Override
        public void run() {
            for (;;) {
                ActiveDescriptor job = null;
                synchronized (queueLock) {
                    if (caput) return;
                    
                    if (queue.isEmpty()) {
                        queueLock.notify();
                        return;
                    }
                    job = queue.remove(0);
                }
                
                try {
                    locator.getServiceHandle(job).destroy();
                }
                catch (Throwable th) {
                    synchronized (queueLock) {
                        parent.lastError = th;
                        parent.lastErrorDescriptor = job;
                        queueLock.notify();
                    }
                }
            }
            
        }
        
    }
    
    /* package */ final static boolean isWouldBlock(Throwable th) {
        return isACertainException(th, WouldBlockException.class);
    }
    
    private final static boolean isWasCancelled(Throwable th) {
        return isACertainException(th, WasCancelledException.class);
    }
    
    private final static boolean isACertainException(Throwable th, Class type) {
        Throwable cause = th;
        while (cause != null) {
            if (cause instanceof MultiException) {
                MultiException me = (MultiException) cause;
                for (Throwable innerMulti : me.getErrors()) {
                    if (isACertainException(innerMulti, type)) {
                        return true;
                    }
                }
            }
            else if (type.isAssignableFrom(cause.getClass())) {
                return true;
            }
            
            cause = cause.getCause();
        }
        
        return false;
    }
    
    @Override
    public String toString() {
        return "RunLevelFuture(proposedLevel=" + proposedLevel + "," +
          System.identityHashCode(this) + ")";
    }

    
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy