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

bndtools.central.Central Maven / Gradle / Ivy

The newest version!
package bndtools.central;

import static aQute.bnd.exceptions.FunctionWithException.asFunction;

import java.io.File;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.Callable;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.BiConsumer;
import java.util.function.BooleanSupplier;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import org.bndtools.api.BndtoolsConstants;
import org.bndtools.api.IStartupParticipant;
import org.bndtools.api.ModelListener;
import org.eclipse.core.resources.IFile;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.resources.IResource;
import org.eclipse.core.resources.IResourceDelta;
import org.eclipse.core.resources.IWorkspaceRoot;
import org.eclipse.core.resources.ResourcesPlugin;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.MultiStatus;
import org.eclipse.core.runtime.NullProgressMonitor;
import org.eclipse.core.runtime.OperationCanceledException;
import org.eclipse.core.runtime.Path;
import org.eclipse.core.runtime.Platform;
import org.eclipse.core.runtime.Status;
import org.eclipse.core.runtime.jobs.Job;
import org.eclipse.jdt.core.IJavaProject;
import org.eclipse.jdt.core.JavaCore;
import org.eclipse.jface.viewers.TreeViewer;
import org.eclipse.swt.widgets.Display;
import org.eclipse.ui.IPerspectiveDescriptor;
import org.eclipse.ui.IPerspectiveRegistry;
import org.eclipse.ui.IWorkbenchPage;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.PlatformUI;
import org.eclipse.ui.intro.IIntroManager;
import org.eclipse.ui.intro.IIntroPart;
import org.osgi.framework.Bundle;
import org.osgi.framework.BundleContext;
import org.osgi.framework.FrameworkUtil;
import org.osgi.framework.ServiceRegistration;
import org.osgi.util.function.Consumer;
import org.osgi.util.promise.Deferred;
import org.osgi.util.promise.Promise;
import org.osgi.util.promise.PromiseFactory;
import org.slf4j.LoggerFactory;

import aQute.bnd.annotation.plugin.InternalPluginDefinition;
import aQute.bnd.build.Project;
import aQute.bnd.build.Workspace;
import aQute.bnd.exceptions.BiFunctionWithException;
import aQute.bnd.exceptions.Exceptions;
import aQute.bnd.exceptions.FunctionWithException;
import aQute.bnd.exceptions.RunnableWithException;
import aQute.bnd.header.Attrs;
import aQute.bnd.memoize.Memoize;
import aQute.bnd.osgi.Constants;
import aQute.bnd.osgi.Processor;
import aQute.bnd.service.Refreshable;
import aQute.bnd.service.RepositoryPlugin;
import aQute.bnd.service.progress.ProgressPlugin.Task;
import aQute.bnd.service.progress.TaskManager;
import aQute.lib.io.IO;
import aQute.libg.ints.IntCounter;
import aQute.service.reporter.Reporter;
import bndtools.Plugin;
import bndtools.central.RepositoriesViewRefresher.RefreshModel;
import bndtools.central.sync.WorkspaceSynchronizer;
import bndtools.preferences.BndPreferences;

public class Central implements IStartupParticipant {

	private static final org.slf4j.Logger						logger						= LoggerFactory
		.getLogger(Central.class);
	private static volatile Central								instance					= null;
	private static final Deferred					anyWorkspaceDeferred		= promiseFactory()
		.deferred();
	private static volatile Deferred					cnfWorkspaceDeferred		= promiseFactory()
		.deferred();
	private static final Memoize						workspace					= Memoize
		.supplier(Central::createWorkspace);

	private static final Supplier	eclipseWorkspaceRepository	= Memoize
		.supplier(EclipseWorkspaceRepository::new);
	private final static AtomicBoolean							syncing						= new AtomicBoolean();

	private static Auxiliary									auxiliary;

	static final AtomicBoolean									indexValid					= new AtomicBoolean(false);

