/* * 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.persistence.detach; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.util.ArrayList; import org.apache.openjpa.conf.Compatibility; import org.apache.openjpa.lib.log.Log; import org.apache.openjpa.persistence.OpenJPAEntityManager; import org.apache.openjpa.persistence.OpenJPAEntityManagerFactorySPI; import org.apache.openjpa.persistence.OpenJPAPersistence; import org.apache.openjpa.persistence.test.SingleEMFTestCase; public class TestDetachNoProxy extends SingleEMFTestCase { private static final int numEntities = 3; private static final String PROXY = new String("$proxy"); private Log _log; public void setUp() { setUp(DROP_TABLES, Entity20.class); _log = emf.getConfiguration().getLog("test"); createEntities(numEntities); } private void createEntities(int count) { Entity20 e20 = null; OpenJPAEntityManager em = emf.createEntityManager(); em.getTransaction().begin(); for (int i=0; i<count; i++) { e20 = new Entity20(i); em.persist(e20); } em.getTransaction().commit(); em.close(); } /* * Verify that an in-place detached entity does not use the proxy classes. */ public void testDetach20() { Integer id = new Integer(0); OpenJPAEntityManagerFactorySPI emf2 = (OpenJPAEntityManagerFactorySPI) OpenJPAPersistence.createEntityManagerFactory( "NoProxy2New", "org/apache/openjpa/persistence/detach/persistence2.xml"); assertNotNull(emf2); Log log = emf2.getConfiguration().getLog("test"); if (log.isTraceEnabled()) log.trace("***** testDetach20() *****"); if (log.isTraceEnabled()) { Compatibility compat = emf2.getConfiguration().getCompatibilityInstance(); assertNotNull(compat); log.trace("FlushBeforeDetach=" + compat.getFlushBeforeDetach()); log.trace("CopyOnDetach=" + compat.getCopyOnDetach()); log.trace("CascadeWithDetach=" + compat.getCascadeWithDetach()); log.trace("IgnoreDetachedStateFieldForProxySerialization=" + compat.getIgnoreDetachedStateFieldForProxySerialization()); } OpenJPAEntityManager em = emf2.createEntityManager(); em.clear(); Entity20 e20 = em.find(Entity20.class, id); if (log.isTraceEnabled()) log.trace("** testDetach20() - after find"); assertTrue(em.contains(e20)); assertFalse(em.isDetached(e20)); verifySerializable(e20, true, false); // new openjpa-2.0.0 behavior, where detach() doesn't return updated entity, but does it in-place em.detach(e20); if (log.isTraceEnabled()) log.trace("** testDetach20() - after detach"); // in-place updated entity should not have any proxy classes and should be detached assertFalse(em.contains(e20)); assertTrue(em.isDetached(e20)); verifySerializable(e20, true, false); em.close(); closeEMF(emf2); } /* * Verify that a detachCopy() returned entity does not contain any proxy classes. */ public void testDetachCopy20() { Integer id = new Integer(0); OpenJPAEntityManagerFactorySPI emf2 = (OpenJPAEntityManagerFactorySPI) OpenJPAPersistence.createEntityManagerFactory( "NoProxy2New", "org/apache/openjpa/persistence/detach/persistence2.xml"); assertNotNull(emf2); Log log = emf2.getConfiguration().getLog("test"); if (log.isTraceEnabled()) log.trace("***** testDetachCopy20() *****"); if (log.isTraceEnabled()) { Compatibility compat = emf2.getConfiguration().getCompatibilityInstance(); assertNotNull(compat); log.trace("FlushBeforeDetach=" + compat.getFlushBeforeDetach()); log.trace("CopyOnDetach=" + compat.getCopyOnDetach()); log.trace("CascadeWithDetach=" + compat.getCascadeWithDetach()); log.trace("IgnoreDetachedStateFieldForProxySerialization=" + compat.getIgnoreDetachedStateFieldForProxySerialization()); } OpenJPAEntityManager em = emf2.createEntityManager(); em.clear(); Entity20 e20 = em.find(Entity20.class, id); if (log.isTraceEnabled()) log.trace("** testDetachCopy20() - after find"); assertTrue(em.contains(e20)); assertFalse(em.isDetached(e20)); verifySerializable(e20, true, false); // Test new detachCopy() method added in 2.0.0 Entity20 e20copy = em.detachCopy(e20); if (log.isTraceEnabled()) log.trace("** TestDetachCopy20() - after detachCopy"); // verify e20 is same as above assertTrue(em.contains(e20)); assertFalse(em.isDetached(e20)); verifySerializable(e20, true, false); // verify copy does not have any proxy classes (in-place updated) is detached assertFalse(em.contains(e20copy)); assertTrue(em.isDetached(e20copy)); verifySerializable(e20copy, false, false); em.close(); closeEMF(emf2); } /* * Verify that in-place detachAll entities do not use the proxy classes. */ public void testDetachAll20() { OpenJPAEntityManagerFactorySPI emf2 = (OpenJPAEntityManagerFactorySPI) OpenJPAPersistence.createEntityManagerFactory( "NoProxy2New", "org/apache/openjpa/persistence/detach/persistence2.xml"); assertNotNull(emf2); Log log = emf2.getConfiguration().getLog("test"); if (log.isTraceEnabled()) log.trace("***** testDetachAll20() *****"); if (log.isTraceEnabled()) { Compatibility compat = emf2.getConfiguration().getCompatibilityInstance(); assertNotNull(compat); log.trace("FlushBeforeDetach=" + compat.getFlushBeforeDetach()); log.trace("CopyOnDetach=" + compat.getCopyOnDetach()); log.trace("CascadeWithDetach=" + compat.getCascadeWithDetach()); log.trace("IgnoreDetachedStateFieldForProxySerialization=" + compat.getIgnoreDetachedStateFieldForProxySerialization()); } OpenJPAEntityManager em = emf2.createEntityManager(); em.clear(); ArrayList<Entity20> e20List = new ArrayList<Entity20>(numEntities); for (int i=0; i<numEntities; i++) { Entity20 e20 = em.find(Entity20.class, new Integer(i)); e20List.add(e20); if (log.isTraceEnabled()) log.trace("** testDetachAll20() - after find Entity20(" + i + ")"); assertTrue(em.contains(e20)); assertFalse(em.isDetached(e20)); verifySerializable(e20, true, false); } // new openjpa-2.0.0 behavior, where detachAll() updates entities in-place // ArrayList<Entity20> e20ListCopy = new ArrayList<Entity20>(em.detachAll(e20List)); // em.detachAll(e20List); // for some reason calling with Collection causes a NPE, so use Object[] instead em.detachAll(e20List.get(0), e20List.get(1), e20List.get(2)); for (int i=0; i<numEntities; i++) { if (log.isTraceEnabled()) log.trace("** testDetachAll20() - after EM.detachAll() verify e20List(" + i + ")"); Entity20 e20 = e20List.get(i); // entity should not have any proxy classes (in-place updated) and is detached assertFalse(em.contains(e20)); assertTrue(em.isDetached(e20)); verifySerializable(e20, true, false); } em.close(); closeEMF(emf2); } /* * Verify that after EM.clear() entities still contain proxy classes for 1.0 apps. */ public void testClear10Compat() { OpenJPAEntityManagerFactorySPI emf1 = (OpenJPAEntityManagerFactorySPI) OpenJPAPersistence.createEntityManagerFactory( "NoProxy1Compat", "org/apache/openjpa/persistence/detach/persistence1.xml"); assertNotNull(emf1); Log log = emf1.getConfiguration().getLog("test"); if (log.isTraceEnabled()) log.trace("***** testClear10Compat() *****"); if (log.isTraceEnabled()) { Compatibility compat = emf1.getConfiguration().getCompatibilityInstance(); assertNotNull(compat); log.trace("FlushBeforeDetach=" + compat.getFlushBeforeDetach()); log.trace("CopyOnDetach=" + compat.getCopyOnDetach()); log.trace("CascadeWithDetach=" + compat.getCascadeWithDetach()); log.trace("IgnoreDetachedStateFieldForProxySerialization=" + compat.getIgnoreDetachedStateFieldForProxySerialization()); } OpenJPAEntityManager em = emf1.createEntityManager(); em.clear(); ArrayList<Entity20> e20List = new ArrayList<Entity20>(numEntities); for (int i=0; i<numEntities; i++) { Entity20 e20 = em.find(Entity20.class, new Integer(i)); e20List.add(e20); if (log.isTraceEnabled()) log.trace("** testClear10Compat() - after find Entity20(" + i + ")"); assertTrue(em.contains(e20)); assertFalse(em.isDetached(e20)); verifySerializable(e20, true, false); } em.clear(); for (int i=0; i<numEntities; i++) { if (log.isTraceEnabled()) log.trace("** testClear10Compat() - after EM.clear() verify Entity20(" + i + ")"); Entity20 e20 = e20List.get(i); assertFalse(em.contains(e20)); assertTrue(em.isDetached(e20)); // Old 1.0/1.2 Behavior - // the $proxy classes are not removed during serialization verifySerializable(e20, true, true); } em.close(); closeEMF(emf1); } /* * Verify that after EM.clear() entities still contain proxy classes for 1.0 apps. */ public void testClear20Compat() { OpenJPAEntityManagerFactorySPI emf2 = (OpenJPAEntityManagerFactorySPI) OpenJPAPersistence.createEntityManagerFactory( "NoProxy2Compat", "org/apache/openjpa/persistence/detach/persistence2.xml"); assertNotNull(emf2); Log log = emf2.getConfiguration().getLog("test"); if (log.isTraceEnabled()) log.trace("***** testClear20Compat() *****"); if (log.isTraceEnabled()) { Compatibility compat = emf2.getConfiguration().getCompatibilityInstance(); assertNotNull(compat); log.trace("FlushBeforeDetach=" + compat.getFlushBeforeDetach()); log.trace("CopyOnDetach=" + compat.getCopyOnDetach()); log.trace("CascadeWithDetach=" + compat.getCascadeWithDetach()); log.trace("IgnoreDetachedStateFieldForProxySerialization=" + compat.getIgnoreDetachedStateFieldForProxySerialization()); } OpenJPAEntityManager em = emf2.createEntityManager(); em.clear(); ArrayList<Entity20> e20List = new ArrayList<Entity20>(numEntities); for (int i=0; i<numEntities; i++) { Entity20 e20 = em.find(Entity20.class, new Integer(i)); e20List.add(e20); if (log.isTraceEnabled()) log.trace("** testClear20Compat() - after find Entity20(" + i + ")"); assertTrue(em.contains(e20)); assertFalse(em.isDetached(e20)); verifySerializable(e20, true, false); } em.clear(); for (int i=0; i<numEntities; i++) { if (log.isTraceEnabled()) log.trace("** testClear20Compat() - after EM.clear() verify Entity20(" + i + ")"); Entity20 e20 = e20List.get(i); assertFalse(em.contains(e20)); assertTrue(em.isDetached(e20)); // Old 1.0/1.2 Behavior - // the $proxy classes are not removed during serialization verifySerializable(e20, true, true); } em.close(); closeEMF(emf2); } /* * Verify that after EM.clear() entities do not contain proxy classes for 2.0 apps. */ public void testClear20New() { OpenJPAEntityManagerFactorySPI emf2 = (OpenJPAEntityManagerFactorySPI) OpenJPAPersistence.createEntityManagerFactory( "NoProxy2New", "org/apache/openjpa/persistence/detach/persistence2.xml"); assertNotNull(emf2); Log log = emf2.getConfiguration().getLog("test"); if (log.isTraceEnabled()) log.trace("***** testClear20New() *****"); if (log.isTraceEnabled()) { Compatibility compat = emf2.getConfiguration().getCompatibilityInstance(); assertNotNull(compat); log.trace("FlushBeforeDetach=" + compat.getFlushBeforeDetach()); log.trace("CopyOnDetach=" + compat.getCopyOnDetach()); log.trace("CascadeWithDetach=" + compat.getCascadeWithDetach()); log.trace("IgnoreDetachedStateFieldForProxySerialization=" + compat.getIgnoreDetachedStateFieldForProxySerialization()); } OpenJPAEntityManager em = emf2.createEntityManager(); em.clear(); ArrayList<Entity20> e20List = new ArrayList<Entity20>(numEntities); for (int i=0; i<numEntities; i++) { Entity20 e20 = em.find(Entity20.class, new Integer(i)); e20List.add(e20); if (log.isTraceEnabled()) log.trace("** testClear20New() - after find Entity20(" + i + ")"); assertTrue(em.contains(e20)); assertFalse(em.isDetached(e20)); verifySerializable(e20, true, false); } em.clear(); for (int i=0; i<numEntities; i++) { if (log.isTraceEnabled()) log.trace("** testClear20New() - after EM.clear() verify Entity20(" + i + ")"); Entity20 e20 = e20List.get(i); assertFalse(em.contains(e20)); assertTrue(em.isDetached(e20)); // OPENJPA-1097 New behavior - $proxy classes are removed verifySerializable(e20, true, false); } em.close(); closeEMF(emf2); } /** * Test that the entity is/is not using our $proxy classes before * and after serialization. * * @param e20 Entity to test. * @param usesProxyBefore verify that the entity uses the $proxy classes * before serialization if true and does not if false. * @param usesProxyAfter verify that the entity uses the $proxy classes * after serialization if true and does not if false. */ private void verifySerializable(Entity20 e20, boolean usesProxyBefore, boolean usesProxyAfter) { ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = null; byte[] e20bytes = null; if (_log.isTraceEnabled()) _log.trace("verifySerializable() - before serialize"); verifyEntities(e20, usesProxyBefore); // first serialize try { oos = new ObjectOutputStream(baos); oos.writeObject(e20); e20bytes = baos.toByteArray(); } catch (IOException e) { fail(e.toString()); } finally { try { if (oos != null) oos.close(); } catch (IOException e) { } } // then deserialize and assert no $proxy classes exist ByteArrayInputStream bais = new ByteArrayInputStream(e20bytes); ObjectInputStream ois = null; Entity20 e20new = null; try { ois = new ObjectInputStream(bais); e20new = (Entity20) ois.readObject(); if (_log.isTraceEnabled()) _log.trace("verifySerializable() - after deserialize"); verifyEntities(e20new, usesProxyAfter); } catch (IOException e) { fail(e.toString()); } catch (ClassNotFoundException e) { fail(e.toString()); } finally { try { if (ois != null) ois.close(); } catch (IOException e) { } } } private void verifyEntities(Entity20 e20, boolean usesProxy) { if (_log.isTraceEnabled()) { _log.trace("verifyEntities() - asserting expected proxy usage is " + usesProxy); printClassNames(e20); } assertTrue("Expected sqlDate endsWith($proxy) to return " + usesProxy, usesProxy == e20.getDate().getClass().getCanonicalName().endsWith(PROXY)); assertTrue("Expected sqlTime endsWith($proxy) to return " + usesProxy, usesProxy == e20.getTime().getClass().getCanonicalName().endsWith(PROXY)); assertTrue("Expected sqlTimestamp endsWith($proxy) to return " + usesProxy, usesProxy == e20.getTimestamp().getClass().getCanonicalName().endsWith(PROXY)); } private void printClassNames(Entity20 e20) { if (_log.isTraceEnabled()) { _log.trace("sqlDate = " + e20.getDate().getClass().getCanonicalName()); _log.trace("sqlTime = " + e20.getTime().getClass().getCanonicalName()); _log.trace("sqlTimestamp = " + e20.getTimestamp().getClass().getCanonicalName()); } } }