| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you 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.persist.jpa; |
| |
| import static com.google.inject.matcher.Matchers.annotatedWith; |
| import static com.google.inject.matcher.Matchers.any; |
| |
| import java.lang.reflect.AccessibleObject; |
| import java.lang.reflect.InvocationHandler; |
| import java.lang.reflect.Method; |
| import java.lang.reflect.Proxy; |
| import java.util.List; |
| import java.util.Map; |
| |
| import javax.persistence.EntityManager; |
| import javax.persistence.EntityManagerFactory; |
| |
| import org.aopalliance.intercept.MethodInterceptor; |
| import org.aopalliance.intercept.MethodInvocation; |
| import org.apache.ambari.server.orm.AmbariJpaLocalTxnInterceptor; |
| import org.apache.ambari.server.orm.AmbariLocalSessionInterceptor; |
| import org.apache.ambari.server.orm.RequiresSession; |
| |
| import com.google.common.collect.Lists; |
| import com.google.inject.Inject; |
| import com.google.inject.Provides; |
| import com.google.inject.Singleton; |
| import com.google.inject.TypeLiteral; |
| import com.google.inject.persist.PersistModule; |
| import com.google.inject.persist.PersistService; |
| import com.google.inject.persist.UnitOfWork; |
| import com.google.inject.persist.finder.DynamicFinder; |
| import com.google.inject.persist.finder.Finder; |
| |
| /** |
| * Copy of guice persist module for local modifications |
| */ |
| public class AmbariJpaPersistModule extends PersistModule { |
| private final String jpaUnit; |
| |
| public AmbariJpaPersistModule(String jpaUnit) { |
| if (null != jpaUnit && jpaUnit.length() > 0) { |
| this.jpaUnit = jpaUnit; |
| } else { |
| throw new IllegalArgumentException("jpaUnit should not be null"); |
| } |
| } |
| |
| private Map<?, ?> properties; |
| private MethodInterceptor transactionInterceptor; |
| |
| @Override protected void configurePersistence() { |
| bindConstant().annotatedWith(Jpa.class).to(jpaUnit); |
| |
| bind(AmbariJpaPersistService.class).in(Singleton.class); |
| |
| bind(PersistService.class).to(AmbariJpaPersistService.class); |
| bind(UnitOfWork.class).to(AmbariJpaPersistService.class); |
| bind(EntityManager.class).toProvider(AmbariJpaPersistService.class); |
| bind(EntityManagerFactory.class).toProvider(JpaPersistService.EntityManagerFactoryProvider.class); |
| |
| transactionInterceptor = new AmbariJpaLocalTxnInterceptor(); |
| requestInjection(transactionInterceptor); |
| |
| MethodInterceptor sessionInterceptor = new AmbariLocalSessionInterceptor(); |
| requestInjection(sessionInterceptor); |
| |
| // Bind dynamic finders. |
| for (Class<?> finder : dynamicFinders) { |
| bindFinder(finder); |
| } |
| |
| bindInterceptor(annotatedWith(RequiresSession.class), any(), sessionInterceptor); |
| bindInterceptor(any(), annotatedWith(RequiresSession.class), sessionInterceptor); |
| } |
| |
| |
| |
| @Override protected MethodInterceptor getTransactionInterceptor() { |
| return transactionInterceptor; |
| } |
| |
| @Provides @Jpa Map<?, ?> provideProperties() { |
| return properties; |
| } |
| |
| /** |
| * Configures the JPA persistence provider with a set of properties. |
| * |
| * @param properties A set of name value pairs that configure a JPA persistence |
| * provider as per the specification. |
| */ |
| public AmbariJpaPersistModule properties(Map<?, ?> properties) { |
| this.properties = properties; |
| return this; |
| } |
| |
| private final List<Class<?>> dynamicFinders = Lists.newArrayList(); |
| |
| /** |
| * Adds an interface to this module to use as a dynamic finder. |
| * |
| * @param iface Any interface type whose methods are all dynamic finders. |
| */ |
| public <T> AmbariJpaPersistModule addFinder(Class<T> iface) { |
| dynamicFinders.add(iface); |
| return this; |
| } |
| |
| private <T> void bindFinder(Class<T> iface) { |
| if (!isDynamicFinderValid(iface)) { |
| return; |
| } |
| |
| InvocationHandler finderInvoker = new InvocationHandler() { |
| @Inject |
| JpaFinderProxy finderProxy; |
| |
| @Override |
| public Object invoke(final Object thisObject, final Method method, final Object[] args) |
| throws Throwable { |
| |
| // Don't intercept non-finder methods like equals and hashcode. |
| if (!method.isAnnotationPresent(Finder.class)) { |
| // This is not ideal, we are using the invocation handler's equals |
| // and hashcode as a proxy (!) for the proxy's equals and hashcode. |
| return method.invoke(this, args); |
| } |
| |
| return finderProxy.invoke(new MethodInvocation() { |
| @Override |
| public Method getMethod() { |
| return method; |
| } |
| |
| @Override |
| public Object[] getArguments() { |
| return null == args ? new Object[0] : args; |
| } |
| |
| @Override |
| public Object proceed() throws Throwable { |
| return method.invoke(thisObject, args); |
| } |
| |
| @Override |
| public Object getThis() { |
| throw new UnsupportedOperationException("Bottomless proxies don't expose a this."); |
| } |
| |
| @Override |
| public AccessibleObject getStaticPart() { |
| throw new UnsupportedOperationException(); |
| } |
| }); |
| } |
| }; |
| requestInjection(finderInvoker); |
| |
| @SuppressWarnings("unchecked") // Proxy must produce instance of type given. |
| T proxy = (T) Proxy |
| .newProxyInstance(Thread.currentThread().getContextClassLoader(), new Class<?>[] { iface }, |
| finderInvoker); |
| |
| bind(iface).toInstance(proxy); |
| } |
| |
| private boolean isDynamicFinderValid(Class<?> iface) { |
| boolean valid = true; |
| if (!iface.isInterface()) { |
| addError(iface + " is not an interface. Dynamic Finders must be interfaces."); |
| valid = false; |
| } |
| |
| for (Method method : iface.getMethods()) { |
| DynamicFinder finder = DynamicFinder.from(method); |
| if (null == finder) { |
| addError("Dynamic Finder methods must be annotated with @Finder, but " + iface |
| + "." + method.getName() + " was not"); |
| valid = false; |
| } |
| } |
| return valid; |
| } |
| |
| private class LocalTypeLiteral<T> extends TypeLiteral<T> { |
| } |
| } |