All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.webbeans.el.ELContextStore Maven / Gradle / Ivy

/*
 * 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.el;

import java.util.HashMap;
import java.util.Map;
import java.util.Set;

import javax.enterprise.context.spi.CreationalContext;
import javax.enterprise.inject.spi.Bean;

/**
 * The ELContextStore serves two different purposes
 *
 * 
    *
  1. * Store {@link javax.enterprise.context.Dependent} objects of the same * invocation. See spec section 6.4.3. Dependent pseudo-scope and Unified EL. * This gets cleaned up with {@link #destroyDependents()} after the whole Expression * got scanned. *
  2. *
  3. * Store the Contextual Reference for each name per request thread. This is a performance * tuning strategy, because creating a {@link org.apache.webbeans.intercept.NormalScopedBeanInterceptorHandler} * for each and every EL call is very expensive. This needs to be cleaned up with * {@link #destroyELContextStore()} at the end of each request. *
  4. *
*/ public class ELContextStore { private static ThreadLocal contextStores = new ThreadLocal(); /** * @param createIfNotExist if false doesn't create a new ELContextStore if none exists * @return */ public static ELContextStore getInstance(boolean createIfNotExist) { ELContextStore store = contextStores.get(); if (store == null && createIfNotExist) { store = new ELContextStore(); contextStores.set(store); } return store; } /** * The same Expression must get same instances of @Dependent beans */ private Map, CreationalStore> dependentObjects = new HashMap, CreationalStore>(); private Map> beanNameToDependentBeanMapping = new HashMap>(); /** * Cache for resolved proxies of @NormalScoped beans. This heavily speeds up pages with * multiple ELs for the same bean. A typical bean invoke through an EL only accesses 1 * property. If we wouldn't cache this, every EL call would create a new proxy and * drops it after the EL. */ private Map normalScopedObjects = new HashMap(); public Object findBeanByName(String name) { Object cachedBean = normalScopedObjects.get(name); if(cachedBean != null) { return cachedBean; } Bean dependentBean = beanNameToDependentBeanMapping.get(name); if(dependentBean == null) { return null; } CreationalStore cs = dependentObjects.get(dependentBean); if (cs != null) { return cs.getObject(); } return null; } private static class CreationalStore { private Object object; private CreationalContext creational; public CreationalStore(Object object, CreationalContext creational) { this.object = object; this.creational = creational; } /** * @return the object */ public Object getObject() { return object; } /** * @return the creational */ public CreationalContext getCreational() { return creational; } } /** * This class can only get constructed via {@link #getInstance(boolean)} */ private ELContextStore() { } /** * Add a @Dependent scoped bean for later use in the same EL. * See spec section 6.4.3. Dependent pseudo-scope and Unified EL. * @param bean * @param dependent * @param creationalContext */ public void addDependent(Bean bean, Object dependent, CreationalContext creationalContext) { dependentObjects.put(bean, new CreationalStore(dependent, creationalContext)); beanNameToDependentBeanMapping.put(bean.getName(), bean); } /** * @see #addDependent(Bean, Object, CreationalContext) * @param bean * @return the previously used dependent bean or null */ public Object getDependent(Bean bean) { CreationalStore sc = dependentObjects.get(bean); return sc != null ? sc.getObject() : null; } /** * We cache resolved @NormalScoped bean proxies on the same for speeding up EL. * @param beanName */ public void addNormalScoped(String beanName, Object contextualInstance) { normalScopedObjects.put(beanName, contextualInstance); } /** * This method have to be called after the EL parsing to cleanup the cache * for @Dependent scoped beans. */ public void destroyDependents() { if (dependentObjects.size() > 0) { Set> beans = dependentObjects.keySet(); for(Bean bean : beans) { Bean o = (Bean)bean; CreationalStore store = dependentObjects.get(bean); o.destroy(store.getObject(), (CreationalContext)store.getCreational()); } dependentObjects.clear(); } beanNameToDependentBeanMapping.clear(); } /** * This needs to be called at the end of each request. * Because after the request ends, a server might reuse * the Thread to serve other requests (from other WebApps) */ public void destroyELContextStore() { normalScopedObjects.clear(); contextStores.set(null); contextStores.remove(); } }