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

org.eclipse.jetty.ee11.annotations.ResourceAnnotationHandler Maven / Gradle / Ivy

//
// ========================================================================
// Copyright (c) 1995 Mort Bay Consulting Pty Ltd and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
// which is available at https://www.apache.org/licenses/LICENSE-2.0.
//
// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
// ========================================================================
//

package org.eclipse.jetty.ee11.annotations;

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.List;
import java.util.Locale;
import javax.naming.InitialContext;
import javax.naming.NameNotFoundException;
import javax.naming.NamingException;

import jakarta.annotation.Resource;
import org.eclipse.jetty.ee11.annotations.AnnotationIntrospector.AbstractIntrospectableAnnotationHandler;
import org.eclipse.jetty.ee11.servlet.ServletContextHandler;
import org.eclipse.jetty.ee11.webapp.MetaData;
import org.eclipse.jetty.ee11.webapp.WebAppContext;
import org.eclipse.jetty.plus.annotation.Injection;
import org.eclipse.jetty.plus.annotation.InjectionCollection;
import org.eclipse.jetty.plus.jndi.NamingEntryUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class ResourceAnnotationHandler extends AbstractIntrospectableAnnotationHandler
{
    private static final Logger LOG = LoggerFactory.getLogger(ResourceAnnotationHandler.class);

    protected static final List> ENV_ENTRY_TYPES = List.of(
        String.class,
        Character.class,
        Integer.class,
        Boolean.class,
        Double.class,
        Byte.class,
        Short.class,
        Long.class,
        Float.class
    );

    public ResourceAnnotationHandler(WebAppContext wac)
    {
        super(true, wac);
    }

    /**
     * Class level Resource annotations declare a name in the
     * environment that will be looked up at runtime. They do
     * not specify an injection.
     */
    @Override
    public void doHandle(Class clazz)
    {
        if (supportsResourceInjection(clazz))
        {
            handleClass(clazz);

            Method[] methods = clazz.getDeclaredMethods();
            for (Method method : methods)
            {
                handleMethod(clazz, method);
            }
            Field[] fields = clazz.getDeclaredFields();
            //For each field, get all of it's annotations
            for (Field field : fields)
            {
                handleField(clazz, field);
            }
        }
    }

    public void handleClass(Class clazz)
    {
        Resource resource = clazz.getAnnotation(Resource.class);
        if (resource != null)
        {
            String name = resource.name();
            String mappedName = resource.mappedName();

            if (name == null || name.trim().isEmpty())
                throw new IllegalStateException("Class level Resource annotations must contain a name (Common Annotations Spec Section 2.3)");

            try
            {
                if (!NamingEntryUtil.bindToENC(_context, name, mappedName))
                    if (!NamingEntryUtil.bindToENC(_context.getServer(), name, mappedName))
                        throw new IllegalStateException("No resource at " + (mappedName == null ? name : mappedName));
            }
            catch (NamingException e)
            {
                LOG.warn("Unable to bind name {} to {} from class {}", name, mappedName, clazz, e);
            }
        }
    }

    public void handleField(Class clazz, Field field)
    {
        Resource resource = field.getAnnotation(Resource.class);
        if (resource != null)
        {
            //JavaEE Spec 5.2.3: Field cannot be static
            if (Modifier.isStatic(field.getModifiers()))
            {
                LOG.warn("Skipping Resource annotation on {}.{}: cannot be static", clazz.getName(), field.getName());
                return;
            }

            //JavaEE Spec 5.2.3: Field cannot be final
            if (Modifier.isFinal(field.getModifiers()))
            {
                LOG.warn("Skipping Resource annotation on {}.{}: cannot be final", clazz.getName(), field.getName());
                return;
            }

            //work out default name
            String name = clazz.getName() + "/" + field.getName();

            //allow @Resource name= to override the field name
            name = (resource.name() != null && !resource.name().trim().isEmpty() ? resource.name() : name);
            String mappedName = (resource.mappedName() != null && !resource.mappedName().trim().isEmpty() ? resource.mappedName() : null);
            //get the type of the Field
            Class type = field.getType();

            //Servlet Spec 3.0 p. 76
            //If a descriptor has specified at least 1 injection target for this
            //resource, then it overrides this annotation
            MetaData metaData = _context.getMetaData();
            if (metaData.getOriginDescriptor("resource-ref." + name + ".injection") != null)
            {
                //at least 1 injection was specified for this resource by a descriptor, so
                //it overrides this annotation
                return;
            }

            //No injections for this resource in any descriptors, so we can add it
            //Does the injection already exist?
            InjectionCollection injections = (InjectionCollection)_context.getAttribute(InjectionCollection.INJECTION_COLLECTION);
            if (injections == null)
            {
                injections = new InjectionCollection();
                _context.setAttribute(InjectionCollection.INJECTION_COLLECTION, injections);
            }
            Injection injection = injections.getInjection(name, clazz, field);
            if (injection == null)
            {
                //No injection has been specified, add it
                try
                {
                    //try webapp scope first
                    boolean bound = NamingEntryUtil.bindToENC(_context, name, mappedName);
                    
                    //try environment scope next
                    if (!bound)
                        bound = NamingEntryUtil.bindToENC(ServletContextHandler.ENVIRONMENT.getName(), name, mappedName);
                    
                    //try Server scope next
                    if (!bound)
                        bound = NamingEntryUtil.bindToENC(_context.getServer(), name, mappedName);
                    
                    //try jvm scope next
                    if (!bound)
                        bound = NamingEntryUtil.bindToENC(null, name, mappedName);
                    if (!bound)
                    {
                        //see if there is an env-entry value been bound
                        try
                        {
                            InitialContext ic = new InitialContext();
                            String nameInEnvironment = (mappedName != null ? mappedName : name);
                            ic.lookup("java:comp/env/" + nameInEnvironment);
                            bound = true;
                        }
                        catch (NameNotFoundException e)
                        {
                            if (LOG.isTraceEnabled())
                                LOG.trace("ignored", e);
                        }
                    }
                    //Check there is a JNDI entry for this annotation
                    if (bound)
                    {
                        LOG.debug("Bound {} as {}", (mappedName == null ? name : mappedName), name);
                        //   Make the Injection for it if the binding succeeded
                        injection = new Injection(clazz, field, type, name, mappedName);
                        injections.add(injection);

                        //TODO - an @Resource is equivalent to a resource-ref, resource-env-ref, message-destination
                        metaData.setOrigin("resource-ref." + name + ".injection", resource, clazz);
                    }
                    else if (!isEnvEntryType(type))
                    {
                        //if this is an env-entry type resource and there is no value bound for it, it isn't
                        //an error, it just means that perhaps the code will use a default value instead
                        // JavaEE Spec. sec 5.4.1.3

                        throw new IllegalStateException("No resource at " + (mappedName == null ? name : mappedName));
                    }
                }
                catch (NamingException e)
                {
                    //if this is an env-entry type resource and there is no value bound for it, it isn't
                    //an error, it just means that perhaps the code will use a default value instead
                    // JavaEE Spec. sec 5.4.1.3
                    if (!isEnvEntryType(type))
                        throw new IllegalStateException(e);
                }
            }
        }
    }

    /**
     * Process a Resource annotation on a Method.
     * 

* This will generate a JNDI entry, and an Injection to be * processed when an instance of the class is created. * * @param clazz the class to process * @param method the method to process */ public void handleMethod(Class clazz, Method method) { Resource resource = method.getAnnotation(Resource.class); if (resource != null) { /* * Commons Annotations Spec 2.3 * " The Resource annotation is used to declare a reference to a resource. * It can be specified on a class, methods or on fields. When the * annotation is applied on a field or method, the container will * inject an instance of the requested resource into the application * when the application is initialized... Even though this annotation * is not marked Inherited, if used all superclasses MUST be examined * to discover all uses of this annotation. All such annotation instances * specify resources that are needed by the application. Note that this * annotation may appear on private fields and methods of the superclasses. * Injection of the declared resources needs to happen in these cases as * well, even if a method with such an annotation is overridden by a subclass." * * Which IMHO, put more succinctly means "If you find a @Resource on any method * or field, inject it!". */ //JavaEE Spec 5.2.3: Method cannot be static if (Modifier.isStatic(method.getModifiers())) { LOG.warn("Skipping Resource annotation on {}.{}: cannot be static", clazz.getName(), method.getName()); return; } // Check it is a valid javabean: must be void return type, the name must start with "set" and it must have // only 1 parameter if (!method.getName().startsWith("set")) { LOG.warn("Skipping Resource annotation on {}.{}: invalid java bean, does not start with 'set'", clazz.getName(), method.getName()); return; } if (method.getParameterCount() != 1) { LOG.warn("Skipping Resource annotation on {}.{}: invalid java bean, not single argument to method", clazz.getName(), method.getName()); return; } if (Void.TYPE != method.getReturnType()) { LOG.warn("Skipping Resource annotation on {}.{}: invalid java bean, not void", clazz.getName(), method.getName()); return; } //default name is the javabean property name String name = method.getName().substring(3); name = name.substring(0, 1).toLowerCase(Locale.ENGLISH) + name.substring(1); name = clazz.getName() + "/" + name; name = (resource.name() != null && !resource.name().trim().isEmpty() ? resource.name() : name); String mappedName = (resource.mappedName() != null && !resource.mappedName().trim().isEmpty() ? resource.mappedName() : null); Class paramType = method.getParameterTypes()[0]; Class resourceType = resource.type(); //Servlet Spec 3.0 p. 76 //If a descriptor has specified at least 1 injection target for this //resource, then it overrides this annotation MetaData metaData = _context.getMetaData(); if (metaData.getOriginDescriptor("resource-ref." + name + ".injection") != null) { //at least 1 injection was specified for this resource by a descriptor, so //it overrides this annotation return; } //check if an injection has already been setup for this target by web.xml InjectionCollection injections = (InjectionCollection)_context.getAttribute(InjectionCollection.INJECTION_COLLECTION); if (injections == null) { injections = new InjectionCollection(); _context.setAttribute(InjectionCollection.INJECTION_COLLECTION, injections); } Injection injection = injections.getInjection(name, clazz, method, paramType); if (injection == null) { try { //try binding name to environment //try the webapp's scope first boolean bound = NamingEntryUtil.bindToENC(_context, name, mappedName); //try the environment's scope if (!bound) bound = NamingEntryUtil.bindToENC(ServletContextHandler.ENVIRONMENT.getName(), name, mappedName); //try the server's scope if (!bound) bound = NamingEntryUtil.bindToENC(_context.getServer(), name, mappedName); //try the jvm's scope if (!bound) bound = NamingEntryUtil.bindToENC(null, name, mappedName); //TODO if it is an env-entry from web.xml it can be injected, in which case there will be no //NamingEntry, just a value bound in java:comp/env if (!bound) { try { InitialContext ic = new InitialContext(); String nameInEnvironment = (mappedName != null ? mappedName : name); ic.lookup("java:comp/env/" + nameInEnvironment); bound = true; } catch (NameNotFoundException e) { if (LOG.isTraceEnabled()) LOG.trace("ignored", e); } } if (bound) { LOG.debug("Bound {} as {}", (mappedName == null ? name : mappedName), name); // Make the Injection for it injection = new Injection(clazz, method, paramType, resourceType, name, mappedName); injections.add(injection); //TODO - an @Resource is equivalent to a resource-ref, resource-env-ref, message-destination metaData.setOrigin("resource-ref." + name + ".injection", resource, clazz); } else if (!isEnvEntryType(paramType)) { //if this is an env-entry type resource and there is no value bound for it, it isn't //an error, it just means that perhaps the code will use a default value instead // JavaEE Spec. sec 5.4.1.3 throw new IllegalStateException("No resource at " + (mappedName == null ? name : mappedName)); } } catch (NamingException e) { //if this is an env-entry type resource and there is no value bound for it, it isn't //an error, it just means that perhaps the code will use a default value instead // JavaEE Spec. sec 5.4.1.3 if (!isEnvEntryType(paramType)) throw new IllegalStateException(e); } } } } /** * Check if the given Class is one that the specification allows to have a Resource annotation. * * @param c the class * @return true if Resource annotation permitted, false otherwise */ public boolean supportsResourceInjection(Class c) { return isAnnotatableServletClass(c); } /** * Check if the class is one of the basic java types permitted as * env-entries. * * @param clazz the class to check * @return true if class is permitted by the spec to be an env-entry value */ public boolean isEnvEntryType(Class clazz) { return ENV_ENTRY_TYPES.contains(clazz); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy