Skip to content

Commit 1d1f395

Browse files
committed
Improve detection of entity state
Hibernate use 0 or user provided positive initial value as seed of integer version. EclipseLink always use 1 as seed of integer version.
1 parent d08b69d commit 1d1f395

File tree

5 files changed

+107
-2
lines changed

5 files changed

+107
-2
lines changed

spring-data-jpa/src/main/java/org/springframework/data/jpa/repository/support/JpaMetamodelEntityInformation.java

+28-2
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,7 @@
5555
* @author Mark Paluch
5656
* @author Jens Schauder
5757
* @author Greg Turnquist
58+
* @author Yanming Zhou
5859
*/
5960
public class JpaMetamodelEntityInformation<T, ID> extends JpaEntityInformationSupport<T, ID> {
6061

@@ -219,13 +220,38 @@ public Object getCompositeIdAttributeValue(Object id, String idAttribute) {
219220
@Override
220221
public boolean isNew(T entity) {
221222

222-
if (versionAttribute.isEmpty()
223-
|| versionAttribute.map(Attribute::getJavaType).map(Class::isPrimitive).orElse(false)) {
223+
if (versionAttribute.isEmpty()) {
224224
return super.isNew(entity);
225225
}
226226

227227
BeanWrapper wrapper = new DirectFieldAccessFallbackBeanWrapper(entity);
228228

229+
if (versionAttribute.map(Attribute::getJavaType).map(Class::isPrimitive).orElse(false)) {
230+
return versionAttribute.map(it -> {
231+
Object version = wrapper.getPropertyValue(it.getName());
232+
if (version instanceof Number value) {
233+
PersistenceProvider provider = PersistenceProvider.fromMetamodel(metamodel);
234+
if (provider == PersistenceProvider.HIBERNATE) {
235+
// Hibernate use 0 or user provided positive initial value as seed of integer version
236+
if (value.longValue() < 0) {
237+
// see org.hibernate.engine.internal.Versioning#isNullInitialVersion()
238+
return true;
239+
}
240+
// TODO Compare version to initial value (field value of transient entity)
241+
// It's unknown if equals because entity maybe transient or just persisted
242+
// But it's absolute not new if not equals
243+
} else if (provider == PersistenceProvider.ECLIPSELINK) {
244+
// EclipseLink always use 1 as seed of integer version
245+
if (value.longValue() < 1) {
246+
return true;
247+
}
248+
// It's unknown because the value may be initial value or persisted entity version
249+
}
250+
}
251+
return super.isNew(entity);
252+
}).orElseThrow(); // should never throw
253+
}
254+
229255
return versionAttribute.map(it -> wrapper.getPropertyValue(it.getName()) == null).orElse(true);
230256
}
231257

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
/*
2+
* Copyright 2013-2024 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.jpa.domain.sample;
17+
18+
import jakarta.persistence.Entity;
19+
import jakarta.persistence.Id;
20+
import jakarta.persistence.Version;
21+
22+
/**
23+
* @author Yanming Zhou
24+
*/
25+
@Entity
26+
public class SampleWithPrimitiveVersion {
27+
28+
@Id private Long id;
29+
30+
@Version private long version = -1;
31+
32+
public void setId(Long id) {
33+
this.id = id;
34+
}
35+
36+
public long getVersion() {
37+
return version;
38+
}
39+
40+
public void setVersion(long version) {
41+
this.version = version;
42+
}
43+
}

spring-data-jpa/src/test/java/org/springframework/data/jpa/repository/support/EclipseLinkJpaMetamodelEntityInformationIntegrationTests.java

+20
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,8 @@
2424
import org.junit.jupiter.api.Test;
2525
import org.junit.jupiter.api.extension.ExtendWith;
2626
import org.springframework.data.jpa.domain.AbstractPersistable;
27+
import org.springframework.data.jpa.domain.sample.SampleWithPrimitiveVersion;
28+
import org.springframework.data.repository.core.EntityInformation;
2729
import org.springframework.test.context.ContextConfiguration;
2830
import org.springframework.test.context.junit.jupiter.SpringExtension;
2931

@@ -33,6 +35,7 @@
3335
* @author Oliver Gierke
3436
* @author Jens Schauder
3537
* @author Greg Turnquist
38+
* @author Yanming Zhou
3639
*/
3740
@ExtendWith(SpringExtension.class)
3841
@ContextConfiguration({ "classpath:infrastructure.xml", "classpath:eclipselink.xml" })
@@ -79,4 +82,21 @@ void correctlyDeterminesIdValueForNestedIdClassesWithNonPrimitiveNonManagedType(
7982
@Override
8083
@Disabled
8184
void prefersPrivateGetterOverFieldAccess() {}
85+
86+
@Override
87+
@Disabled
88+
// superseded by #nonPositiveVersionedEntityIsNew()
89+
void considersEntityAsNotNewWhenHavingIdSetAndUsingPrimitiveTypeForVersionProperty() {}
90+
91+
@Test
92+
void nonPositiveVersionedEntityIsNew() {
93+
EntityInformation<SampleWithPrimitiveVersion, Long> information = new JpaMetamodelEntityInformation<>(SampleWithPrimitiveVersion.class,
94+
em.getMetamodel(), em.getEntityManagerFactory().getPersistenceUnitUtil());
95+
96+
SampleWithPrimitiveVersion entity = new SampleWithPrimitiveVersion();
97+
entity.setId(23L); // assigned
98+
assertThat(information.isNew(entity)).isTrue();
99+
entity.setVersion(0);
100+
assertThat(information.isNew(entity)).isTrue();
101+
}
82102
}

spring-data-jpa/src/test/java/org/springframework/data/jpa/repository/support/HibernateJpaMetamodelEntityInformationIntegrationTests.java

+15
Original file line numberDiff line numberDiff line change
@@ -17,14 +17,19 @@
1717

1818
import org.junit.jupiter.api.Test;
1919
import org.junit.jupiter.api.extension.ExtendWith;
20+
import org.springframework.data.jpa.domain.sample.SampleWithPrimitiveVersion;
2021
import org.springframework.data.jpa.util.DisabledOnHibernate61;
22+
import org.springframework.data.repository.core.EntityInformation;
2123
import org.springframework.test.context.ContextConfiguration;
2224
import org.springframework.test.context.junit.jupiter.SpringExtension;
2325

26+
import static org.assertj.core.api.Assertions.assertThat;
27+
2428
/**
2529
* Hibernate execution for {@link JpaMetamodelEntityInformationIntegrationTests}.
2630
*
2731
* @author Greg Turnquist
32+
* @author Yanming Zhou
2833
*/
2934
@ExtendWith(SpringExtension.class)
3035
@ContextConfiguration("classpath:infrastructure.xml")
@@ -55,4 +60,14 @@ void prefersPrivateGetterOverFieldAccess() {
5560
void findsIdClassOnMappedSuperclass() {
5661
super.findsIdClassOnMappedSuperclass();
5762
}
63+
64+
@Test
65+
void negativeVersionedEntityIsNew() {
66+
EntityInformation<SampleWithPrimitiveVersion, Long> information = new JpaMetamodelEntityInformation<>(SampleWithPrimitiveVersion.class,
67+
em.getMetamodel(), em.getEntityManagerFactory().getPersistenceUnitUtil());
68+
69+
SampleWithPrimitiveVersion entity = new SampleWithPrimitiveVersion();
70+
entity.setId(23L); // assigned
71+
assertThat(information.isNew(entity)).isTrue();
72+
}
5873
}

spring-data-jpa/src/test/resources/META-INF/persistence.xml

+1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@
4646
<class>org.springframework.data.jpa.domain.sample.SampleEntityPK</class>
4747
<class>org.springframework.data.jpa.domain.sample.SampleWithIdClass</class>
4848
<class>org.springframework.data.jpa.domain.sample.SampleWithPrimitiveId</class>
49+
<class>org.springframework.data.jpa.domain.sample.SampleWithPrimitiveVersion</class>
4950
<class>org.springframework.data.jpa.domain.sample.SampleWithTimestampVersion</class>
5051
<class>org.springframework.data.jpa.domain.sample.Site</class>
5152
<class>org.springframework.data.jpa.domain.sample.SpecialUser</class>

0 commit comments

Comments
 (0)