	private final BundleContext									bundleContext;
	private final Map					javaProjectToModel			= new HashMap<>();
	private final List							listeners					= new CopyOnWriteArrayList<>();
	private RepositoryListenerPluginTracker						repoListenerTracker;
	private final InternalPluginTracker							internalPlugins;

	@SuppressWarnings("unused")
	private static WorkspaceRepositoryChangeDetector			workspaceRepositoryChangeDetector;

	private static RepositoriesViewRefresher					repositoriesViewRefresher	= new RepositoriesViewRefresher();
	private static BundleContext								context;
	private static ServiceRegistration				workspaceService;

	static {
		try {
			context = FrameworkUtil.getBundle(Central.class)
				.getBundleContext();
			Bundle bndlib = FrameworkUtil.getBundle(Workspace.class);
			auxiliary = new Auxiliary(context, bndlib);

		} catch (Exception e) {
			// ignore
		}
	}

	/**
	 * WARNING: Do not instantiate this class. It must be public to allow
	 * instantiation by the Eclipse registry, but it is not intended for direct
	 * creation by clients. Instead call Central.getInstance().
	 */
	@Deprecated
	public Central() {
		bundleContext = FrameworkUtil.getBundle(Central.class)
			.getBundleContext();
		internalPlugins = new InternalPluginTracker(bundleContext);

	}

	@Override
	public void start() {
		instance = this;
		repoListenerTracker = new RepositoryListenerPluginTracker(bundleContext);
		repoListenerTracker.open();
		internalPlugins.open();
	}

	@Override
	public void stop() {
		repoListenerTracker.close();
		ServiceRegistration service = workspaceService;
		if (service != null) {
			service.unregister();
		}
		instance = null;

		Workspace ws = workspace.peek();
		if (ws != null) {
			ws.close();
		}

		if (auxiliary != null)
			try {
				auxiliary.close();
			} catch (Exception e) {
				throw new RuntimeException(e);
			}
		internalPlugins.close();
	}

	public static Central getInstance() {
		return instance;
	}

	public Project getModel(IJavaProject project) {
		try {
			Project model = javaProjectToModel.get(project);
			if (model == null) {
				try {
					model = getProject(project.getProject());
				} catch (IllegalArgumentException e) {
					// initialiseWorkspace();
					// model = Central.getProject(projectDir);
					return null;
				}
				if (model != null) {
					javaProjectToModel.put(project, model);
				}
			}
			return model;
		} catch (Exception e) {
			// TODO do something more useful here
			throw new RuntimeException(e);
		}
	}

	public static IFile getWorkspaceBuildFile() throws Exception {
		IWorkspaceRoot wsroot = ResourcesPlugin.getWorkspace()
			.getRoot();
		IProject cnf = wsroot.getProject(Workspace.CNFDIR);
		if (cnf == null || !cnf.isAccessible())
			return null;
		return cnf.getFile(Workspace.BUILDFILE);
	}

	public static EclipseWorkspaceRepository getEclipseWorkspaceRepository() {
		return eclipseWorkspaceRepository.get();
	}

	public synchronized static RepositoryPlugin getWorkspaceRepository() throws Exception {
		return getWorkspace().getWorkspaceRepository();
	}

	public static Workspace getWorkspaceIfPresent() {
		try {
			if (getInstance() == null)
				return null;
			return getWorkspace();
		} catch (IllegalStateException e) {
			throw e;
		} catch (Exception e) {
			return null;
		}
	}

