diff --git a/openjpa-kernel/src/main/java/org/apache/openjpa/enhance/PCEnhancer.java b/openjpa-kernel/src/main/java/org/apache/openjpa/enhance/PCEnhancer.java index 52b4572a1..5b056f928 100644 --- a/openjpa-kernel/src/main/java/org/apache/openjpa/enhance/PCEnhancer.java +++ b/openjpa-kernel/src/main/java/org/apache/openjpa/enhance/PCEnhancer.java @@ -118,7 +118,10 @@ public class PCEnhancer { public static final int ENHANCE_INTERFACE = 2 << 1; public static final int ENHANCE_PC = 2 << 2; - private static final String PRE = "pc"; + public static final String PRE = "pc"; + public static final String ISDETACHEDSTATEDEFINITIVE = PRE + + "isDetachedStateDefinitive"; + private static final Class PCTYPE = PersistenceCapable.class; private static final String SM = PRE + "StateManager"; private static final Class SMTYPE = StateManager.class; @@ -2999,12 +3002,27 @@ public class PCEnhancer { */ private void addIsDetachedMethod() throws NoSuchMethodException { - // public boolean pcIsDetached () + // public boolean pcIsDetached() BCMethod method = _pc.declareMethod(PRE + "IsDetached", Boolean.class, null); method.makePublic(); Code code = method.getCode(true); - writeIsDetachedMethod(code); + boolean needsDefinitiveMethod = writeIsDetachedMethod(code); + code.calculateMaxStack(); + code.calculateMaxLocals(); + if (!needsDefinitiveMethod) + return; + + // private boolean pcIsDetachedStateDefinitive() + // return false; + // auxilliary enhancers may change the return value of this method + // if their specs consider detached state definitive + method = _pc.declareMethod(ISDETACHEDSTATEDEFINITIVE, boolean.class, + null); + method.makePrivate(); + code = method.getCode(true); + code.constant().setValue(false); + code.ireturn(); code.calculateMaxStack(); code.calculateMaxLocals(); } @@ -3012,14 +3030,17 @@ public class PCEnhancer { /** * Creates the body of the pcIsDetached() method to determine if an * instance is detached. + * + * @return true if we need a pcIsDetachedStateDefinitive method, false + * otherwise */ - private void writeIsDetachedMethod(Code code) + private boolean writeIsDetachedMethod(Code code) throws NoSuchMethodException { // not detachable: return Boolean.FALSE if (!_meta.isDetachable()) { code.getstatic().setField(Boolean.class, "FALSE", Boolean.class); code.areturn(); - return; + return false; } // if (sm != null) @@ -3067,7 +3088,7 @@ public class PCEnhancer { ifins.setTarget(target); notdeser.setTarget(target); code.areturn(); - return; + return false; } } @@ -3077,9 +3098,9 @@ public class PCEnhancer { if (notdeser != null) notdeser.setTarget(target); - // allow users with version fields to manually construct a "detached" - // instance, so check version before taking into account non-existent - // detached state + // allow users with version or auto-assigned pk fields to manually + // construct a "detached" instance, so check these before taking into + // account non-existent detached state // consider detached if version is non-default FieldMetaData version = _meta.getVersionField(); @@ -3094,41 +3115,13 @@ public class PCEnhancer { ifins.setTarget(code.getstatic().setField(Boolean.class, "FALSE", Boolean.class)); code.areturn(); - return; - } - - // no detached state: if instance uses detached state and it's not - // synthetic or the instance is not serializable or the state isn't - // transient, must not be detached - if (state == null - && (!ClassMetaData.SYNTHETIC.equals(_meta.getDetachedState()) - || !Serializable.class.isAssignableFrom(_meta.getDescribedType()) - || !_repos.getConfiguration().getDetachStateInstance(). - isDetachedStateTransient())) { - // return Boolean.FALSE - code.getstatic().setField(Boolean.class, "FALSE", Boolean.class); - code.areturn(); - return; - } - - // no detached state: if instance uses detached state (and must be - // synthetic and transient in serializable instance at this point), - // not detached if state not set to DESERIALIZED - if (state == null) { - // if (pcGetDetachedState () == null) // instead of DESERIALIZED - // return Boolean.FALSE; - loadManagedInstance(code, false); - code.invokevirtual().setMethod(PRE + "GetDetachedState", - Object.class, null); - ifins = code.ifnonnull(); - code.getstatic().setField(Boolean.class, "FALSE", Boolean.class); - code.areturn(); - ifins.setTarget(code.nop()); + return false; } // consider detached if auto-genned primary keys are non-default ifins = null; JumpInstruction ifins2 = null; + boolean hasAutoAssignedPK = false; if (state != Boolean.TRUE && _meta.getIdentityType() == ClassMetaData.ID_APPLICATION) { // for each pk field: @@ -3162,13 +3155,65 @@ public class PCEnhancer { } } - // give up; we just don't know - target = code.constant().setNull(); + // create artificial target to simplify + target = code.nop(); if (ifins != null) ifins.setTarget(target); if (ifins2 != null) ifins2.setTarget(target); + + // if has auto-assigned pk and we get to this point, must have default + // value, so must be new instance + if (hasAutoAssignedPK) { + code.getstatic().setField(Boolean.class, "FALSE", Boolean.class); + code.areturn(); + return false; + } + + // if detached state is not definitive, just give up now and return + // null so that the runtime will perform a DB lookup to determine + // whether we're detached or new + code.aload().setThis(); + code.invokespecial().setMethod(ISDETACHEDSTATEDEFINITIVE, boolean.class, + null); + ifins = code.ifne(); + code.constant().setNull(); code.areturn(); + ifins.setTarget(code.nop()); + + // no detached state: if instance uses detached state and it's not + // synthetic or the instance is not serializable or the state isn't + // transient, must not be detached + if (state == null + && (!ClassMetaData.SYNTHETIC.equals(_meta.getDetachedState()) + || !Serializable.class.isAssignableFrom(_meta.getDescribedType()) + || !_repos.getConfiguration().getDetachStateInstance(). + isDetachedStateTransient())) { + // return Boolean.FALSE + code.getstatic().setField(Boolean.class, "FALSE", Boolean.class); + code.areturn(); + return true; + } + + // no detached state: if instance uses detached state (and must be + // synthetic and transient in serializable instance at this point), + // not detached if state not set to DESERIALIZED + if (state == null) { + // if (pcGetDetachedState () == null) // instead of DESERIALIZED + // return Boolean.FALSE; + loadManagedInstance(code, false); + code.invokevirtual().setMethod(PRE + "GetDetachedState", + Object.class, null); + ifins = code.ifnonnull(); + code.getstatic().setField(Boolean.class, "FALSE", Boolean.class); + code.areturn(); + ifins.setTarget(code.nop()); + } + + // give up; we just don't know + code.constant().setNull(); + code.areturn(); + return true; } /** diff --git a/openjpa-kernel/src/main/java/org/apache/openjpa/kernel/VersionAttachStrategy.java b/openjpa-kernel/src/main/java/org/apache/openjpa/kernel/VersionAttachStrategy.java index 7fa741f7b..c01dcb2fa 100644 --- a/openjpa-kernel/src/main/java/org/apache/openjpa/kernel/VersionAttachStrategy.java +++ b/openjpa-kernel/src/main/java/org/apache/openjpa/kernel/VersionAttachStrategy.java @@ -69,19 +69,12 @@ class VersionAttachStrategy public Object attach(AttachManager manager, Object toAttach, ClassMetaData meta, PersistenceCapable into, OpenJPAStateManager owner, ValueMetaData ownerMeta, boolean explicit) { - - // VersionAttachStrategy is invoked in the case where no more - // intelligent strategy could be found; let's be more lenient - // about new vs. detached record determination. - if (into == null) - into = findFromDatabase(manager, toAttach); - BrokerImpl broker = manager.getBroker(); PersistenceCapable pc = ImplHelper.toPersistenceCapable(toAttach, meta.getRepository().getConfiguration()); boolean embedded = ownerMeta != null && ownerMeta.isEmbeddedPC(); - boolean isNew = !broker.isDetached(pc) && into == null; + boolean isNew = !broker.isDetached(pc); Object version = null; StateManagerImpl sm; diff --git a/openjpa-kernel/src/main/resources/org/apache/openjpa/kernel/localizer.properties b/openjpa-kernel/src/main/resources/org/apache/openjpa/kernel/localizer.properties index cfeac3d4f..5c8d73732 100644 --- a/openjpa-kernel/src/main/resources/org/apache/openjpa/kernel/localizer.properties +++ b/openjpa-kernel/src/main/resources/org/apache/openjpa/kernel/localizer.properties @@ -116,7 +116,9 @@ not-managed: The given instance "{0}" is not managed by this context. trans-not-managed: This broker is not configured to use managed transactions. bad-detached-op: You cannot perform operation {0} on detached object "{1}". \ This operation only applies to managed objects. -persist-detached: Attempt to persist detached object "{0}". +persist-detached: Attempt to persist detached object "{0}". If this is a new \ + instance, make sure any versino and/or auto-generated primary key fields are \ + null/default when persisting. null-value: The field "{0}" of instance "{1}" contained a null value; \ the metadata for this field specifies that nulls are illegal. change-identity: Attempt to change a primary key field of an instance that \ @@ -396,4 +398,4 @@ cant-serialize-pessimistic-broker: Serialization not allowed for brokers with \ an active datastore (pessimistic) transaction. cant-serialize-connected-broker: Serialization not allowed for brokers with \ an active connection to the database. -no-interface-metadata: No metadata was found for managed interface {0}. \ No newline at end of file +no-interface-metadata: No metadata was found for managed interface {0}. diff --git a/openjpa-persistence-jdbc/src/test/java/org/apache/openjpa/persistence/detachment/TestAttachConstructedCopy.java b/openjpa-persistence-jdbc/src/test/java/org/apache/openjpa/persistence/detachment/TestAttachConstructedCopy.java new file mode 100755 index 000000000..1af43571a --- /dev/null +++ b/openjpa-persistence-jdbc/src/test/java/org/apache/openjpa/persistence/detachment/TestAttachConstructedCopy.java @@ -0,0 +1,72 @@ +/* + * 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.detachment; + +import javax.persistence.EntityManager; + +import junit.textui.TestRunner; +import org.apache.openjpa.persistence.test.SingleEMFTestCase; + +/** + * Test that manually constructing instances with existing primary key values + * and attaching them works. + * + * @author Abe White + */ +public class TestAttachConstructedCopy + extends SingleEMFTestCase { + + public void setUp() { + setUp(Record.class); + } + + public void testAttachConstructedCopyWithGeneratedPKAndNoVersion() { + Record record = new Record(); + record.setContent("orig"); + + EntityManager em = emf.createEntityManager(); + em.getTransaction().begin(); + em.persist(record); + em.getTransaction().commit(); + em.close(); + int id = record.getId(); + + Record copy = new Record(); + copy.setId(id); + copy.setContent("new"); + + em = emf.createEntityManager(); + em.getTransaction().begin(); + record = em.merge(copy); + assertTrue(record != copy); + assertEquals("new", record.getContent()); + em.getTransaction().commit(); + em.close(); + + em = emf.createEntityManager(); + record = em.find(Record.class, id); + assertEquals("new", record.getContent()); + em.close(); + } + + public static void main(String[] args) { + TestRunner.run(TestAttachConstructedCopy.class); + } +} + diff --git a/openjpa-project/src/doc/manual/ref_guide_remote.xml b/openjpa-project/src/doc/manual/ref_guide_remote.xml index 7c3dffede..d0dc4bac6 100644 --- a/openjpa-project/src/doc/manual/ref_guide_remote.xml +++ b/openjpa-project/src/doc/manual/ref_guide_remote.xml @@ -195,7 +195,10 @@ attached instance's corresponding fields to null. If the instance has a Version field, OpenJPA will consider the object detached if the version field has a non-default -value, and new otherwise. +value, and new otherwise. Similarly, if the instance has +GeneratedValue primary key fields, OpenJPA will consider the +object detached if any of these fields have non-default values, and new +otherwise. When attaching null fields in these cases, OpenJPA cannot distinguish between a