blob: 4cf9179856c8da1ac996c44670a3ba58c703ebe7 [file] [log] [blame]
/*~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
~ Licensed to the Apache Software Foundation (ASF) under one
~ or more contributor license agreements. See the NOTICE file
~ distributed with this work for additional information
~ regarding copyright ownership. The ASF licenses this file
~ to you under the Apache License, Version 2.0 (the
~ "License"); you may not use this file except in compliance
~ with the License. You may obtain a copy of the License at
~
~ http://www.apache.org/licenses/LICENSE-2.0
~
~ Unless required by applicable law or agreed to in writing,
~ software distributed under the License is distributed on an
~ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
~ KIND, either express or implied. See the License for the
~ specific language governing permissions and limitations
~ under the License.
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~*/
package org.apache.sling.scriptingbundle.plugin.capability;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Collections;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Stream;
import org.apache.sling.api.servlets.ServletResolverConstants;
import org.apache.sling.scriptingbundle.plugin.processor.Constants;
import org.apache.sling.scriptingbundle.plugin.processor.FileProcessor;
import org.apache.sling.scriptingbundle.plugin.processor.Logger;
import org.apache.sling.scriptingbundle.plugin.processor.PathOnlyScriptAnalyser;
import org.apache.sling.scriptingbundle.plugin.processor.ResourceTypeFolderAnalyser;
import org.jetbrains.annotations.NotNull;
import org.osgi.framework.VersionRange;
import aQute.bnd.header.Attrs;
import aQute.bnd.header.Parameters;
public class Capabilities {
private final Set<ProvidedResourceTypeCapability> providedResourceTypeCapabilities;
private final Set<ProvidedScriptCapability> providedScriptCapabilities;
private final Set<RequiredResourceTypeCapability> requiredResourceTypeCapabilities;
public static final Capabilities EMPTY = new Capabilities(Collections.emptySet(), Collections.emptySet(), Collections.emptySet());
public Capabilities(
@NotNull Set<ProvidedResourceTypeCapability> providedResourceTypeCapabilities,
@NotNull Set<ProvidedScriptCapability> providedScriptCapabilities,
@NotNull Set<RequiredResourceTypeCapability> requiredResourceTypeCapabilities) {
this.providedResourceTypeCapabilities = providedResourceTypeCapabilities;
this.providedScriptCapabilities = providedScriptCapabilities;
this.requiredResourceTypeCapabilities = requiredResourceTypeCapabilities;
}
public @NotNull Set<ProvidedResourceTypeCapability> getProvidedResourceTypeCapabilities() {
return Collections.unmodifiableSet(providedResourceTypeCapabilities);
}
public @NotNull Set<ProvidedScriptCapability> getProvidedScriptCapabilities() {
return Collections.unmodifiableSet(providedScriptCapabilities);
}
public @NotNull Set<RequiredResourceTypeCapability> getRequiredResourceTypeCapabilities() {
return Collections.unmodifiableSet(requiredResourceTypeCapabilities);
}
public @NotNull String getProvidedCapabilitiesString() {
Parameters parameters = new Parameters();
for (ProvidedResourceTypeCapability capability : getProvidedResourceTypeCapabilities()) {
Attrs attributes = new Attrs();
attributes.putTyped(Constants.CAPABILITY_RESOURCE_TYPE_AT, capability.getResourceTypes());
Optional.ofNullable(capability.getScriptEngine()).ifPresent(scriptEngine ->
attributes.put(Constants.CAPABILITY_SCRIPT_ENGINE_AT, scriptEngine)
);
Optional.ofNullable(capability.getScriptExtension()).ifPresent(scriptExtension ->
attributes.put(Constants.CAPABILITY_SCRIPT_EXTENSION_AT, scriptExtension)
);
Optional.ofNullable(capability.getVersion()).ifPresent(version ->
attributes.putTyped(Constants.CAPABILITY_VERSION_AT, new aQute.bnd.version.Version(version.toString()))
);
Optional.ofNullable(capability.getExtendsResourceType()).ifPresent(extendedResourceType ->
attributes.put(Constants.CAPABILITY_EXTENDS_AT, extendedResourceType)
);
Optional.ofNullable(capability.getRequestMethod()).ifPresent(method ->
attributes.put(Constants.CAPABILITY_METHODS_AT, method)
);
Optional.ofNullable(capability.getRequestExtension()).ifPresent(requestExtension ->
attributes.put(Constants.CAPABILITY_EXTENSIONS_AT, requestExtension)
);
if (!capability.getSelectors().isEmpty()) {
attributes.putTyped(Constants.CAPABILITY_SELECTORS_AT, capability.getSelectors());
}
parameters.add(Constants.CAPABILITY_NS, attributes);
}
for (ProvidedScriptCapability scriptCapability : getProvidedScriptCapabilities()) {
Attrs attributes = new Attrs();
attributes.put(Constants.CAPABILITY_PATH_AT, scriptCapability.getPath());
attributes.put(Constants.CAPABILITY_SCRIPT_ENGINE_AT, scriptCapability.getScriptEngine());
attributes.put(Constants.CAPABILITY_SCRIPT_EXTENSION_AT, scriptCapability.getScriptExtension());
parameters.add(Constants.CAPABILITY_NS, attributes);
}
return parameters.toString();
}
public @NotNull String getRequiredCapabilitiesString() {
Parameters parameters = new Parameters();
for (RequiredResourceTypeCapability capability : getRequiredResourceTypeCapabilities()) {
Attrs attributes = new Attrs();
StringBuilder filterValue = new StringBuilder("(&(!(" + ServletResolverConstants.SLING_SERVLET_SELECTORS + "=*))");
VersionRange versionRange = capability.getVersionRange();
if (versionRange != null) {
filterValue.append("(&").append(versionRange.toFilterString("version")).append("(").append(ServletResolverConstants.SLING_SERVLET_RESOURCE_TYPES).append(
"=").append(capability.getResourceType()).append(")))");
} else {
filterValue.append("(").append(ServletResolverConstants.SLING_SERVLET_RESOURCE_TYPES).append("=").append(capability.getResourceType()).append("))");
}
attributes.put(aQute.bnd.osgi.Constants.FILTER_DIRECTIVE, filterValue.toString());
if (capability.isOptional()) {
attributes.put(aQute.bnd.osgi.Constants.RESOLUTION_DIRECTIVE, aQute.bnd.osgi.Constants.OPTIONAL);
}
parameters.add(Constants.CAPABILITY_NS, attributes);
}
return parameters.toString();
}
public static @NotNull Capabilities fromFileSystemTree(@NotNull Path root, @NotNull Stream<Path> files, @NotNull Logger logger,
@NotNull Set<String> searchPaths,
@NotNull Map<String, String> scriptEngineMappings,
boolean missingRequirementsOptional,
boolean inContentPackage) {
Set<ProvidedResourceTypeCapability> providedResourceTypeCapabilities = new LinkedHashSet<>();
Set<ProvidedScriptCapability> providedScriptCapabilities = new LinkedHashSet<>();
Set<RequiredResourceTypeCapability> requiredResourceTypeCapabilities = new LinkedHashSet<>();
FileProcessor fileProcessor = new FileProcessor(logger, searchPaths, scriptEngineMappings);
ResourceTypeFolderAnalyser resourceTypeFolderAnalyser = new ResourceTypeFolderAnalyser(logger, root, fileProcessor, inContentPackage);
PathOnlyScriptAnalyser
pathOnlyScriptAnalyser = new PathOnlyScriptAnalyser(logger, root, scriptEngineMappings, fileProcessor, inContentPackage);
files.forEach(path -> {
if (Files.isDirectory(path)) {
Capabilities resourceTypeCapabilities = resourceTypeFolderAnalyser.getCapabilities(path);
providedResourceTypeCapabilities.addAll(resourceTypeCapabilities.getProvidedResourceTypeCapabilities());
requiredResourceTypeCapabilities.addAll(resourceTypeCapabilities.getRequiredResourceTypeCapabilities());
} else {
Capabilities pathCapabilities = pathOnlyScriptAnalyser.getProvidedScriptCapability(path);
providedScriptCapabilities.addAll(pathCapabilities.getProvidedScriptCapabilities());
requiredResourceTypeCapabilities.addAll(pathCapabilities.getRequiredResourceTypeCapabilities());
}
});
final Set<RequiredResourceTypeCapability> required = new LinkedHashSet<>();
if (missingRequirementsOptional) {
Set<RequiredResourceTypeCapability> unresolvedRequiredResourceTypeCapabilities =
new LinkedHashSet<>(requiredResourceTypeCapabilities);
providedResourceTypeCapabilities.forEach(providedResourceTypeCapability -> unresolvedRequiredResourceTypeCapabilities
.removeIf(
requiredResourceTypeCapability -> requiredResourceTypeCapability.isSatisfied(providedResourceTypeCapability)));
requiredResourceTypeCapabilities.forEach(requiredResourceTypeCapability -> {
if (unresolvedRequiredResourceTypeCapabilities.contains(requiredResourceTypeCapability)) {
required.add(
RequiredResourceTypeCapability.builder().withResourceType(requiredResourceTypeCapability.getResourceType())
.withVersionRange(requiredResourceTypeCapability.getVersionRange()).withIsOptional().build());
} else {
required.add(requiredResourceTypeCapability);
}
});
} else {
required.addAll(requiredResourceTypeCapabilities);
}
return new Capabilities(providedResourceTypeCapabilities, providedScriptCapabilities, required);
}
}