	public static Workspace getWorkspace() throws Exception {
		if (getInstance() == null) {
			throw new IllegalStateException("Central is not initialised");
		}
		final Workspace ws;
		Consumer afterLock = null;
		synchronized (workspace) {
			if (workspace.peek() == null) { // No workspace has been created
				ws = workspace.get();
				// Resolve with new workspace
				afterLock = tryResolve(anyWorkspaceDeferred);
				if (!ws.isDefaultWorkspace()) {
					afterLock = afterLock.andThen(tryResolve(cnfWorkspaceDeferred));
				}
			} else {
				ws = workspace.get();
				// get the parent directory of the "cnf" project, if there is
				// one
				File workspaceDirectory = getWorkspaceDirectory();
				// Check to see if we need to adjust it...
				if (workspaceDirectory != null && !workspaceDirectory.equals(ws.getBase())) {
					// There is a "cnf" project and the current workspace is
					// not the same as the directory the cnf project is in,
					// so switch the workspace to the directory
					afterLock = Central::adjustWorkspace;
				} else if (workspaceDirectory == null && !ws.isDefaultWorkspace()) {
					// There is no "cnf" project and the current workspace is
					// not the default, so switch the workspace to the default
					afterLock = Central::adjustWorkspace;
				}
			}
		}
		if (afterLock != null) { // perform after lock action
			afterLock.accept(ws);
		}
		return ws;
	}

	private static void adjustWorkspace(Workspace ws) throws Exception {
		// Get write lock on another thread
		promiseFactory().submit(() -> {
			Consumer afterLock = ws.writeLocked(() -> {
				// get the parent directory of the "cnf" project, if there is
				// one
				File workspaceDirectory = getWorkspaceDirectory();
				// Check to see if we need to convert it...
				if (workspaceDirectory != null && !workspaceDirectory.equals(ws.getBase())) {
					// There is a "cnf" project and the current workspace is
					// not the same as the directory the cnf project is in,
					// so switch the workspace to the directory
					ws.setFileSystem(workspaceDirectory, Workspace.CNFDIR);
					ws.forceRefresh();
					ws.refresh();
					ws.refreshProjects();
					return tryResolve(cnfWorkspaceDeferred);
				} else if (workspaceDirectory == null && !ws.isDefaultWorkspace()) {
					// There is no "cnf" project and the current workspace is
					// not the default, so switch the workspace to the default
					cnfWorkspaceDeferred = promiseFactory().deferred();
					ws.setFileSystem(Workspace.BND_DEFAULT_WS, Workspace.CNFDIR);
					ws.forceRefresh();
					ws.refresh();
					ws.refreshProjects();
					return null;
				}
				return null;
			});
			if (afterLock != null) { // perform after lock action
				afterLock.accept(ws);
			}
			return ws;
		});
	}

	private static  Consumer tryResolve(Deferred deferred) {
		return value -> {
			try {
				deferred.resolve(value);
			} catch (IllegalStateException e) {
				// ignore race for already resolved
			}
		};
	}

	private static Workspace createWorkspace() {
		Workspace ws = null;
		try {
			Workspace.setDriver(Constants.BNDDRIVER_ECLIPSE);
			Workspace.addGestalt(Constants.GESTALT_INTERACTIVE, new Attrs());
			File workspaceDirectory = getWorkspaceDirectory();
			if (workspaceDirectory == null) {
				// There is no "cnf" project so we create a default
				// workspace
				ws = Workspace.createDefaultWorkspace();
			} else {
				// There is a "cnf" project so we create a normal
				// workspace
				ws = new Workspace(workspaceDirectory);
			}

			ws.setOffline(new BndPreferences().isWorkspaceOffline());

			ws.addBasicPlugin(new SWTClipboard());
			ws.addBasicPlugin(getInstance().repoListenerTracker);
			ws.addBasicPlugin(getEclipseWorkspaceRepository());
			ws.addBasicPlugin(new JobProgress());

			// Initialize projects in synchronized block
			ws.getBuildOrder();

			workspaceRepositoryChangeDetector = new WorkspaceRepositoryChangeDetector(ws);
			workspaceService = context.registerService(Workspace.class, ws, null);
			return ws;
		} catch (Exception e) {
			if (ws != null) {
				ws.close();
			}
			logger.error("Workspace creation failure", e);
			throw Exceptions.duck(e);
		}
	}

	public static Promise onAnyWorkspace(Consumer callback) {
		return callback(anyWorkspaceDeferred.getPromise(), callback, "onAnyWorkspace callback failed");
	}

	public static Promise onCnfWorkspace(Consumer callback) {
		return callback(cnfWorkspaceDeferred.getPromise(), callback, "onCnfWorkspace callback failed");
	}

	private static Promise callback(Promise promise, Consumer callback,
		String failureMessage) {
		return promise.thenAccept(callback)
			.onFailure(failure -> logger.error(failureMessage, failure));
	}

	public static Promise onAnyWorkspaceAsync(Consumer callback) {
		return callbackAsync(anyWorkspaceDeferred.getPromise(), callback, "onAnyWorkspaceAsync callback failed");
	}

	public static Promise onCnfWorkspaceAsync(Consumer callback) {
		return callbackAsync(cnfWorkspaceDeferred.getPromise(), callback, "onCnfWorkspaceAsync callback failed");
	}

	private static Promise callbackAsync(Promise promise, Consumer callback,
		String failureMessage) {
		return promise.then(resolved -> {
			Workspace workspace = resolved.getValue();
			Deferred completion = promiseFactory().deferred();
			Display.getDefault()
				.asyncExec(() -> {
					try {
						callback.accept(workspace);
						completion.resolve(workspace);
					} catch (Throwable e) {
						completion.fail(e);
					}
				});
			return completion.getPromise();
		})
			.onFailure(failure -> logger.error(failureMessage, failure));
	}

	public static PromiseFactory promiseFactory() {
		return Processor.getPromiseFactory();
	}

	public static boolean hasAnyWorkspace() {
		return anyWorkspaceDeferred.getPromise()
			.isDone();
	}

	public static boolean hasCnfWorkspace() {
		return cnfWorkspaceDeferred.getPromise()
			.isDone();
	}

	/**
	 * Returns the Bnd Workspace directory IF there is a "cnf" project
	 * in the Eclipse workspace.
	 *
	 * @return The returned directory is the parent of the "cnf" project's
	 *         directory. Otherwise, {@code null}.
	 */
	private static File getWorkspaceDirectory() throws CoreException {
		IWorkspaceRoot eclipseWorkspace = ResourcesPlugin.getWorkspace()
			.getRoot();

		IProject cnfProject = eclipseWorkspace.getProject(Workspace.CNFDIR);
		if (cnfProject.exists()) {
			if (!cnfProject.isOpen())
				cnfProject.open(null);
			return cnfProject.getLocation()
				.toFile()
				.getParentFile();
		}

		String path = Platform.getInstanceLocation()
			.getURL()
			.getPath();

		if (IO.isWindows() && path.startsWith("/"))
			path = path.substring(1);

		File folder = new File(path);
		File build = IO.getFile(folder, "cnf/build.bnd");
		if (build.isFile()) {
			if (syncing.getAndSet(true) == false) {
				Job job = Job.create("sync ws", mon -> {
					WorkspaceSynchronizer wss = new WorkspaceSynchronizer();
					wss.synchronize(false, mon, () -> {
						syncing.set(false);
					});
					setBndtoolsPerspective();
					final IIntroManager introManager = PlatformUI.getWorkbench()
						.getIntroManager();
					IIntroPart part = introManager.getIntro();
					introManager.closeIntro(part);
				});
				job.schedule();
			}
			return folder;
		}

		return null;
	}

	public static void setBndtoolsPerspective() {
		Display.getDefault()
			.syncExec(() -> {
				IWorkbenchWindow window = PlatformUI.getWorkbench()
					.getActiveWorkbenchWindow();
				if (window != null) {
					IWorkbenchPage page = window.getActivePage();
					IPerspectiveRegistry reg = PlatformUI.getWorkbench()
						.getPerspectiveRegistry();
					// Replace "your.perspective.id" with the
					// actual ID
					// of
					// the perspective you want to switch to
					IPerspectiveDescriptor bndtools = reg.findPerspectiveWithId("bndtools.perspective");
					if (bndtools != null)
						page.setPerspective(bndtools);

					return;
				}
			});
	}

