blob: 8d235a9f57519612e122bc560f0e5e15dc6188a5 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2003, 2007 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.wst.common.componentcore.internal.util;
import java.util.Hashtable;
import java.util.Iterator;
import java.util.Map;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.resources.IResourceDelta;
import org.eclipse.core.runtime.IConfigurationElement;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.ISafeRunnable;
import org.eclipse.core.runtime.Path;
import org.eclipse.core.runtime.SafeRunner;
import org.eclipse.jem.util.RegistryReader;
import org.eclipse.jem.util.emf.workbench.ISynchronizerExtender;
import org.eclipse.jem.util.emf.workbench.ProjectResourceSet;
import org.eclipse.jem.util.emf.workbench.WorkbenchResourceHelperBase;
import org.eclipse.wst.common.componentcore.ModuleCoreNature;
import org.eclipse.wst.common.componentcore.internal.ModulecorePlugin;
import org.eclipse.wst.common.componentcore.internal.resources.ResourceTimestampMappings;
import org.eclipse.wst.common.componentcore.internal.resources.VirtualArchiveComponent;
import org.eclipse.wst.common.componentcore.internal.resources.VirtualComponent;
import org.eclipse.wst.common.componentcore.internal.resources.VirtualFolder;
import org.eclipse.wst.common.componentcore.resources.IVirtualComponent;
import org.eclipse.wst.common.componentcore.resources.IVirtualFolder;
import org.eclipse.wst.common.project.facet.core.IFacetedProject;
import org.eclipse.wst.common.project.facet.core.IProjectFacet;
import org.eclipse.wst.common.project.facet.core.ProjectFacetsManager;
public class ComponentImplManager implements ISynchronizerExtender{
private static final String NO_FACETS = "NONE";//$NON-NLS-1$
private static final String COMPONENT_IMPL_EXTENSION_POINT = "componentimpl"; //$NON-NLS-1$
private static final String TAG_COMPONENT_IMPL = "componentimpl"; //$NON-NLS-1$
private static final String ATT_TYPE = "typeID"; //$NON-NLS-1$
private static final String ATT_CLASS = "class"; //$NON-NLS-1$
private static final ComponentImplManager instance = new ComponentImplManager();
// private static final Object LOAD_FAILED = new Object();
private final Map/* <String, ComponentImplDescriptor> */ descriptors = new Hashtable();
private final Map/* <ComponentImplDescriptor, IComponentImplFactory> */ instances = new Hashtable();
/**
* @return Returns the instance.
*/
public static ComponentImplManager instance() {
/* already initialized and registry read by the time the class initializes */
return instance;
}
public ComponentImplManager() {
SafeRunner.run(new ISafeRunnable() {
public void handleException(Throwable exception) {
ModulecorePlugin.logError(0, exception.getMessage(), exception);
}
public void run() throws Exception {
new ComponentImplRegistryReader().readRegistry();
}
});
}
private IComponentImplFactory getComponentImplFactory(String typeID) {
ComponentImplDescriptor descriptor = (ComponentImplDescriptor) descriptors.get(typeID);
IComponentImplFactory factory = null;
if (descriptor != null) {
factory = (IComponentImplFactory) instances.get(descriptor);
if (factory == null) {
if ((factory = descriptor.createFactory()) != null) {
instances.put(descriptor, factory);
} else {
descriptors.remove(descriptor);
}
}
}
return factory;
}
private IComponentImplFactory findFactoryForProject(IProject project, Map descriptors){
try {
IComponentImplFactory factory = ComponentCacheManager.instance().getComponentImplFactory(project);
if(factory != null)
return factory;
IFacetedProject facetedProject = ProjectFacetsManager.create(project);
if (facetedProject == null){
factory = getComponentImplFactory(NO_FACETS);
ComponentCacheManager.instance().setComponentImplFactory(project, factory);
return factory;
}
Iterator keys = descriptors.keySet().iterator();
while (keys.hasNext()) {
String typeID = (String) keys.next();
if(typeID.equals(NO_FACETS)){
continue;
}
try {
IProjectFacet projectFacet = ProjectFacetsManager.getProjectFacet(typeID);
if (projectFacet != null && facetedProject.hasProjectFacet(projectFacet)){
factory = getComponentImplFactory(typeID);
if(null != factory){
ComponentCacheManager.instance().setComponentImplFactory(project, factory);
return factory;
}
}
} catch (Exception e) {
continue;
}
}
} catch (Exception e) {
ModulecorePlugin.logError(0, "Returning null factory for project: " + project, e); //$NON-NLS-1$
ComponentCacheManager.instance().markErrorComponentImplFactory(project);
}
return null;
}
public IVirtualFolder createFolder(IProject aProject, IPath aRuntimePath){
try {
IComponentImplFactory factory = findFactoryForProject(aProject, descriptors);
if(null != factory){
return factory.createFolder(aProject, aRuntimePath);
}
} catch (Exception e) {
// Just return a default folder
}
ComponentCacheManager.instance().setComponentImplFactory(aProject, null);
return new VirtualFolder(aProject, aRuntimePath);
}
public IVirtualComponent createComponent(IProject project) {
return createComponent(project, true);
}
public IVirtualComponent createComponent(IProject project, boolean checkSettings) {
try {
IVirtualComponent component = ComponentCacheManager.instance().getComponent(project);
if(component != null) {
return component;
}
IComponentImplFactory factory = findFactoryForProject(project, descriptors);
if(null != factory){
component = factory.createComponent(project);
if(component != null) {
ComponentCacheManager.instance().setComponent(project, component);
registerListener(project);
}
return component;
}
} catch (Exception e) {
// Just return a default component
}
if(checkSettings) {
if (!ModuleCoreNature.isFlexibleProject(project)){
return null;
}
}
else {
if (ModuleCoreNature.getModuleCoreNature(project) == null){
return null;
}
}
IVirtualComponent component = new VirtualComponent(project, new Path("/")); //$NON-NLS-1$
if(component != null) {
ComponentCacheManager.instance().setComponentImplFactory(project, null);
ComponentCacheManager.instance().setComponent(project, component);
registerListener(project);
}
return component;
}
public IVirtualComponent createArchiveComponent(IProject aProject, String aComponentName) {
try {
IVirtualComponent component = ComponentCacheManager.instance().getArchiveComponent(aProject, aComponentName);
if(component != null)
return component;
if(!ComponentCacheManager.instance().isValidComponentImplFactory(aProject)) {
registerListener(aProject);
}
IComponentImplFactory factory = findFactoryForProject(aProject, descriptors);
if(null != factory){
IVirtualComponent archiveComponent = factory.createArchiveComponent(aProject, aComponentName, new Path("/")); //$NON-NLS-1$
ComponentCacheManager.instance().setArchiveComponent(aProject, aComponentName, archiveComponent);
return archiveComponent;
}
} catch (Exception e) {
// Just return a default archive component
}
ComponentCacheManager.instance().setComponentImplFactory(aProject, null);
IVirtualComponent archiveComponent = new VirtualArchiveComponent(aProject, aComponentName, new Path("/")); //$NON-NLS-1$
ComponentCacheManager.instance().setArchiveComponent(aProject, aComponentName, archiveComponent);
return archiveComponent;
}
private void registerListener(IProject aProject) {
ProjectResourceSet resSet = getResourceSet(aProject);
if (resSet == null)
return;
resSet.getSynchronizer().addExtender(this);
}
protected ProjectResourceSet getResourceSet(IProject proj) {
return (ProjectResourceSet)WorkbenchResourceHelperBase.getResourceSet(proj);
}
public void projectChanged(IResourceDelta delta) {
// TODO Auto-generated method stub
}
public synchronized void projectClosed() {
ComponentCacheManager.instance().clearCache();
}
private static class ComponentCacheManager {
private static final ComponentCacheManager instance = new ComponentCacheManager();
private final ResourceTimestampMappings factoryMap = new ResourceTimestampMappings();
private final Map <IProject , IVirtualComponent> componentsMap = new Hashtable<IProject , IVirtualComponent>();
private final Map <IProject , Map<String, IVirtualComponent>> componentsArchivesMap = new Hashtable<IProject , Map<String, IVirtualComponent>>();
private Object cacheLock = new Object();
public ComponentCacheManager() {}
public static ComponentCacheManager instance() {
return instance;
}
public IComponentImplFactory getComponentImplFactory(IProject project) {
synchronized (cacheLock) {
if(isValidComponentImplFactory(project)) {
Object data = factoryMap.getData(project);
if(data instanceof IComponentImplFactory)
return (IComponentImplFactory) data;
}
return null;
}
}
public boolean isValidComponentImplFactory(IProject project) {
synchronized (cacheLock) {
if(!factoryMap.hasChanged(project) && !factoryMap.hasCacheError(project) && factoryMap.hasCacheData(project))
return true;
return false;
}
}
public void setComponentImplFactory(IProject project, IComponentImplFactory factory){
synchronized (cacheLock) {
if(factory != null) {
factoryMap.mark(project, factory);
}
else {
factoryMap.mark(project, project);
}
}
}
public void markErrorComponentImplFactory(IProject project){
synchronized (cacheLock) {
factoryMap.markError(project);
}
}
public IVirtualComponent getComponent(IProject project) {
synchronized (cacheLock) {
if(componentsMap.containsKey(project)) {
if(isValidComponentImplFactory(project)) {
return componentsMap.get(project);
} else {
componentsMap.remove(project);
}
}
return null;
}
}
public void setComponent(IProject project, IVirtualComponent component) {
synchronized (cacheLock) {
if(component != null)
componentsMap.put(project, component);
}
}
public IVirtualComponent getArchiveComponent(IProject project, String componentName) {
synchronized (cacheLock) {
Map archives = getComponentArchives(project);
if(isValidComponentImplFactory(project)) {
if(archives.containsKey(componentName)) {
return (IVirtualComponent) archives.get(componentName);
}
}
else {
archives = new Hashtable<String, IVirtualComponent>();
componentsArchivesMap.put(project, archives);
}
return null;
}
}
public Map getComponentArchives(IProject project) {
synchronized (cacheLock) {
Map archives = componentsArchivesMap.get(project);
if(archives == null) {
archives = new Hashtable<String, IVirtualComponent>();
componentsArchivesMap.put(project, archives);
}
return archives;
}
}
public void setArchiveComponent(IProject project, String componentName, IVirtualComponent archiveComponent) {
synchronized (cacheLock) {
if(archiveComponent != null) {
Map archives = ComponentCacheManager.instance().getComponentArchives(project);
archives.put(componentName, archiveComponent);
}
}
}
public void clearCache() {
Object[] components = null;
Object[] componentsArchives = null;
synchronized (cacheLock) {
components = (Object[]) componentsMap.values().toArray();
componentsMap.clear();
componentsArchives = (Object[]) componentsArchivesMap.values().toArray();
componentsArchivesMap.clear();
}
for(int i = 0; i < components.length; i++) {
if(components[i] instanceof VirtualComponent)
((VirtualComponent)components[i]).dispose();
}
for(int i = 0; i < componentsArchives.length; i++) {
if(componentsArchives[i] instanceof VirtualComponent)
((VirtualComponent)componentsArchives[i]).dispose();
}
}
}
private class ComponentImplDescriptor {
private final IConfigurationElement element;
private final String type;
public ComponentImplDescriptor(IConfigurationElement configElement) {
element = configElement;
type = element.getAttribute(ATT_TYPE);
}
/**
* Create and return an {@link IArtifactEditFactory} for the given descriptor or <b>null</b>
* if there are problems instantiating the extension.
*
* @return An {@link IArtifactEditFactory} for the given descriptor or <b>null</b> if there
* are problems instantiating the extension.
*/
public IComponentImplFactory createFactory() {
final IComponentImplFactory[] factory = new IComponentImplFactory[1];
SafeRunner.run(new ISafeRunnable() {
public void handleException(Throwable exception) {
ModulecorePlugin.logError(0, exception.getMessage(), exception);
}
public void run() throws Exception {
factory[0] = (IComponentImplFactory) element.createExecutableExtension(ATT_CLASS);
}
});
return factory[0];
}
/**
*
* @return The type id of this ArtifactEdit definition
*/
public String getType() {
return type;
}
}
private class ComponentImplRegistryReader extends RegistryReader {
public ComponentImplRegistryReader() {
super(ModulecorePlugin.PLUGIN_ID, COMPONENT_IMPL_EXTENSION_POINT);
}
/**
* @see org.eclipse.wst.common.frameworks.internal.RegistryReader#readElement(org.eclipse.core.runtime.IConfigurationElement)
*/
public boolean readElement(IConfigurationElement element) {
if (TAG_COMPONENT_IMPL.equals(element.getName())) {
/*
* Because the only instance of this type is created from a static singleton field, and
* the registry is initialized in the constructor of this type, other threads cannot
* compete with readElement() for access to <i>descriptors</i>
*/
String type = element.getAttribute(ATT_TYPE);
if (type != null)
descriptors.put(element.getAttribute(ATT_TYPE), new ComponentImplDescriptor(element));
else
ModulecorePlugin.logError(0, "No type attribute is specified for " + //$NON-NLS-1$
ModulecorePlugin.PLUGIN_ID + "." + COMPONENT_IMPL_EXTENSION_POINT + //$NON-NLS-1$
" extension in " + element.getDeclaringExtension().getNamespaceIdentifier(), null); //$NON-NLS-1$
return true;
}
return false;
}
}
}