org.glassfish.hk2.runlevel.internal.AsyncRunLevelContext Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2013-2016 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.security.AccessController;
import java.security.PrivilegedAction;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Timer;
import java.util.concurrent.Executor;
import java.util.concurrent.SynchronousQueue;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.inject.Inject;
import org.glassfish.hk2.api.ActiveDescriptor;
import org.glassfish.hk2.api.Context;
import org.glassfish.hk2.api.DescriptorVisibility;
import org.glassfish.hk2.api.MultiException;
import org.glassfish.hk2.api.ServiceHandle;
import org.glassfish.hk2.api.ServiceLocator;
import org.glassfish.hk2.api.Visibility;
import org.glassfish.hk2.runlevel.CurrentlyRunningException;
import org.glassfish.hk2.runlevel.RunLevel;
import org.glassfish.hk2.runlevel.RunLevelController;
import org.glassfish.hk2.runlevel.RunLevelFuture;
import org.glassfish.hk2.runlevel.utilities.Utilities;
import org.jvnet.hk2.annotations.Service;
/**
* @author jwells
*
*/
@Service @Visibility(DescriptorVisibility.LOCAL)
public class AsyncRunLevelContext {
private final static String DEBUG_CONTEXT_PROPERTY = "org.jvnet.hk2.properties.debug.runlevel.context";
private final static boolean DEBUG_CONTEXT = AccessController.doPrivileged(new PrivilegedAction() {
@Override
public Boolean run() {
return Boolean.parseBoolean(System.getProperty(DEBUG_CONTEXT_PROPERTY, "false"));
}
});
private static final Logger logger = Logger.getLogger(AsyncRunLevelContext.class.getName());
private static final Timer timer = new Timer(true);
private static final ThreadFactory THREAD_FACTORY = new RunLevelThreadFactory();
private final org.glassfish.hk2.utilities.reflection.Logger hk2Logger = org.glassfish.hk2.utilities.reflection.Logger.getLogger();
private int currentLevel = RunLevel.RUNLEVEL_VAL_INITIAL;
private CurrentTaskFutureWrapper currentTask = null;
private static final Executor DEFAULT_EXECUTOR = new ThreadPoolExecutor(0, Integer.MAX_VALUE,
60L, TimeUnit.SECONDS,
new SynchronousQueue(true),
THREAD_FACTORY);
/**
* The backing maps for this context.
*/
private final Map, Object> backingMap =
new HashMap, Object>();
/*
* The within level errors thrown. This prevents double
* starting a service that failed within a level
*/
private final Map, Throwable> levelErrorMap =
new HashMap, Throwable>();
private boolean wasCancelled = false;
/**
* The set of services currently being created
*/
private final HashMap, Long> creatingDescriptors = new HashMap, Long>();
/**
* The set of descriptors that have been hard cancelled
*/
private final HashSet> hardCancelledDescriptors = new HashSet>();
private final LinkedList> orderedCreationList = new LinkedList>();
private Executor executor = DEFAULT_EXECUTOR;
private final ServiceLocator locator;
private int maxThreads = Integer.MAX_VALUE;
private RunLevelController.ThreadingPolicy policy = RunLevelController.ThreadingPolicy.FULLY_THREADED;
private long cancelTimeout = 5 * 1000;
private Integer modeOverride = null;
/**
* Constructor for the guy who does the work
*
* @param locator The locator to use
*/
@Inject
private AsyncRunLevelContext(ServiceLocator locator) {
this.locator = locator;
}
/**
* This is from the {@link Context} API, called by the wrapper
*
* @param activeDescriptor the descriptor to create
* @param root The root descriptor
* @return The object created
*/
@SuppressWarnings("unchecked")
public U findOrCreate(ActiveDescriptor activeDescriptor,
ServiceHandle> root) {
String oneLineDescriptor = null;
if (DEBUG_CONTEXT) {
oneLineDescriptor = oneLineDescriptor(activeDescriptor);
hk2Logger.debug("AsyncRunLevelController findOrCreate for " + oneLineDescriptor + " and root " + oneLineRoot(root));
}
boolean throwWouldBlock;
if (root == null) {
throwWouldBlock = false;
}
else {
Object rootBlockObject = root.getServiceData();
if (rootBlockObject == null) {
throwWouldBlock = false;
}
else {
throwWouldBlock = ((Boolean) rootBlockObject).booleanValue();
}
}
U retVal = null;
int localCurrentLevel;
Integer localModeOverride;
long tid = -1L;
synchronized (this) {
localModeOverride = modeOverride;
retVal = (U) backingMap.get(activeDescriptor);
if (retVal != null) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController found " + oneLineDescriptor);
}
return retVal;
}
Throwable previousException = levelErrorMap.get(activeDescriptor);
if (previousException != null) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController tried once, it failed, rethrowing " + oneLineDescriptor, previousException);
}
if (previousException instanceof RuntimeException) {
throw (RuntimeException) previousException;
}
throw new RuntimeException(previousException);
}
if (hardCancelledDescriptors.contains(activeDescriptor)) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController hard cancelled " + oneLineDescriptor);
}
throw new MultiException(new WasCancelledException(activeDescriptor), false);
}
while (creatingDescriptors.containsKey(activeDescriptor)) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController already being created " + oneLineDescriptor);
}
long holdingLock = creatingDescriptors.get(activeDescriptor);
if (holdingLock == Thread.currentThread().getId()) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController circular dependency " + oneLineDescriptor);
}
throw new MultiException(new IllegalStateException(
"Circular dependency involving " + activeDescriptor.getImplementation() +
" was found. Full descriptor is " + activeDescriptor));
}
if (throwWouldBlock) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController would block optimization " + oneLineDescriptor);
}
throw new MultiException(new WouldBlockException(activeDescriptor), false);
}
try {
this.wait();
}
catch (InterruptedException ie) {
throw new MultiException(ie);
}
}
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController finished creating wait for " + oneLineDescriptor);
}
retVal = (U) backingMap.get(activeDescriptor);
if (retVal != null) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController second chance found " + oneLineDescriptor);
}
return retVal;
}
previousException = levelErrorMap.get(activeDescriptor);
if (previousException != null) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController service already threw " + oneLineDescriptor);
}
if (previousException instanceof RuntimeException) {
throw (RuntimeException) previousException;
}
throw new RuntimeException(previousException);
}
if (hardCancelledDescriptors.contains(activeDescriptor)) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController second chance hard cancel " + oneLineDescriptor);
}
throw new MultiException(new WasCancelledException(activeDescriptor), false);
}
tid = Thread.currentThread().getId();
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController am creating " + oneLineDescriptor + " in thread " + tid);
}
creatingDescriptors.put(activeDescriptor, tid);
localCurrentLevel = currentLevel;
if (currentTask != null && currentTask.isUp()) {
localCurrentLevel++;
if (localCurrentLevel > currentTask.getProposedLevel()) {
localCurrentLevel = currentTask.getProposedLevel();
}
}
}
Throwable error = null;
try {
int mode = Utilities.getRunLevelMode(locator, activeDescriptor, localModeOverride);
if (mode == RunLevel.RUNLEVEL_MODE_VALIDATING) {
validate(activeDescriptor, localCurrentLevel);
}
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController prior to actual create " + oneLineDescriptor + " in thread " + tid);
}
retVal = activeDescriptor.create(root);
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController after actual create " + oneLineDescriptor + " in thread " + tid);
}
return retVal;
}
catch (Throwable th) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController got exception for " + oneLineDescriptor + " in thread " + tid, th);
}
if (th instanceof MultiException) {
if (!CurrentTaskFuture.isWouldBlock((MultiException) th)) {
error = th;
}
// We want WouldBlock rethrown
}
else {
error = th;
}
if (th instanceof RuntimeException) {
throw (RuntimeException) th;
}
throw new RuntimeException(th);
}
finally {
synchronized (this) {
boolean hardCancelled = hardCancelledDescriptors.remove(activeDescriptor);
if (retVal != null) {
if (!hardCancelled) {
backingMap.put(activeDescriptor, retVal);
orderedCreationList.addFirst(activeDescriptor);
}
if (wasCancelled || hardCancelled) {
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController cancellation race failed " + oneLineDescriptor + " in thread " + tid);
}
// Even though this service actually ran to completion, we
// are going to pretend it failed. Putting it in the lists
// above will ensure it gets properly shutdown
MultiException cancelledException = new MultiException(new WasCancelledException(activeDescriptor), false);
if (!hardCancelled) {
levelErrorMap.put(activeDescriptor, cancelledException);
}
creatingDescriptors.remove(activeDescriptor);
this.notifyAll();
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController other threads notified cancellation path for " +
oneLineDescriptor + " in thread " + tid);
}
throw cancelledException;
}
}
else if (error != null) {
if (!hardCancelled) {
levelErrorMap.put(activeDescriptor, error);
}
}
creatingDescriptors.remove(activeDescriptor);
this.notifyAll();
if (DEBUG_CONTEXT) {
hk2Logger.debug("AsyncRunLevelController other threads notified for " +
oneLineDescriptor + " in thread " + tid);
}
}
}
}
/**
* The {@link Context} API for discovering if a descriptor has been created
*
* @param descriptor The descriptor to find
* @return true if already created, false otherwise
*/
public boolean containsKey(ActiveDescriptor> descriptor) {
synchronized (this) {
return backingMap.containsKey(descriptor);
}
}
/* package */ boolean wouldBlockRightNow(ActiveDescriptor> desc) {
synchronized (this) {
return creatingDescriptors.containsKey(desc);
}
}
/**
* No need to lock this, it is called with the lock already held
*
* @param descriptor the non-null descriptor to hard cancel
*/
/* package */ void hardCancelOne(ActiveDescriptor> descriptor) {
if (creatingDescriptors.containsKey(descriptor)) {
// This guy has been hard-cancelled, mark it down
hardCancelledDescriptors.add(descriptor);
}
}
/**
* The {@link Context} API. Removes a descriptor from the set
*
* @param descriptor The descriptor to destroy
*/
@SuppressWarnings("unchecked")
public void destroyOne(ActiveDescriptor> descriptor) {
Object retVal = null;
synchronized (this) {
retVal = backingMap.remove(descriptor);
if (retVal == null) return;
}
((ActiveDescriptor
© 2015 - 2025 Weber Informatics LLC | Privacy Policy