	/**
	 * Determine if the given directory is a workspace.
	 *
	 * @param directory the directory that must hold cnf/build.bnd
	 * @return true if a workspace directory
	 */
	public static boolean isWorkspace(File directory) {
		File build = IO.getFile(directory, "cnf/build.bnd");
		return build.isFile();
	}

	public static boolean hasWorkspaceDirectory() {
		try {
			return getWorkspaceDirectory() != null;
		} catch (CoreException e) {
			return false;
		}
	}

	public static boolean isChangeDelta(IResourceDelta delta) {
		if (IResourceDelta.MARKERS == delta.getFlags())
			return false;
		if ((delta.getKind() & (IResourceDelta.ADDED | IResourceDelta.CHANGED | IResourceDelta.REMOVED)) == 0)
			return false;
		return true;
	}

	public void changed(Project model) {
		model.setChanged();
		for (ModelListener m : listeners) {
			try {
				m.modelChanged(model);
			} catch (Exception e) {
				logger.error("While notifying ModelListener {} of change to project {}", m, model, e);
			}
		}
	}

	public void addModelListener(ModelListener m) {
		if (!listeners.contains(m)) {
			listeners.add(m);
		}
	}

	public void removeModelListener(ModelListener m) {
		listeners.remove(m);
	}

	public static IJavaProject getJavaProject(Project model) {
		return getProject(model).map(JavaCore::create)
			.filter(IJavaProject::exists)
			.orElse(null);
	}

	public static Optional getProject(Project model) {
		String name = model.getName();
		return Arrays.stream(ResourcesPlugin.getWorkspace()
			.getRoot()
			.getProjects())
			.filter(p -> p.getName()
				.equals(name))
			.findFirst();
	}

	public static IPath toPath(File file) throws Exception {
		File absolute = file.getCanonicalFile();
		return toFullPath(absolute).orElseGet(() -> {
			try {
				String workspacePath = getWorkspace().getBase()
					.getAbsolutePath();
				String absolutePath = absolute.getPath();
				if (absolutePath.startsWith(workspacePath))
					return new Path(absolutePath.substring(workspacePath.length()));
				return null;
			} catch (Exception e) {
				throw Exceptions.duck(e);
			}
		});
	}

	public static IPath toPathMustBeInEclipseWorkspace(File file) throws Exception {
		File absolute = file.getCanonicalFile();
		return toFullPath(absolute).orElse(null);
	}

	private static Optional toFullPath(File file) {
		IWorkspaceRoot wsroot = ResourcesPlugin.getWorkspace()
			.getRoot();
		IFile[] candidates = wsroot.findFilesForLocationURI(file.toURI());
		return Stream.of(candidates)
			.map(IFile::getFullPath)
			.min((a, b) -> Integer.compare(a.segmentCount(), b.segmentCount()));
	}

	public static Optional toBestPath(IResource resource) {
		return Optional.ofNullable(resource.getLocationURI())
			.map(File::new)
			.flatMap(Central::toFullPath);
	}

	public static void refresh(IPath path) {
		try {
			IResource r = ResourcesPlugin.getWorkspace()
				.getRoot()
				.findMember(path);
			if (r != null)
				return;

			IPath p = (IPath) path.clone();
			while (p.segmentCount() > 0) {
				p = p.removeLastSegments(1);
				IResource resource = ResourcesPlugin.getWorkspace()
					.getRoot()
					.findMember(p);
				if (resource != null) {
					resource.refreshLocal(IResource.DEPTH_INFINITE, null);
					return;
				}
			}
		} catch (Exception e) {
			logger.error("While refreshing path {}", path, e);
		}
	}

	public static void refreshPlugins() throws Exception {
		List refreshedFiles = new ArrayList<>();
		List rps = getWorkspace().getPlugins(Refreshable.class);
		boolean changed = false;
		boolean repoChanged = false;
		for (Refreshable rp : rps) {
			if (rp.refresh()) {
				changed = true;
				File root = rp.getRoot();
				if (root != null)
					refreshedFiles.add(root);
				if (rp instanceof RepositoryPlugin) {
					repoChanged = true;
				}
			}
		}

		//
		// If repos were refreshed then
		// we should also update the classpath
		// containers. We can force this by setting the "bndtools.refresh"
		// property.
		//

		if (changed) {
			for (File file : refreshedFiles) {
				refreshFile(file);
			}

			if (repoChanged) {
				repositoriesViewRefresher.repositoriesRefreshed();
			}
			refreshProjects();
		}
	}

	public static void refreshPlugin(Refreshable plugin) throws Exception {
		refreshPlugin(plugin, false);
	}

	public static void refreshPlugin(Refreshable plugin, boolean force) throws Exception {
		boolean refresh = plugin.refresh();
		if (refresh || force) {
			refreshFile(plugin.getRoot());
			if (plugin instanceof RepositoryPlugin) {
				repositoriesViewRefresher.repositoryRefreshed((RepositoryPlugin) plugin);
			}
			refreshProjects();
		}
	}

	public static void refreshProjects() throws Exception {
		Collection allProjects = getWorkspace().getAllProjects();
		// Mark all projects changed before we notify model listeners
		// since the listeners can take actions on project's other than
		// the specified project.
		for (Project p : allProjects) {
			p.setChanged();
		}
		for (Project p : allProjects) {
			for (ModelListener m : getInstance().listeners) {
				try {
					m.modelChanged(p);
				} catch (Exception e) {
					logger.error("While notifying ModelListener {} of change to project {}", m, p, e);
				}
			}
		}
	}

	public static void refreshFile(File f) throws CoreException {
		refreshFile(f, null, false);
	}

	public static void refreshFile(File file, IProgressMonitor monitor, boolean derived) throws CoreException {
		IResource target = toResource(file);
		if (target == null) {
			return;
		}
		int depth = target.getType() == IResource.FILE ? IResource.DEPTH_ZERO : IResource.DEPTH_INFINITE;
		if (!target.isSynchronized(depth)) {
			target.refreshLocal(depth, monitor);
			if (target.exists() && (target.isDerived() != derived)) {
				target.setDerived(derived, monitor);
			}
		}
	}

	public static void refresh(Project p) throws Exception {
		IJavaProject jp = getJavaProject(p);
		if (jp != null)
			jp.getProject()
				.refreshLocal(IResource.DEPTH_INFINITE, null);
	}

	public void close() {
		repositoriesViewRefresher.close();
	}

	public static void invalidateIndex() {
		indexValid.set(false);
	}

	public static boolean needsIndexing() {
		return indexValid.compareAndSet(false, true);
	}

	public static Project getProject(File projectDir) throws Exception {
		return getWorkspace().getProjectFromFile(projectDir);
	}

	public static Project getProject(IProject p) throws Exception {
		return Optional.ofNullable(p.getLocation())
			.map(IPath::toFile)
			.map(asFunction(Central::getProject))
			.orElse(null);
	}

	public static boolean isBndProject(IProject project) {
		return Optional.ofNullable(project)
			.map(asFunction(p -> p.getNature(Plugin.BNDTOOLS_NATURE)))
			.isPresent();
	}

	/**
	 * Return the IResource associated with a file
	 *
	 * @param file
	 */

	public static IResource toResource(File file) {
		if (file == null)
			return null;

		IWorkspaceRoot root = ResourcesPlugin.getWorkspace()
			.getRoot();
		return toFullPath(file).map(p -> file.isDirectory() ? root.getFolder(p) : root.getFile(p))
			.orElse(null);
	}

