ru.vyarus.guice.persist.orient.RepositoryModule Maven / Gradle / Ivy
package ru.vyarus.guice.persist.orient;
import com.google.common.base.Objects;
import com.google.inject.AbstractModule;
import com.google.inject.matcher.AbstractMatcher;
import com.google.inject.matcher.Matchers;
import com.google.inject.multibindings.Multibinder;
import com.google.inject.name.Names;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import ru.vyarus.guice.persist.orient.db.DbType;
import ru.vyarus.guice.persist.orient.repository.RepositoryMethodInterceptor;
import ru.vyarus.guice.persist.orient.repository.core.MethodDefinitionException;
import ru.vyarus.guice.persist.orient.repository.core.executor.RepositoryExecutor;
import ru.vyarus.guice.persist.orient.repository.core.executor.impl.DocumentRepositoryExecutor;
import ru.vyarus.guice.persist.orient.repository.core.ext.service.result.converter.ResultConverter;
import ru.vyarus.guice.persist.orient.repository.core.ext.util.ExtUtils;
import ru.vyarus.guice.persist.orient.repository.core.util.RepositoryUtils;
import javax.inject.Singleton;
import java.lang.reflect.Method;
/**
* Module provides support for spring-data like repositories. Must be used together with main orient module.
* It's not limited to repository pattern and may be used to usual daos. Repositories usage is very similar
* to spring-data (the most popular repository implementation). Annotations intentionally named the same way
* as in spring-data.
* In contrast to spring, which use instance proxies, guice aop is tied to class proxies. So aop could be
* applied to bean only if guice creates it's instance. Repositories implementation completely relies on guice aop.
* To make it work for interfaces and abstract classes, special class proxy must be generated (to let guice properly
* instantiate normal class). Implementation of this mechanism is in
* separate project.
* Repositories based on plugin architecture, so new types could be easily added and existing types extended.
* All extensions are annotation driven (annotation must be marked with special annotation, containing
* extension class). Repository methods are handled with guice aop, bu simply searching for annotated annotations (
* {@link ru.vyarus.guice.persist.orient.repository.core.spi.method.RepositoryMethod}). Two more generic types
* of extensions supported: parameter extensions, when some parameters should get special meaning. And
* amend extensions, which may be used to amend method call behaviour (e.g. set timeout).
* Repository method processing starts with method descriptor creation. During descriptor creation, extension
* could analyze method definition correctness and prepare all required information for fast execution. Generated
* descriptors are cached to reuse on future calls. In order to process method, extension is called with prepared
* descriptor.
* After execution result could be automatically converted (e.g. between collections, arrays, get first element
* of result list, etc.).
* See {@link ru.vyarus.guice.persist.orient.repository.core.ext.service.result.converter.ResultConverter}.
*
* Based on guice-persist jpa module {@link com.google.inject.persist.jpa.JpaPersistModule}
*
* @author Vyacheslav Rusakov
* @since 30.07.2014
*/
public class RepositoryModule extends AbstractModule {
private final Logger logger = LoggerFactory.getLogger(RepositoryModule.class);
private DbType defaultConnectionToUse = DbType.DOCUMENT;
private Multibinder executorsMultibinder;
/**
* By default document connection is used.
*
* @param connection connection type to use for ambiguous cases (when it's impossible to recognize)
* @return module itself for chained calls
*/
public RepositoryModule defaultConnectionType(final DbType connection) {
this.defaultConnectionToUse = Objects.firstNonNull(connection, defaultConnectionToUse);
return this;
}
@Override
protected void configure() {
bind(DbType.class).annotatedWith(Names.named("orient.repository.default.connection"))
.toInstance(defaultConnectionToUse);
// extension points
bind(ResultConverter.class);
final RepositoryMethodInterceptor proxy = new RepositoryMethodInterceptor();
requestInjection(proxy);
// repository specific method annotations (query, function, delegate, etc.)
bindInterceptor(Matchers.any(), new AbstractMatcher() {
@Override
public boolean matches(final Method method) {
// this will throw error if two or more annotations specified (fail fast)
try {
return ExtUtils.findMethodAnnotation(method) != null;
} catch (Exception ex) {
throw new MethodDefinitionException(String.format("Error declaration on method %s",
RepositoryUtils.methodToString(method)), ex);
}
}
}, proxy);
executorsMultibinder = Multibinder.newSetBinder(binder(), RepositoryExecutor.class);
configureExecutors();
}
/**
* Configures executors, used to execute queries in different connection types.
* Override to register different executors.
*/
protected void configureExecutors() {
bindExecutor(DocumentRepositoryExecutor.class);
loadOptionalExecutor("ru.vyarus.guice.persist.orient.support.repository.ObjectExecutorBinder");
loadOptionalExecutor("ru.vyarus.guice.persist.orient.support.repository.GraphExecutorBinder");
}
/**
* Register executor for specific connection type.
*
* @param executor executor type
*/
protected void bindExecutor(final Class extends RepositoryExecutor> executor) {
bind(executor).in(Singleton.class);
executorsMultibinder.addBinding().to(executor);
}
/**
* Allows to load executor only if required jars are in classpath.
* For example, no need for graph dependencies if only object db is used.
*
* @param executorBinder executor binder class
* @see ru.vyarus.guice.persist.orient.support.repository.ObjectExecutorBinder as example
*/
protected void loadOptionalExecutor(final String executorBinder) {
try {
final Method bindExecutor = RepositoryModule.class.getDeclaredMethod("bindExecutor", Class.class);
bindExecutor.setAccessible(true);
try {
Class.forName(executorBinder)
.getConstructor(RepositoryModule.class, Method.class)
.newInstance(this, bindExecutor);
} finally {
bindExecutor.setAccessible(false);
}
} catch (Exception ignore) {
if (logger.isTraceEnabled()) {
logger.trace("Failed to process executor loader " + executorBinder, ignore);
}
}
}
}