| /* |
| * 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 org.apache.openjpa.enhance; |
| |
| import java.lang.reflect.Field; |
| import java.lang.reflect.InvocationHandler; |
| import java.lang.reflect.Method; |
| import java.lang.reflect.Proxy; |
| |
| import org.apache.openjpa.kernel.OpenJPAStateManager; |
| import org.apache.openjpa.kernel.StateManagerImpl; |
| import org.apache.openjpa.meta.FieldMetaData; |
| import org.apache.openjpa.meta.JavaTypes; |
| import org.apache.openjpa.util.ImplHelper; |
| |
| /** |
| * Helper methods for managed types that use method redefinition for field |
| * tracking. |
| * |
| * @since 1.0.0 |
| */ |
| public class RedefinitionHelper { |
| |
| /** |
| * Call {@link StateManagerImpl#dirtyCheck} if the argument is a |
| * {@link StateManagerImpl}. |
| */ |
| public static void dirtyCheck(StateManager sm) { |
| if (sm instanceof StateManagerImpl) |
| ((StateManagerImpl) sm).dirtyCheck(); |
| } |
| |
| /** |
| * Notify the state manager for <code>o</code> (if any) that a field |
| * is about to be accessed. |
| */ |
| public static void accessingField(Object o, int absoluteIndex) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.accessingField(absoluteIndex); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, boolean cur, |
| boolean next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingBooleanField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, char cur, char next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingCharField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, byte cur, byte next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingByteField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, short cur, short next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingShortField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, int cur, int next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingIntField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, long cur, long next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingLongField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, float cur, float next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingFloatField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, double cur, |
| double next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingDoubleField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, String cur, |
| String next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingStringField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Setting state callback. |
| */ |
| public static void settingField(Object o, int idx, Object cur, |
| Object next) { |
| PersistenceCapable pc = ImplHelper.toPersistenceCapable(o, null); |
| if (pc == null) |
| return; |
| StateManager sm = pc.pcGetStateManager(); |
| if (sm != null) |
| sm.settingObjectField(pc, idx, cur, next, |
| OpenJPAStateManager.SET_USER); |
| } |
| |
| /** |
| * Create a container instance that will delegate back to the state |
| * manager to emulate lazy loading. This is used by PC subclasses for |
| * unenhanced types that could not be redefined, and thus do not have |
| * field-interception capabilities. Do this for all collection and |
| * map field types, even if they are in the dfg, in case the fetch |
| * groups are reset at runtime. |
| * |
| * @since 1.1.0 |
| */ |
| public static void assignLazyLoadProxies(StateManagerImpl sm) { |
| FieldMetaData[] fmds = sm.getMetaData().getFields(); |
| for (int i = 0; i < fmds.length; i++) { |
| switch (fmds[i].getTypeCode()) { |
| case JavaTypes.COLLECTION: |
| case JavaTypes.MAP: |
| PersistenceCapable pc = sm.getPersistenceCapable(); |
| Field field = (Field) fmds[i].getBackingMember(); |
| Reflection.set(pc, field, |
| newLazyLoadingProxy(fmds[i].getDeclaredType(), i, sm)); |
| break; |
| } |
| } |
| } |
| |
| private static Object newLazyLoadingProxy(Class type, final int idx, |
| final StateManagerImpl sm) { |
| InvocationHandler handler = new InvocationHandler() { |
| |
| @Override |
| public Object invoke(Object proxy, Method method, Object[] args) |
| throws Throwable { |
| // this will replace the field in the instance, so the dynamic |
| // proxy should only be called the first time a |
| // lazy-load-proxied field is used in normal usage. |
| Object delegate = sm.fetch(idx); |
| return method.invoke(delegate, args); |
| } |
| }; |
| return Proxy.newProxyInstance(type.getClassLoader(), |
| new Class[] { type }, handler); |
| } |
| } |