blob: c9c17f025bfd858263b56e2e5f170fa152cabd4e [file] [log] [blame]
/**
* Copyright (C) 2008 Google Inc.
*
* 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
*
* 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 com.google.inject.commands;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.inject.Binder;
import com.google.inject.Key;
import com.google.inject.Module;
import com.google.inject.Provider;
import com.google.inject.Scope;
import com.google.inject.Stage;
import com.google.inject.TypeLiteral;
import com.google.inject.binder.AnnotatedBindingBuilder;
import com.google.inject.binder.AnnotatedConstantBindingBuilder;
import com.google.inject.matcher.Matcher;
import com.google.inject.spi.Message;
import com.google.inject.spi.SourceProviders;
import static com.google.inject.spi.SourceProviders.defaultSource;
import com.google.inject.spi.TypeConverter;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Set;
import org.aopalliance.intercept.MethodInterceptor;
/**
* Records commands executed by a module so they can be inspected or
* {@link CommandReplayer replayed}.
*
* @author jessewilson@google.com (Jesse Wilson)
*/
public final class CommandRecorder {
private Stage currentStage = Stage.DEVELOPMENT;
private final EarlyRequestsProvider earlyRequestsProvider;
static {
SourceProviders.skip(RecordingBinder.class);
}
/**
* @param earlyRequestsProvider satisfies requests to
* {@link Binder#getProvider} at module execution time. For modules that
* will be used to create an injector, use {@link FutureInjector}.
*/
public CommandRecorder(EarlyRequestsProvider earlyRequestsProvider) {
this.earlyRequestsProvider = earlyRequestsProvider;
}
/**
* Sets the stage reported by the binder.
*/
public void setCurrentStage(Stage currentStage) {
this.currentStage = currentStage;
}
/**
* Records the commands executed by {@code modules}.
*/
public List<Command> recordCommands(Module... modules) {
return recordCommands(Arrays.asList(modules));
}
/**
* Records the commands executed by {@code modules}.
*/
public List<Command> recordCommands(Iterable<Module> modules) {
RecordingBinder binder = new RecordingBinder();
for (Module module : modules) {
binder.install(module);
}
return Collections.unmodifiableList(binder.commands);
}
private class RecordingBinder implements Binder {
private final Set<Module> modules = Sets.newHashSet();
private final List<Command> commands = Lists.newArrayList();
public void bindInterceptor(
Matcher<? super Class<?>> classMatcher,
Matcher<? super Method> methodMatcher,
MethodInterceptor... interceptors) {
commands.add(new BindInterceptorCommand(
defaultSource(), classMatcher, methodMatcher, interceptors));
}
public void bindScope(Class<? extends Annotation> annotationType, Scope scope) {
commands.add(new BindScopeCommand(defaultSource(), annotationType, scope));
}
public void requestStaticInjection(Class<?>... types) {
commands.add(new RequestStaticInjectionCommand(defaultSource(), types));
}
public void install(Module module) {
if (modules.add(module)) {
module.configure(this);
}
}
public Stage currentStage() {
return currentStage;
}
public void addError(String message, Object... arguments) {
commands.add(new AddMessageErrorCommand(defaultSource(), message, arguments));
}
public void addError(Throwable t) {
commands.add(new AddThrowableErrorCommand(defaultSource(), t));
}
public void addError(Message message) {
throw new UnsupportedOperationException("TODO");
}
public <T> BindCommand<T>.BindingBuilder bind(Key<T> key) {
BindCommand<T> bindCommand = new BindCommand<T>(defaultSource(), key);
commands.add(bindCommand);
return bindCommand.bindingBuilder(RecordingBinder.this);
}
public <T> AnnotatedBindingBuilder<T> bind(TypeLiteral<T> typeLiteral) {
return bind(Key.get(typeLiteral));
}
public <T> AnnotatedBindingBuilder<T> bind(Class<T> type) {
return bind(Key.get(type));
}
public AnnotatedConstantBindingBuilder bindConstant() {
BindConstantCommand bindConstantCommand = new BindConstantCommand(defaultSource());
commands.add(bindConstantCommand);
return bindConstantCommand.bindingBuilder(RecordingBinder.this);
}
public <T> Provider<T> getProvider(final Key<T> key) {
commands.add(new GetProviderCommand<T>(defaultSource(), key, earlyRequestsProvider));
return new Provider<T>() {
public T get() {
return earlyRequestsProvider.get(key);
}
@Override public String toString() {
return "Provider<" + key.getTypeLiteral() + ">";
}
};
}
public <T> Provider<T> getProvider(Class<T> type) {
return getProvider(Key.get(type));
}
public void convertToTypes(Matcher<? super TypeLiteral<?>> typeMatcher,
TypeConverter converter) {
commands.add(new ConvertToTypesCommand(defaultSource(), typeMatcher, converter));
}
@Override public String toString() {
return "Binder";
}
}
}