	/**
	 * Used to serialize access to the Bnd Workspace.
	 *
	 * @param lockMethod The Workspace lock method to use.
	 * @param callable The code to execute while holding the lock. The argument
	 *            can be used to register after lock release actions.
	 * @param monitorOrNull If the monitor is cancelled, a TimeoutException will
	 *            be thrown, can be null
	 * @return The result of the specified callable.
	 * @throws InterruptedException If the thread is interrupted while waiting
	 *             for the lock.
	 * @throws TimeoutException If the lock was not obtained within the timeout
	 *             period or the specified monitor is cancelled while waiting to
	 *             obtain the lock.
	 * @throws Exception If the callable throws an exception.
	 */
	public static  V bndCall(BiFunctionWithException, BooleanSupplier, V> lockMethod,
		FunctionWithException, V> callable, IProgressMonitor monitorOrNull)
		throws Exception {
		IProgressMonitor monitor = monitorOrNull == null ? new NullProgressMonitor() : monitorOrNull;
		Task task = new Task() {
			@Override
			public void worked(int units) {
				monitor.worked(units);
			}

			@Override
			public void done(String message, Throwable e) {}

			@Override
			public boolean isCanceled() {
				return monitor.isCanceled();
			}

			@Override
			public void abort() {
				monitor.setCanceled(true);
			}
		};
		List after = new ArrayList<>();
		MultiStatus status = new MultiStatus(Plugin.PLUGIN_ID, 0,
			"Errors occurred while calling bndCall after actions");
		try {
			Callable with = () -> TaskManager.with(task, () -> callable.apply((name, runnable) -> after.add(() -> {
				monitor.subTask(name);
				try {
					runnable.run();
				} catch (Exception e) {
					if (!(e instanceof OperationCanceledException)) {
						status.add(new Status(IStatus.ERROR, runnable.getClass(),
							"Unexpected exception in bndCall after action: " + name, e));
					}
				}
			})));
			return lockMethod.apply(with, monitor::isCanceled);
		} finally {
			for (Runnable runnable : after) {
				runnable.run();
			}
			if (!status.isOK()) {
				throw new CoreException(status);
			}
		}
	}

	/**
	 * Convert a processor to a status object
	 */

	public static IStatus toStatus(Processor processor, String message) {
		int severity = IStatus.INFO;
		List statuses = new ArrayList<>();
		for (String error : processor.getErrors()) {
			Status status = new Status(IStatus.ERROR, BndtoolsConstants.CORE_PLUGIN_ID, error);
			statuses.add(status);
			severity = IStatus.ERROR;
		}
		for (String warning : processor.getWarnings()) {
			Status status = new Status(IStatus.WARNING, BndtoolsConstants.CORE_PLUGIN_ID, warning);
			statuses.add(status);
			severity = IStatus.WARNING;
		}

		IStatus[] array = statuses.toArray(new IStatus[0]);
		return new MultiStatus(//
			BndtoolsConstants.CORE_PLUGIN_ID, //
			severity, //
			array, message, null);
	}

	/**
	 * Register a viewer with repositories
	 */

	public static void addRepositoriesViewer(TreeViewer viewer, RepositoriesViewRefresher.RefreshModel model) {
		repositoriesViewRefresher.addViewer(viewer, model);
	}

	/**
	 * Unregister a viewer with repositories
	 */
	public static void removeRepositoriesViewer(TreeViewer viewer) {
		repositoriesViewRefresher.removeViewer(viewer);
	}

	public static void setRepositories(TreeViewer viewer, RefreshModel model) {
		repositoriesViewRefresher.setRepositories(viewer, model);
	}

	public static boolean refreshFiles(Reporter reporter, Collection files, IProgressMonitor monitor,
		boolean derived) {
		IntCounter errors = new IntCounter();

		files.forEach(t -> {
			try {
				Central.refreshFile(t, monitor, derived);
			} catch (CoreException e) {
				errors.inc();
				if (reporter != null)
					reporter.error("failed to refresh %s : %s", t, Exceptions.causes(e));
				else
					throw Exceptions.duck(e);
			}
		});
		return errors.isZero();
	}

	public static List getInternalPluginDefinitions() {
		return instance.internalPlugins.getTracked()
			.values()
			.stream()
			.flatMap(Collection::stream)
			.collect(Collectors.toList());
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy