limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 1 | /** |
| 2 | * Copyright (C) 2007 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.throwingproviders; |
| 18 | |
limpbizkit | 49f67c0 | 2008-06-10 20:56:17 +0000 | [diff] [blame] | 19 | import com.google.inject.Binder; |
limpbizkit | 49f67c0 | 2008-06-10 20:56:17 +0000 | [diff] [blame] | 20 | import com.google.inject.Key; |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 21 | import com.google.inject.Module; |
limpbizkit | 49f67c0 | 2008-06-10 20:56:17 +0000 | [diff] [blame] | 22 | import com.google.inject.Provider; |
| 23 | import com.google.inject.TypeLiteral; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 24 | import com.google.inject.binder.ScopedBindingBuilder; |
sberlin | d9c913a | 2011-06-26 21:02:54 +0000 | [diff] [blame^] | 25 | import static com.google.common.base.Preconditions.checkNotNull; |
limpbizkit | e39d8d8 | 2008-03-26 07:19:25 +0000 | [diff] [blame] | 26 | import com.google.inject.internal.UniqueAnnotations; |
sberlin | d9c913a | 2011-06-26 21:02:54 +0000 | [diff] [blame^] | 27 | import com.google.common.collect.ImmutableList; |
| 28 | import com.google.common.collect.ImmutableSet; |
| 29 | import com.google.common.collect.Lists; |
sberlin | 5ac45a8 | 2010-10-16 21:38:22 +0000 | [diff] [blame] | 30 | import com.google.inject.spi.Dependency; |
| 31 | import com.google.inject.spi.ProviderWithDependencies; |
limpbizkit | 49f67c0 | 2008-06-10 20:56:17 +0000 | [diff] [blame] | 32 | import com.google.inject.util.Types; |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 33 | |
| 34 | import java.io.Serializable; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 35 | import java.lang.annotation.Annotation; |
limpbizkit | 49f67c0 | 2008-06-10 20:56:17 +0000 | [diff] [blame] | 36 | import java.lang.reflect.InvocationHandler; |
| 37 | import java.lang.reflect.Method; |
| 38 | import java.lang.reflect.ParameterizedType; |
| 39 | import java.lang.reflect.Proxy; |
| 40 | import java.lang.reflect.Type; |
| 41 | import java.lang.reflect.TypeVariable; |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 42 | import java.util.Arrays; |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 43 | import java.util.List; |
sberlin | 5ac45a8 | 2010-10-16 21:38:22 +0000 | [diff] [blame] | 44 | import java.util.Set; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 45 | |
| 46 | /** |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 47 | * <p>Builds a binding for a {@link CheckedProvider}. |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 48 | * |
| 49 | * <p>You can use a fluent API and custom providers: |
limpbizkit | 0c43f22 | 2007-12-19 01:38:16 +0000 | [diff] [blame] | 50 | * <pre><code>ThrowingProviderBinder.create(binder()) |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 51 | * .bind(RemoteProvider.class, Customer.class) |
| 52 | * .to(RemoteCustomerProvider.class) |
| 53 | * .in(RequestScope.class); |
limpbizkit | 0c43f22 | 2007-12-19 01:38:16 +0000 | [diff] [blame] | 54 | * </code></pre> |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 55 | * or, you can use throwing provider methods: |
| 56 | * <pre><code>class MyModule extends AbstractModule { |
| 57 | * configure() { |
| 58 | * ThrowingProviderBinder.install(this, binder()); |
| 59 | * } |
| 60 | * |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 61 | * {@literal @}CheckedProvides(RemoteProvider.class) |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 62 | * {@literal @}RequestScope |
| 63 | * Customer provideCustomer(FlakyCustomerCreator creator) throws RemoteException { |
| 64 | * return creator.getCustomerOrThrow(); |
| 65 | * } |
| 66 | * } |
| 67 | * </code></pre> |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 68 | * |
| 69 | * @author jmourits@google.com (Jerome Mourits) |
| 70 | * @author jessewilson@google.com (Jesse Wilson) |
| 71 | */ |
| 72 | public class ThrowingProviderBinder { |
| 73 | |
| 74 | private final Binder binder; |
| 75 | |
| 76 | private ThrowingProviderBinder(Binder binder) { |
| 77 | this.binder = binder; |
| 78 | } |
| 79 | |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 80 | public static ThrowingProviderBinder create(Binder binder) { |
| 81 | return new ThrowingProviderBinder(binder.skipSources( |
| 82 | ThrowingProviderBinder.class, |
| 83 | ThrowingProviderBinder.SecondaryBinder.class)); |
| 84 | } |
| 85 | |
sberlin | c13b545 | 2010-10-31 18:38:24 +0000 | [diff] [blame] | 86 | /** |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 87 | * Returns a module that installs {@literal @}{@link CheckedProvides} methods. |
sberlin | c13b545 | 2010-10-31 18:38:24 +0000 | [diff] [blame] | 88 | * |
| 89 | * @since 3.0 |
| 90 | */ |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 91 | public static Module forModule(Module module) { |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 92 | return CheckedProviderMethodsModule.forModule(module); |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 93 | } |
| 94 | |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 95 | public <P extends CheckedProvider> SecondaryBinder<P> |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 96 | bind(final Class<P> interfaceType, final Type valueType) { |
| 97 | return new SecondaryBinder<P>(interfaceType, valueType); |
| 98 | } |
| 99 | |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 100 | public class SecondaryBinder<P extends CheckedProvider> { |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 101 | private final Class<P> interfaceType; |
| 102 | private final Type valueType; |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 103 | private final List<Class<? extends Throwable>> exceptionTypes; |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 104 | private final boolean valid; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 105 | |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 106 | private Class<? extends Annotation> annotationType; |
| 107 | private Annotation annotation; |
| 108 | private Key<P> interfaceKey; |
| 109 | |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 110 | public SecondaryBinder(Class<P> interfaceType, Type valueType) { |
kevinb9n | 1601ae5 | 2008-06-03 22:21:04 +0000 | [diff] [blame] | 111 | this.interfaceType = checkNotNull(interfaceType, "interfaceType"); |
| 112 | this.valueType = checkNotNull(valueType, "valueType"); |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 113 | if(checkInterface()) { |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 114 | this.exceptionTypes = getExceptionType(interfaceType); |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 115 | valid = true; |
| 116 | } else { |
| 117 | valid = false; |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 118 | this.exceptionTypes = ImmutableList.of(); |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 119 | } |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 120 | } |
| 121 | |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 122 | List<Class<? extends Throwable>> getExceptionTypes() { |
| 123 | return exceptionTypes; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 124 | } |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 125 | |
| 126 | Key<P> getKey() { |
| 127 | return interfaceKey; |
| 128 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 129 | |
| 130 | public SecondaryBinder<P> annotatedWith(Class<? extends Annotation> annotationType) { |
| 131 | if (!(this.annotationType == null && this.annotation == null)) { |
| 132 | throw new IllegalStateException(); |
| 133 | } |
| 134 | this.annotationType = annotationType; |
| 135 | return this; |
| 136 | } |
| 137 | |
| 138 | public SecondaryBinder<P> annotatedWith(Annotation annotation) { |
| 139 | if (!(this.annotationType == null && this.annotation == null)) { |
| 140 | throw new IllegalStateException(); |
| 141 | } |
| 142 | this.annotation = annotation; |
| 143 | return this; |
| 144 | } |
| 145 | |
| 146 | public ScopedBindingBuilder to(P target) { |
limpbizkit | e39d8d8 | 2008-03-26 07:19:25 +0000 | [diff] [blame] | 147 | Key<P> targetKey = Key.get(interfaceType, UniqueAnnotations.create()); |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 148 | binder.bind(targetKey).toInstance(target); |
| 149 | return to(targetKey); |
| 150 | } |
| 151 | |
| 152 | public ScopedBindingBuilder to(Class<? extends P> targetType) { |
| 153 | return to(Key.get(targetType)); |
| 154 | } |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 155 | |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 156 | ScopedBindingBuilder toProviderMethod(CheckedProviderMethod<?> target) { |
| 157 | Key<CheckedProviderMethod> targetKey = |
| 158 | Key.get(CheckedProviderMethod.class, UniqueAnnotations.create()); |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 159 | binder.bind(targetKey).toInstance(target); |
| 160 | |
| 161 | return toInternal(targetKey); |
| 162 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 163 | |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 164 | public ScopedBindingBuilder to(Key<? extends P> targetKey) { |
kevinb9n | 1601ae5 | 2008-06-03 22:21:04 +0000 | [diff] [blame] | 165 | checkNotNull(targetKey, "targetKey"); |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 166 | return toInternal(targetKey); |
| 167 | } |
| 168 | |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 169 | private ScopedBindingBuilder toInternal(final Key<? extends CheckedProvider> targetKey) { |
limpbizkit | e39d8d8 | 2008-03-26 07:19:25 +0000 | [diff] [blame] | 170 | final Key<Result> resultKey = Key.get(Result.class, UniqueAnnotations.create()); |
sberlin | 5ac45a8 | 2010-10-16 21:38:22 +0000 | [diff] [blame] | 171 | final Provider<Result> resultProvider = binder.getProvider(resultKey); |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 172 | final Provider<? extends CheckedProvider> targetProvider = binder.getProvider(targetKey); |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 173 | interfaceKey = createKey(); |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 174 | |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 175 | // don't bother binding the proxy type if this is in an invalid state. |
| 176 | if(valid) { |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 177 | binder.bind(interfaceKey).toProvider(new ProviderWithDependencies<P>() { |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 178 | private final P instance = interfaceType.cast(Proxy.newProxyInstance( |
| 179 | interfaceType.getClassLoader(), new Class<?>[] { interfaceType }, |
| 180 | new InvocationHandler() { |
| 181 | public Object invoke(Object proxy, Method method, Object[] args) |
| 182 | throws Throwable { |
| 183 | return resultProvider.get().getOrThrow(); |
| 184 | } |
| 185 | })); |
| 186 | |
| 187 | public P get() { |
| 188 | return instance; |
| 189 | } |
| 190 | |
| 191 | public Set<Dependency<?>> getDependencies() { |
| 192 | return ImmutableSet.<Dependency<?>>of(Dependency.get(resultKey)); |
| 193 | } |
| 194 | }); |
| 195 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 196 | |
sberlin | 5ac45a8 | 2010-10-16 21:38:22 +0000 | [diff] [blame] | 197 | return binder.bind(resultKey).toProvider(new ProviderWithDependencies<Result>() { |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 198 | public Result get() { |
| 199 | try { |
sberlin | 5ac45a8 | 2010-10-16 21:38:22 +0000 | [diff] [blame] | 200 | return Result.forValue(targetProvider.get().get()); |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 201 | } catch (Exception e) { |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 202 | for(Class<? extends Throwable> exceptionType : exceptionTypes) { |
| 203 | if (exceptionType.isInstance(e)) { |
| 204 | return Result.forException(e); |
| 205 | } |
| 206 | } |
| 207 | |
| 208 | if (e instanceof RuntimeException) { |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 209 | throw (RuntimeException) e; |
| 210 | } else { |
| 211 | // this should never happen |
| 212 | throw new RuntimeException(e); |
| 213 | } |
| 214 | } |
| 215 | } |
sberlin | 5ac45a8 | 2010-10-16 21:38:22 +0000 | [diff] [blame] | 216 | |
| 217 | public Set<Dependency<?>> getDependencies() { |
| 218 | return ImmutableSet.<Dependency<?>>of(Dependency.get(targetKey)); |
| 219 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 220 | }); |
| 221 | } |
| 222 | |
| 223 | /** |
| 224 | * Returns the exception type declared to be thrown by the get method of |
| 225 | * {@code interfaceType}. |
| 226 | */ |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 227 | private List<Class<? extends Throwable>> getExceptionType(Class<P> interfaceType) { |
| 228 | try { |
| 229 | Method getMethod = interfaceType.getMethod("get"); |
| 230 | List<TypeLiteral<?>> exceptionLiterals = |
| 231 | TypeLiteral.get(interfaceType).getExceptionTypes(getMethod); |
| 232 | List<Class<? extends Throwable>> results = Lists.newArrayList(); |
| 233 | for (TypeLiteral<?> exLiteral : exceptionLiterals) { |
| 234 | results.add(exLiteral.getRawType().asSubclass(Throwable.class)); |
| 235 | } |
| 236 | return results; |
| 237 | } catch (SecurityException e) { |
| 238 | throw new IllegalStateException("Not allowed to inspect exception types", e); |
| 239 | } catch (NoSuchMethodException e) { |
| 240 | throw new IllegalStateException("No 'get'method available", e); |
| 241 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 242 | } |
| 243 | |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 244 | private boolean checkInterface() { |
| 245 | if(!checkArgument(interfaceType.isInterface(), |
| 246 | "%s must be an interface", interfaceType.getName())) { |
| 247 | return false; |
| 248 | } |
| 249 | if(!checkArgument(interfaceType.getGenericInterfaces().length == 1, |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 250 | "%s must extend CheckedProvider (and only CheckedProvider)", |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 251 | interfaceType)) { |
| 252 | return false; |
| 253 | } |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 254 | |
| 255 | boolean tpMode = interfaceType.getInterfaces()[0] == ThrowingProvider.class; |
| 256 | if(!tpMode) { |
| 257 | if(!checkArgument(interfaceType.getInterfaces()[0] == CheckedProvider.class, |
| 258 | "%s must extend CheckedProvider (and only CheckedProvider)", |
| 259 | interfaceType)) { |
| 260 | return false; |
| 261 | } |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 262 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 263 | |
| 264 | // Ensure that T is parameterized and unconstrained. |
| 265 | ParameterizedType genericThrowingProvider |
| 266 | = (ParameterizedType) interfaceType.getGenericInterfaces()[0]; |
| 267 | if (interfaceType.getTypeParameters().length == 1) { |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 268 | String returnTypeName = interfaceType.getTypeParameters()[0].getName(); |
| 269 | Type returnType = genericThrowingProvider.getActualTypeArguments()[0]; |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 270 | if(!checkArgument(returnType instanceof TypeVariable, |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 271 | "%s does not properly extend CheckedProvider, the first type parameter of CheckedProvider (%s) is not a generic type", |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 272 | interfaceType, returnType)) { |
| 273 | return false; |
| 274 | } |
| 275 | if(!checkArgument(returnTypeName.equals(((TypeVariable) returnType).getName()), |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 276 | "The generic type (%s) of %s does not match the generic type of CheckedProvider (%s)", |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 277 | returnTypeName, interfaceType, ((TypeVariable)returnType).getName())) { |
| 278 | return false; |
| 279 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 280 | } else { |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 281 | if(!checkArgument(interfaceType.getTypeParameters().length == 0, |
| 282 | "%s has more than one generic type parameter: %s", |
| 283 | interfaceType, Arrays.asList(interfaceType.getTypeParameters()))) { |
| 284 | return false; |
| 285 | } |
| 286 | if(!checkArgument(genericThrowingProvider.getActualTypeArguments()[0].equals(valueType), |
| 287 | "%s expects the value type to be %s, but it was %s", |
| 288 | interfaceType, genericThrowingProvider.getActualTypeArguments()[0], valueType)) { |
| 289 | return false; |
| 290 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 291 | } |
| 292 | |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 293 | if(tpMode) { // only validate exception in ThrowingProvider mode. |
| 294 | Type exceptionType = genericThrowingProvider.getActualTypeArguments()[1]; |
| 295 | if(!checkArgument(exceptionType instanceof Class, |
| 296 | "%s has the wrong Exception generic type (%s) when extending CheckedProvider", |
| 297 | interfaceType, exceptionType)) { |
| 298 | return false; |
| 299 | } |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 300 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 301 | |
| 302 | if (interfaceType.getDeclaredMethods().length == 1) { |
| 303 | Method method = interfaceType.getDeclaredMethods()[0]; |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 304 | if(!checkArgument(method.getName().equals("get"), |
| 305 | "%s may not declare any new methods, but declared %s", |
| 306 | interfaceType, method)) { |
| 307 | return false; |
| 308 | } |
| 309 | if(!checkArgument(method.getParameterTypes().length == 0, |
| 310 | "%s may not declare any new methods, but declared %s", |
| 311 | interfaceType, method.toGenericString())) { |
| 312 | return false; |
| 313 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 314 | } else { |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 315 | if(!checkArgument(interfaceType.getDeclaredMethods().length == 0, |
| 316 | "%s may not declare any new methods, but declared %s", |
| 317 | interfaceType, Arrays.asList(interfaceType.getDeclaredMethods()))) { |
| 318 | return false; |
| 319 | } |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 320 | } |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 321 | |
| 322 | return true; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 323 | } |
| 324 | |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 325 | private boolean checkArgument(boolean condition, |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 326 | String messageFormat, Object... args) { |
| 327 | if (!condition) { |
sberlin | ee446ca | 2010-10-24 21:36:39 +0000 | [diff] [blame] | 328 | binder.addError(messageFormat, args); |
| 329 | return false; |
| 330 | } else { |
| 331 | return true; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 332 | } |
| 333 | } |
| 334 | |
| 335 | @SuppressWarnings({"unchecked"}) |
| 336 | private Key<P> createKey() { |
| 337 | TypeLiteral<P> typeLiteral; |
| 338 | if (interfaceType.getTypeParameters().length == 1) { |
limpbizkit | 49f67c0 | 2008-06-10 20:56:17 +0000 | [diff] [blame] | 339 | ParameterizedType type = Types.newParameterizedTypeWithOwner( |
| 340 | interfaceType.getEnclosingClass(), interfaceType, valueType); |
| 341 | typeLiteral = (TypeLiteral<P>) TypeLiteral.get(type); |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 342 | } else { |
| 343 | typeLiteral = TypeLiteral.get(interfaceType); |
| 344 | } |
| 345 | |
| 346 | if (annotation != null) { |
| 347 | return Key.get(typeLiteral, annotation); |
| 348 | |
| 349 | } else if (annotationType != null) { |
| 350 | return Key.get(typeLiteral, annotationType); |
| 351 | |
| 352 | } else { |
| 353 | return Key.get(typeLiteral); |
| 354 | } |
| 355 | } |
| 356 | } |
| 357 | |
| 358 | /** |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 359 | * Represents the returned value from a call to {@link |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 360 | * CheckedProvider#get()}. This is the value that will be scoped by Guice. |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 361 | */ |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 362 | static class Result implements Serializable { |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 363 | private final Object value; |
| 364 | private final Exception exception; |
| 365 | |
| 366 | private Result(Object value, Exception exception) { |
| 367 | this.value = value; |
| 368 | this.exception = exception; |
| 369 | } |
| 370 | |
| 371 | public static Result forValue(Object value) { |
| 372 | return new Result(value, null); |
| 373 | } |
| 374 | |
| 375 | public static Result forException(Exception e) { |
| 376 | return new Result(null, e); |
| 377 | } |
| 378 | |
| 379 | public Object getOrThrow() throws Exception { |
| 380 | if (exception != null) { |
| 381 | throw exception; |
| 382 | } else { |
| 383 | return value; |
| 384 | } |
| 385 | } |
sberlin | 488284d | 2010-11-17 14:27:41 +0000 | [diff] [blame] | 386 | |
sberlin@gmail.com | ba8a4cd | 2010-11-27 01:56:53 +0000 | [diff] [blame] | 387 | private static final long serialVersionUID = 0L; |
limpbizkit | e451ef7 | 2007-12-19 01:15:59 +0000 | [diff] [blame] | 388 | } |
| 389 | } |