/*
 * Copyright 2002-2017 the original author or authors.
 *
 * Licensed 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
 *
 *      https://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.springframework.context.annotation;

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

import org.springframework.beans.factory.annotation.AnnotatedBeanDefinition;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.lang.Nullable;

Simple ScopeMetadataResolver implementation that follows JSR-330 scoping rules: defaulting to prototype scope unless Singleton is present.

This scope resolver can be used with ClassPathBeanDefinitionScanner and AnnotatedBeanDefinitionReader for standard JSR-330 compliance. However, in practice, you will typically use Spring's rich default scoping instead - or extend this resolver with custom scoping annotations that point to extended Spring scopes.

Author:Juergen Hoeller
See Also:
Since:3.0
/** * Simple {@link ScopeMetadataResolver} implementation that follows JSR-330 scoping rules: * defaulting to prototype scope unless {@link javax.inject.Singleton} is present. * * <p>This scope resolver can be used with {@link ClassPathBeanDefinitionScanner} and * {@link AnnotatedBeanDefinitionReader} for standard JSR-330 compliance. However, * in practice, you will typically use Spring's rich default scoping instead - or extend * this resolver with custom scoping annotations that point to extended Spring scopes. * * @author Juergen Hoeller * @since 3.0 * @see #registerScope * @see #resolveScopeName * @see ClassPathBeanDefinitionScanner#setScopeMetadataResolver * @see AnnotatedBeanDefinitionReader#setScopeMetadataResolver */
public class Jsr330ScopeMetadataResolver implements ScopeMetadataResolver { private final Map<String, String> scopeMap = new HashMap<>(); public Jsr330ScopeMetadataResolver() { registerScope("javax.inject.Singleton", BeanDefinition.SCOPE_SINGLETON); }
Register an extended JSR-330 scope annotation, mapping it onto a specific Spring scope by name.
Params:
  • annotationType – the JSR-330 annotation type as a Class
  • scopeName – the Spring scope name
/** * Register an extended JSR-330 scope annotation, mapping it onto a * specific Spring scope by name. * @param annotationType the JSR-330 annotation type as a Class * @param scopeName the Spring scope name */
public final void registerScope(Class<?> annotationType, String scopeName) { this.scopeMap.put(annotationType.getName(), scopeName); }
Register an extended JSR-330 scope annotation, mapping it onto a specific Spring scope by name.
Params:
  • annotationType – the JSR-330 annotation type by name
  • scopeName – the Spring scope name
/** * Register an extended JSR-330 scope annotation, mapping it onto a * specific Spring scope by name. * @param annotationType the JSR-330 annotation type by name * @param scopeName the Spring scope name */
public final void registerScope(String annotationType, String scopeName) { this.scopeMap.put(annotationType, scopeName); }
Resolve the given annotation type into a named Spring scope.

The default implementation simply checks against registered scopes. Can be overridden for custom mapping rules, e.g. naming conventions.

Params:
  • annotationType – the JSR-330 annotation type
Returns:the Spring scope name
/** * Resolve the given annotation type into a named Spring scope. * <p>The default implementation simply checks against registered scopes. * Can be overridden for custom mapping rules, e.g. naming conventions. * @param annotationType the JSR-330 annotation type * @return the Spring scope name */
@Nullable protected String resolveScopeName(String annotationType) { return this.scopeMap.get(annotationType); } @Override public ScopeMetadata resolveScopeMetadata(BeanDefinition definition) { ScopeMetadata metadata = new ScopeMetadata(); metadata.setScopeName(BeanDefinition.SCOPE_PROTOTYPE); if (definition instanceof AnnotatedBeanDefinition) { AnnotatedBeanDefinition annDef = (AnnotatedBeanDefinition) definition; Set<String> annTypes = annDef.getMetadata().getAnnotationTypes(); String found = null; for (String annType : annTypes) { Set<String> metaAnns = annDef.getMetadata().getMetaAnnotationTypes(annType); if (metaAnns.contains("javax.inject.Scope")) { if (found != null) { throw new IllegalStateException("Found ambiguous scope annotations on bean class [" + definition.getBeanClassName() + "]: " + found + ", " + annType); } found = annType; String scopeName = resolveScopeName(annType); if (scopeName == null) { throw new IllegalStateException( "Unsupported scope annotation - not mapped onto Spring scope name: " + annType); } metadata.setScopeName(scopeName); } } } return metadata; } }