blob: 78f5667e915c1f0e08c8e5c4468a72502bf6fd1a [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
* <p>
* http://www.apache.org/licenses/LICENSE-2.0
* <p>
* 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.scripting.sightly.models.impl;
import java.util.HashMap;
import java.util.Map;
import java.util.regex.Pattern;
import javax.script.Bindings;
import javax.servlet.ServletRequest;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.Properties;
import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.Reference;
import org.apache.felix.scr.annotations.Service;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.scripting.SlingBindings;
import org.apache.sling.commons.classloader.DynamicClassLoaderManager;
import org.apache.sling.models.factory.ModelFactory;
import org.apache.sling.scripting.sightly.render.RenderContext;
import org.apache.sling.scripting.sightly.use.ProviderOutcome;
import org.apache.sling.scripting.sightly.use.UseProvider;
import org.osgi.framework.Constants;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* <p>
* Sightly {@link UseProvider} which will instantiate a referenced Sling Model.
* </p>
* <p>
* For that it tries to use the {@link ModelFactory#createModel(Object, Class)} first with the adaptable {@link SlingHttpServletRequest}
* then with the adaptable {@link Resource}. It will always fail with an exception (i.e. no other {@code UseProvider} is
* asked afterwards and the exception is being rethrown) in case the following two preconditions are fulfilled:
* </p>
* <ol>
* <li>the given identifier specifies a class which can be loaded by a {@link org.apache.sling.commons.classloader.DynamicClassLoader}</li>
* <li>the loaded class has a {@link org.apache.sling.models.annotations.Model} annotation</li>
* </ol>
*
* <p>
* In case any of those preconditions are not fulfilled the other registered {@link UseProvider}s will be queried.
* </p>
*/
@Component(
metatype = true,
label = "Apache Sling Scripting Sightly Sling Models Use Provider",
description = "The Sling Models Use Provider is responsible for instantiating Sling Models to be used with Sightly's Use-API."
)
@Service
@Properties({
@Property(
name = Constants.SERVICE_RANKING,
label = "Service Ranking",
description =
"The Service Ranking value acts as the priority with which this Use Provider is queried to return an Use-object. A higher " +
"value represents a higher priority.",
/**
* Must have a higher priority than {@link org.apache.sling.scripting.sightly.impl.engine.extension.use.JavaUseProvider} but lower
* than {@link org.apache.sling.scripting.sightly.impl.engine.extension.use.RenderUnitProvider} to kick in before the
* JavaUseProvider but after the RenderUnitProvider.
*/
intValue = 95,
propertyPrivate = false
)
})
public class SlingModelsUseProvider implements UseProvider {
private static final Logger LOGGER = LoggerFactory.getLogger(SlingModelsUseProvider.class);
private static final Pattern JAVA_PATTERN = Pattern.compile(
"([[\\p{L}&&[^\\p{Lu}]]_$][\\p{L}\\p{N}_$]*\\.)*[\\p{Lu}_$][\\p{L}\\p{N}_$]*");
@Reference
private ModelFactory modelFactory = null;
@Reference
private DynamicClassLoaderManager dynamicClassLoaderManager = null;
@Override
public ProviderOutcome provide(final String identifier, final RenderContext renderContext, final Bindings arguments) {
if (!JAVA_PATTERN.matcher(identifier).matches()) {
LOGGER.debug("Identifier {} does not match a Java class name pattern.", identifier);
return ProviderOutcome.failure();
}
final Class<?> cls;
try {
cls = dynamicClassLoaderManager.getDynamicClassLoader().loadClass(identifier);
} catch (ClassNotFoundException e) {
LOGGER.debug("Could not find class with the given name {}: {}", identifier, e.getMessage());
// next use provider will be queried
return ProviderOutcome.failure();
}
if (!modelFactory.isModelClass(cls)) {
LOGGER.debug("{} is not a Sling Model.");
// next use provider will be queried
return ProviderOutcome.failure();
}
Bindings globalBindings = renderContext.getBindings();
SlingHttpServletRequest request = (SlingHttpServletRequest) globalBindings.get(SlingBindings.REQUEST);
if (request == null) {
return ProviderOutcome.failure(new IllegalStateException("Could not get request from bindings."));
}
// pass parameters as request attributes
Map<String, Object> overrides = setRequestAttributes(request, arguments);
try {
// try to instantiate class via Sling Models (first via request, then via resource)
if (modelFactory.canCreateFromAdaptable(request, cls)) {
return ProviderOutcome.notNullOrFailure(modelFactory.createModel(request, cls));
}
Resource resource = (Resource) globalBindings.get(SlingBindings.RESOURCE);
if (resource == null) {
return ProviderOutcome.failure(new IllegalStateException("Could not get resource from bindings."));
}
if (modelFactory.canCreateFromAdaptable(resource, cls)) {
return ProviderOutcome.notNullOrFailure(modelFactory.createModel(resource, cls));
}
return ProviderOutcome.failure(
new IllegalStateException("Could not adapt the given Sling Model from neither request nor resource: " + cls));
} catch (Throwable e) {
return ProviderOutcome.failure(e);
} finally {
resetRequestAttribute(request, overrides);
}
}
private Map<String, Object> setRequestAttributes(final ServletRequest request, final Bindings arguments) {
Map<String, Object> overrides = new HashMap<String, Object>();
for (Map.Entry<String, Object> entry : arguments.entrySet()) {
String key = entry.getKey();
Object value = entry.getValue();
Object oldValue = request.getAttribute(key);
if (oldValue != null) {
overrides.put(key, oldValue);
} else {
overrides.put(key, null);
}
request.setAttribute(key, value);
}
return overrides;
}
private void resetRequestAttribute(final ServletRequest request, final Map<String, Object> overrides) {
for (Map.Entry<String, Object> entry : overrides.entrySet()) {
String key = entry.getKey();
Object value = entry.getValue();
if (value == null) {
request.removeAttribute(key);
} else {
request.setAttribute(key, value);
}
}
}
}