blob: 5c08618c99fb5c2e74c105a39589849ee57fa391 [file] [log] [blame]
/*
* 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;
@Override
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<>(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<>(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<>(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<>(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());
}
}
}