
org.apache.webbeans.el22.WebBeansELResolver Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of openwebbeans-el22 Show documentation
Show all versions of openwebbeans-el22 Show documentation
Apache OpenWebBeans EL 2.2 integration
The newest version!
/*
* 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.webbeans.el22;
import jakarta.el.ELContext;
import jakarta.el.ELException;
import jakarta.el.ELResolver;
import jakarta.enterprise.context.Dependent;
import jakarta.enterprise.context.spi.CreationalContext;
import jakarta.enterprise.inject.spi.Bean;
import org.apache.webbeans.component.OwbBean;
import org.apache.webbeans.config.WebBeansContext;
import org.apache.webbeans.container.BeanManagerImpl;
import org.apache.webbeans.el.ELContextStore;
import java.beans.FeatureDescriptor;
import java.lang.reflect.Type;
import java.util.Iterator;
import java.util.Set;
import java.util.concurrent.CopyOnWriteArraySet;
import java.util.stream.Collectors;
/**
* JSF or JSP expression language a.k.a EL resolver.
*
*
* EL is registered with the JSF in faces-config.xml if there exist a faces-config.xml
* in the application location WEB-INF/
. Otherwise it is registered with
* JspApplicationContext at start-up.
*
*
*
* All @Dependent
scoped contextual instances created during an EL
* expression evaluation are destroyed when the evaluation completes.
*
*
* @version $Rev: 1307826 $ $Date: 2012-03-31 18:24:37 +0300 (Sat, 31 Mar 2012) $
*
*/
public class WebBeansELResolver extends ELResolver
{
private final WebBeansContext webBeansContext;
private Set dotNamedBeansNegativeCache;
public WebBeansELResolver()
{
webBeansContext = WebBeansContext.getInstance();
dotNamedBeansNegativeCache = new CopyOnWriteArraySet<>();
}
/**
* {@inheritDoc}
*/
@Override
public Class> getCommonPropertyType(ELContext context, Object base)
{
return null;
}
/**
* {@inheritDoc}
*/
@Override
public Iterator getFeatureDescriptors(ELContext context, Object base)
{
return null;
}
/**
* {@inheritDoc}
*/
@Override
public Class> getType(ELContext context, Object base, Object property) throws ELException
{
return null;
}
/**
* {@inheritDoc}
*/
@Override
@SuppressWarnings({"unchecked","deprecation"})
public Object getValue(ELContext context, Object base, Object property) throws ELException
{
// we only check root beans
// or if its wrapped because of dot-names
if (base != null && !(base instanceof WrappedValueExpressionNode))
{
return null;
}
// Check if the OWB actually got used in this application
final BeanManagerImpl beanManager = webBeansContext.getBeanManagerImpl();
if (!beanManager.isInUse())
{
return null;
}
//Name of the bean
final String beanName = (String) property;
// Local store, create if not exist
final ELContextStore elContextStore = ELContextStore.getInstance(true);
// Already available in the cache. Let's return it
final Object contextualInstance = elContextStore.findBeanByName(beanName);
if(contextualInstance != null)
{
context.setPropertyResolved(true);
return contextualInstance;
}
// check if it's a recursive call to handle dotted based names
if (base instanceof WrappedValueExpressionNode)
{
final String baseBeanName = ((WrappedValueExpressionNode) base).getFqBeanName();
return findDottedName(context, baseBeanName, beanManager, elContextStore, beanName);
}
// Get bean candidates
final Set> beans = beanManager.getBeans(beanName);
// Found?
if(beans != null && !beans.isEmpty())
{
return getBeanWithScope(context, beanManager, beanName, elContextStore, beans);
}
else
{
// Fallback for TCK because CDI allows CDI beans to contain dots like @Named("magic.golden.fish")
return findDottedName(context, null, beanManager, elContextStore, beanName);
}
}
private Object getBeanWithScope(final ELContext context, final BeanManagerImpl beanManager, final String beanName,
final ELContextStore elContextStore, final Set> beans)
{
// Managed bean
final Bean> bean = beanManager.resolve(beans);
if(bean.getScope().equals(Dependent.class))
{
return getDependentContextualInstance(beanManager, elContextStore, context, bean);
}
else
{
// now we check for NormalScoped beans
return getNormalScopedContextualInstance(beanManager, elContextStore, context, bean, beanName);
}
}
private Object findDottedName(final ELContext context, final Object base, final BeanManagerImpl beanManager,
final ELContextStore elContextStore, final String beanName)
{
final String fqBeanName = base == null ? beanName : base + "." + beanName;
if (dotNamedBeansNegativeCache.contains(fqBeanName))
{
return null;
}
final Set> anyBeanName = beanManager.getBeans().stream()
.filter(b -> b.getName() != null)
.filter(b -> b.getName().equals(fqBeanName) || b.getName().startsWith(fqBeanName + "."))
.collect(Collectors.toSet());
// no exact and no startsWith match
if (anyBeanName.isEmpty())
{
dotNamedBeansNegativeCache.add(fqBeanName);
return null;
}
// exact match
if (anyBeanName.size() == 1 && fqBeanName.equals(anyBeanName.iterator().next().getName()))
{
return getBeanWithScope(context, beanManager, fqBeanName, elContextStore, anyBeanName);
}
// more than one bean with the same beginning
context.setPropertyResolved(true);
return new WrappedValueExpressionNode(fqBeanName);
}
protected Object getNormalScopedContextualInstance(BeanManagerImpl manager, ELContextStore store, ELContext context,
Bean> bean, String beanName)
{
CreationalContext> creationalContext = manager.createCreationalContext(bean);
Object contextualInstance = manager.getReference(bean, Object.class, creationalContext);
if (contextualInstance != null)
{
context.setPropertyResolved(true);
//Adding into store
store.addNormalScoped(beanName, contextualInstance);
}
return contextualInstance;
}
protected Object getDependentContextualInstance(BeanManagerImpl manager, ELContextStore store, ELContext context, Bean> bean)
{
Object contextualInstance = store.getDependent(bean);
if(contextualInstance != null)
{
//Object found on the store
context.setPropertyResolved(true);
}
else
{
// If no contextualInstance found on the store
CreationalContext> creationalContext = manager.createCreationalContext(bean);
contextualInstance = manager.getReference(bean, bestType(bean), creationalContext);
if (contextualInstance != null)
{
context.setPropertyResolved(true);
//Adding into store
store.addDependent(bean, contextualInstance, creationalContext);
}
}
return contextualInstance;
}
private static Type bestType(Bean> bean)
{
if (bean == null)
{
return Object.class;
}
Class> bc = bean.getBeanClass();
if (bc != null)
{
return bc;
}
if (OwbBean.class.isInstance(bean))
{
return OwbBean.class.cast(bean).getReturnType();
}
return Object.class;
}
/**
* {@inheritDoc}
*/
@Override
public boolean isReadOnly(ELContext context, Object base, Object property) throws ELException
{
return false;
}
/**
* {@inheritDoc}
*/
@Override
public void setValue(ELContext context, Object base, Object property, Object value) throws ELException
{
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy