1 /*
2  * Copyright (C) 2006 Google Inc.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.google.inject;
18 
19 import com.google.inject.internal.CircularDependencyProxy;
20 import com.google.inject.internal.LinkedBindingImpl;
21 import com.google.inject.internal.SingletonScope;
22 import com.google.inject.spi.BindingScopingVisitor;
23 import com.google.inject.spi.ExposedBinding;
24 import java.lang.annotation.Annotation;
25 
26 /**
27  * Built-in scope implementations.
28  *
29  * @author crazybob@google.com (Bob Lee)
30  */
31 public class Scopes {
32 
33   private Scopes() {}
34 
35   /** One instance per {@link Injector}. Also see {@code @}{@link Singleton}. */
36   public static final Scope SINGLETON = new SingletonScope();
37 
38   /**
39    * No scope; the same as not applying any scope at all. Each time the Injector obtains an instance
40    * of an object with "no scope", it injects this instance then immediately forgets it. When the
41    * next request for the same binding arrives it will need to obtain the instance over again.
42    *
43    * <p>This exists only in case a class has been annotated with a scope annotation such as {@link
44    * Singleton @Singleton}, and you need to override this to "no scope" in your binding.
45    *
46    * @since 2.0
47    */
48   public static final Scope NO_SCOPE =
49       new Scope() {
50         @Override
51         public <T> Provider<T> scope(Key<T> key, Provider<T> unscoped) {
52           return unscoped;
53         }
54 
55         @Override
56         public String toString() {
57           return "Scopes.NO_SCOPE";
58         }
59       };
60 
61   private static final BindingScopingVisitor<Boolean> IS_SINGLETON_VISITOR =
62       new BindingScopingVisitor<Boolean>() {
63         @Override
64         public Boolean visitNoScoping() {
65           return false;
66         }
67 
68         @Override
69         public Boolean visitScopeAnnotation(Class<? extends Annotation> scopeAnnotation) {
70           return scopeAnnotation == Singleton.class
71               || scopeAnnotation == javax.inject.Singleton.class;
72         }
73 
74         @Override
75         public Boolean visitScope(Scope scope) {
76           return scope == Scopes.SINGLETON;
77         }
78 
79         @Override
80         public Boolean visitEagerSingleton() {
81           return true;
82         }
83       };
84 
85   /**
86    * Returns true if {@code binding} is singleton-scoped. If the binding is a {@link
87    * com.google.inject.spi.LinkedKeyBinding linked key binding} and belongs to an injector (ie. it
88    * was retrieved via {@link Injector#getBinding Injector.getBinding()}), then this method will
89    * also true if the target binding is singleton-scoped.
90    *
91    * @since 3.0
92    */
93   public static boolean isSingleton(Binding<?> binding) {
94     do {
95       boolean singleton = binding.acceptScopingVisitor(IS_SINGLETON_VISITOR);
96       if (singleton) {
97         return true;
98       }
99 
100       if (binding instanceof LinkedBindingImpl) {
101         LinkedBindingImpl<?> linkedBinding = (LinkedBindingImpl) binding;
102         Injector injector = linkedBinding.getInjector();
103         if (injector != null) {
104           binding = injector.getBinding(linkedBinding.getLinkedKey());
105           continue;
106         }
107       } else if (binding instanceof ExposedBinding) {
108         ExposedBinding<?> exposedBinding = (ExposedBinding) binding;
109         Injector injector = exposedBinding.getPrivateElements().getInjector();
110         if (injector != null) {
111           binding = injector.getBinding(exposedBinding.getKey());
112           continue;
113         }
114       }
115 
116       return false;
117     } while (true);
118   }
119 
120   /**
121    * Returns true if {@code binding} has the given scope. If the binding is a {@link
122    * com.google.inject.spi.LinkedKeyBinding linked key binding} and belongs to an injector (ie. it
123    * was retrieved via {@link Injector#getBinding Injector.getBinding()}), then this method will
124    * also true if the target binding has the given scope.
125    *
126    * @param binding binding to check
127    * @param scope scope implementation instance
128    * @param scopeAnnotation scope annotation class
129    * @since 4.0
130    */
131   public static boolean isScoped(
132       Binding<?> binding, final Scope scope, final Class<? extends Annotation> scopeAnnotation) {
133     do {
134       boolean matches =
135           binding.acceptScopingVisitor(
136               new BindingScopingVisitor<Boolean>() {
137                 @Override
138                 public Boolean visitNoScoping() {
139                   return false;
140                 }
141 
142                 @Override
143                 public Boolean visitScopeAnnotation(Class<? extends Annotation> visitedAnnotation) {
144                   return visitedAnnotation == scopeAnnotation;
145                 }
146 
147                 @Override
148                 public Boolean visitScope(Scope visitedScope) {
149                   return visitedScope == scope;
150                 }
151 
152                 @Override
153                 public Boolean visitEagerSingleton() {
154                   return false;
155                 }
156               });
157 
158       if (matches) {
159         return true;
160       }
161 
162       if (binding instanceof LinkedBindingImpl) {
163         LinkedBindingImpl<?> linkedBinding = (LinkedBindingImpl) binding;
164         Injector injector = linkedBinding.getInjector();
165         if (injector != null) {
166           binding = injector.getBinding(linkedBinding.getLinkedKey());
167           continue;
168         }
169       } else if (binding instanceof ExposedBinding) {
170         ExposedBinding<?> exposedBinding = (ExposedBinding) binding;
171         Injector injector = exposedBinding.getPrivateElements().getInjector();
172         if (injector != null) {
173           binding = injector.getBinding(exposedBinding.getKey());
174           continue;
175         }
176       }
177 
178       return false;
179     } while (true);
180   }
181 
182   /**
183    * Returns true if the object is a proxy for a circular dependency, constructed by Guice because
184    * it encountered a circular dependency. Scope implementations should be careful to <b>not cache
185    * circular proxies</b>, because the proxies are not intended for general purpose use. (They are
186    * designed just to fulfill the immediate injection, not all injections. Caching them can lead to
187    * IllegalArgumentExceptions or ClassCastExceptions.)
188    *
189    * @since 4.0
190    */
191   public static boolean isCircularProxy(Object object) {
192     return object instanceof CircularDependencyProxy;
193   }
194 }
195