blob: bae6f5bf4d6591955e1431325e59e25db05dc6c2 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2001, 2020 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* IBM Corporation - initial API and implementation
* Jens Lukowski/Innoopract - initial renaming/restructuring
* David Carver (Intalio) - bug 300434 - Make inner classes static where
* possible
*
*******************************************************************************/
package org.eclipse.wst.sse.core.internal.tasks;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IConfigurationElement;
import org.eclipse.core.runtime.IExtensionPoint;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.Platform;
import org.eclipse.core.runtime.content.IContentType;
import org.eclipse.core.runtime.content.IContentTypeManager;
import org.eclipse.wst.sse.core.internal.Logger;
import org.eclipse.wst.sse.core.internal.SSECorePlugin;
import org.eclipse.wst.sse.core.internal.provisional.tasks.IFileTaskScanner;
import org.eclipse.wst.sse.core.utils.StringUtils;
public class FileTaskScannerRegistryReader {
static class EnabledDisabledScanners {
public IFileTaskScanner[] enabled;
public IFileTaskScanner[] disabled;
EnabledDisabledScanners(IFileTaskScanner[] enabled, IFileTaskScanner[] disabled) {
this.enabled = enabled;
this.disabled = disabled;
}
}
private static class ScannerInfo {
String fId;
IFileTaskScanner fScanner;
ScannerInfo(String id, IFileTaskScanner scanner) {
super();
fId = id;
fScanner = scanner;
}
public boolean equals(Object obj) {
return obj instanceof ScannerInfo && fId.equals(((ScannerInfo) obj).fId);
}
public IFileTaskScanner getScanner() {
return fScanner;
}
}
private static FileTaskScannerRegistryReader _instance = null;
public static FileTaskScannerRegistryReader getInstance() {
if (_instance == null) {
_instance = new FileTaskScannerRegistryReader();
}
return _instance;
}
private String ATT_CLASS = "class"; //$NON-NLS-1$
private String ATT_ID = "id"; //$NON-NLS-1$
private String ATT_CONTENT_TYPES = "contentTypeIds"; //$NON-NLS-1$
private String ATT_FILENAME_EXTENSIONS = "fileExtensions"; //$NON-NLS-1$
private IConfigurationElement[] fScannerElements;
// a mapping from content types to ScannerInfo instances
private Map<String, ScannerInfo[]> fScannerInfoByContentTypes = null;
// a mapping from filename extensions to scanner configuration elements
private Map<String, IConfigurationElement[]> fConfigurationElementsByFilenameExtensions = null;
private String NAME_SCANNER = "scanner"; //$NON-NLS-1$
private String SCANNER_EXTENSION_POINT_ID = SSECorePlugin.ID + ".taskscanner"; //$NON-NLS-1$
private FileTaskScannerRegistryReader() {
super();
}
IFileTaskScanner[] getFileTaskScanners(IContentType[] contentTypes) {
if (fScannerElements == null) {
readRegistry();
}
List<ScannerInfo> scannerInfos = new ArrayList<>(1);
for (int i = 0; i < contentTypes.length; i++) {
ScannerInfo[] scannerInfosForContentType = fScannerInfoByContentTypes.get(contentTypes[i].getId());
if (scannerInfosForContentType == null) {
scannerInfosForContentType = loadScanners(contentTypes[i]);
}
// only add non-duplicate scanners
for (int j = 0; j < scannerInfosForContentType.length; j++) {
if (!scannerInfos.contains(scannerInfosForContentType[j])) {
scannerInfos.add(scannerInfosForContentType[j]);
}
}
}
IFileTaskScanner[] scanners = new IFileTaskScanner[scannerInfos.size()];
for (int i = 0; i < scanners.length; i++) {
scanners[i] = scannerInfos.get(i).getScanner();
}
return scanners;
}
EnabledDisabledScanners getFileTaskScanners(IPath filePath, IContentType[] exceptForContentTypeIds) {
if (fScannerElements == null) {
readRegistry();
}
String fileExtension = filePath.getFileExtension();
// what if there is no filename extension?
if (fileExtension == null) {
return new EnabledDisabledScanners(new IFileTaskScanner[0], new IFileTaskScanner[0]);
}
String loweredExtension = fileExtension.toLowerCase(Locale.US);
IConfigurationElement[] matchingElements = fConfigurationElementsByFilenameExtensions.get(loweredExtension);
if (matchingElements == null) {
Set<String> classnames = new HashSet<>();
List<IConfigurationElement> collectedMatchingElements = new ArrayList<>(0);
for (int i = 0; i < fScannerElements.length; i++) {
if (!fScannerElements[i].getName().equals(NAME_SCANNER))
continue;
String[] supportedFilenameExtensions = StringUtils.unpack(fScannerElements[i].getAttribute(ATT_FILENAME_EXTENSIONS));
for (int k = 0; k < supportedFilenameExtensions.length; k++) {
String scannerClassName = fScannerElements[i].getAttribute(ATT_CLASS);
if (scannerClassName != null && loweredExtension.equals(supportedFilenameExtensions[k].toLowerCase(Locale.US)) && !classnames.contains(scannerClassName)) {
classnames.add(scannerClassName);
collectedMatchingElements.add(fScannerElements[i]);
}
}
}
matchingElements = collectedMatchingElements.toArray(new IConfigurationElement[collectedMatchingElements.size()]);
fConfigurationElementsByFilenameExtensions.put(loweredExtension, matchingElements);
}
// instantiate and return the scanners
List<IFileTaskScanner> enabled = new ArrayList<>();
List<IFileTaskScanner> disabled = new ArrayList<>();
for (int i = 0; i < matchingElements.length; i++) {
boolean disallowed = false;
if (exceptForContentTypeIds.length > 0) {
String[] supportedContentTypeIds = StringUtils.unpack(matchingElements[i].getAttribute(ATT_CONTENT_TYPES));
for (int j = 0; !disallowed && j < supportedContentTypeIds.length; j++) {
for (int k = 0; !disallowed && k < exceptForContentTypeIds.length; k++) {
if (supportedContentTypeIds[j].equals(exceptForContentTypeIds[k].getId()))
disallowed = true;
}
}
}
try {
IFileTaskScanner scanner = (IFileTaskScanner) matchingElements[i].createExecutableExtension(ATT_CLASS);
if (disallowed) {
disabled.add(scanner);
}
else {
enabled.add(scanner);
}
}
catch (CoreException e) {
Logger.logException("Non-fatal exception creating file task scanner for " + filePath + " from " + matchingElements[i].getContributor().getName(), e); //$NON-NLS-1$
}
}
return new EnabledDisabledScanners(enabled.toArray(new IFileTaskScanner[enabled.size()]), disabled.toArray(new IFileTaskScanner[disabled.size()]));
}
public String[] getSupportedContentTypeIds() {
if (fScannerElements == null) {
readRegistry();
}
// find the relevant extensions
List<String> types = new ArrayList<>(0);
IConfigurationElement[] scannerElements = fScannerElements;
for (int j = 0; j < scannerElements.length; j++) {
if (!scannerElements[j].getName().equals(NAME_SCANNER))
continue;
String[] contentTypeIds = StringUtils.unpack(scannerElements[j].getAttribute(ATT_CONTENT_TYPES));
for (int i = 0; i < contentTypeIds.length; i++) {
if (!types.contains(contentTypeIds[i])) {
types.add(contentTypeIds[i]);
}
}
}
return types.toArray(new String[types.size()]);
}
private ScannerInfo[] loadScanners(IContentType contentType) {
List<IConfigurationElement> elements = new ArrayList<>(0);
ScannerInfo[] scannerInfos = null;
IConfigurationElement[] delegateElements = fScannerElements;
if (contentType != null) {
IContentTypeManager contentTypeManager = Platform.getContentTypeManager();
for (int j = 0; j < delegateElements.length; j++) {
if (!delegateElements[j].getName().equals(NAME_SCANNER))
continue;
String[] supportedContentTypeIds = StringUtils.unpack(delegateElements[j].getAttribute(ATT_CONTENT_TYPES));
IContentType[] supportedContentTypes = new IContentType[supportedContentTypeIds.length];
for (int k = 0; k < supportedContentTypeIds.length; k++) {
supportedContentTypes[k] = contentTypeManager.getContentType(supportedContentTypeIds[k].trim());
}
for (int k = 0; k < supportedContentTypeIds.length; k++) {
// allow subtypes to be returned as well
if (supportedContentTypes[k] != null && contentType.isKindOf(supportedContentTypes[k])) {
elements.add(delegateElements[j]);
}
}
}
// instantiate and save the scanners
List<ScannerInfo> scannerInfoList = new ArrayList<>(elements.size());
for (int i = 0; i < elements.size(); i++) {
try {
IFileTaskScanner scanner = (IFileTaskScanner) elements.get(i).createExecutableExtension(ATT_CLASS);
if (scanner != null) {
scannerInfoList.add(new ScannerInfo(elements.get(i).getAttribute(ATT_ID), scanner));
}
}
catch (CoreException e) {
Logger.logException("Non-fatal exception creating task scanner for " + contentType.getId(), e); //$NON-NLS-1$
}
}
scannerInfos = scannerInfoList.toArray(new ScannerInfo[scannerInfoList.size()]);
fScannerInfoByContentTypes.put(contentType.getId(), scannerInfos);
if (Logger.DEBUG_TASKSREGISTRY) {
System.out.println("Created " + scannerInfos.length + " task scanner for " + contentType.getId()); //$NON-NLS-1$ //$NON-NLS-2$
}
}
return scannerInfos;
}
private void readRegistry() {
fScannerInfoByContentTypes = new HashMap<>();
fConfigurationElementsByFilenameExtensions = new HashMap<>();
// Just remember the elements, so plug-ins don't have to be activated,
// unless extension attributes match those of interest
IExtensionPoint point = Platform.getExtensionRegistry().getExtensionPoint(SCANNER_EXTENSION_POINT_ID);
if (point != null) {
fScannerElements = point.getConfigurationElements();
}
}
}