Issues
- Batched unidirectional one-to-many update failingHHH-19322
- Improve inline documentation for @SuppressWarnings in registerStoredProcedureParameter, getResultList, and unwrap methodsHHH-19321
- assigned id value is not passed into BeforeExecutionGenerator#generate() method when allowAssignedIdentifiers() is true and id has been assignedHHH-19320Nathan Xu
- StatelessSession.findMultiple() accepting a LockModeHHH-19319Resolved issue: HHH-19319Gavin King
- follow-on locking and StatelessSessionHHH-19318Resolved issue: HHH-19318Gavin King
- Mark org.hibernate.boot.models as incubatingHHH-19317
- subselect fetching defined in maping.xml not honoredHHH-19316
- SINGLE_TABLE inheritance makes treat() pick wrong columns when multiple subclasses have a field with the same nameHHH-19315
- StackOverflowException when using onConflict with createCriteriaInsertValues and createCriteriaInsertSelectHHH-19314Andrea Boriero
- EntityGraph and OneToOne relationship is loading twice as many queriesHHH-19313
- BytecodeProviderImpl throwing java.lang.IndexOutOfBoundsExceptionHHH-19312
- NPE with Query Cache, Left Join Fetch, Inheritance on Both Sides, and No AssociationHHH-19311
- Simplified declaration of type for basic mappings in XMLHHH-19310
- Switch to Central Publishing Portal APi for publishing to Maven CentralHHH-19309
- lock timeoutsHHH-19308
- NPE when entity class missing from persistence.xml is id of another entityHHH-19307Jan Schatteman
- Composite generator may not respect the event types of generators it consits ofHHH-19306Marko Bekhta
- NPE in EntityEntryContext nonEnhancedEntityXref.get( entity )HHH-19305
- NPE in ResultSetMappingProcessor when using createNativeQuery with {x.*} notation and Entity with Embeddable that contains relational mappingsHHH-19304
- validate @Id fields against @IdClass in ProcessorHHH-19303Resolved issue: HHH-19303Gavin King
- composite ids with no id classHHH-19302
- Must import FQCN when generating metamodel class for inner Jakarta Data repository interfaceHHH-19301Resolved issue: HHH-19301Čedomir Igaly
- more ConstraintKindsHHH-19300Resolved issue: HHH-19300Gavin King
- <element-collection/> with LIST classification interpreted as BAGHHH-19299
- add convenience overloads of StatelessSession.get() which default GraphSemantic.LOADHHH-19298Resolved issue: HHH-19298Gavin King
- Register json functions in SingleStore community dialectHHH-19297Oleksandr Yeliseiev
- overload createSelectionQuery() to accept an EntityGraph instead of a result classHHH-19296Resolved issue: HHH-19296Gavin King
- Foreign keys are dropped when using schema = "public" after migrating to Spring Boot 3HHH-19295
- NodeBuilder collection*() doesn't work with enum collectionsHHH-19294
- Criteria isMember() doesn't work with collections mapped as arrayHHH-19293Resolved issue: HHH-19293
- Significant Memory Increase After Upgrading from Spring Boot 3.3.5 to 3.4.4HHH-19292
- Expressions.nullExpresion() in querydsl result in NPE in SqmExpressible with named parametersHHH-19291Resolved issue: HHH-19291Andrea Boriero
- Ability to generate ProxyFactory, AccessOptimizer and InstantiationOptimizer at build timeHHH-19290
- Integrate release_announcement.md into website publishingHHH-19289
- Allow HIbernate Reactive to extend JdbcEnvironmentInitiatorHHH-19288Resolved issue: HHH-19288Davide D'Alto
- CURRENT DATE is not recognized as a keyword using DB2DialectHHH-19287Resolved issue: HHH-19287
- Ignoring auto-applied conversions on special mappingsHHH-19286
- @ManyToAny with @FilterJoinTable KOHHH-19285
- Extract Duplicated Vector Function Registration LogicHHH-19284donghyun Lee
- Hibernate Gradle Plugin configuration cache supportHHH-19283
- Hibernate Envers & Oracle: Automatic column name conversion to uppercaseHHH-19282
- Identify and handle tests which don't need to be run on all databasesHHH-19281
- ResourceRegistryStandardImpl#close(java.sql.Statement) is called on already closed statementsHHH-19280Andrea Boriero
- @Basic not implicit optionalHHH-19279Resolved issue: HHH-19279Gavin King
- fixes to logic in MultiIdEntityLoadersHHH-19278Resolved issue: HHH-19278Gavin King
- Native query EntityManager.createNativeQuery(…, EntityType.class) returns different result when run via EntityManager.createNativeQuery(…)HHH-19277Resolved issue: HHH-19277
- Native query with enum list param leads to OutOfMemoryHHH-19276
- Missing 6.6.11 release for hibernate-orm-toolsHHH-19275Resolved issue: HHH-19275Andrea Boriero
- Deprecate MetadataBuilder#applyIndexView and friendsHHH-19274Resolved issue: HHH-19274Steve Ebersole
- FetchNotFoundException after Upgrade to 6.6.XHHH-19273
50 of
Batched unidirectional one-to-many update failing
Details
Assignee
UnassignedUnassignedReporter
Konrad KueglerKonrad KueglerComponents
Priority
Major
Details
Details
Assignee
Unassigned
UnassignedReporter
Konrad Kuegler
Konrad KueglerComponents
Priority
Created 7 hours ago
Updated 7 hours ago
Activity
Show:
For a unidirectional one-to-many relation moving a "many" entity from a "one" entities' collection to another "one" can fail with a
StaleStateException
in batched mode if the SQL statements happen to be executed in the "wrong” order.Working example
Considering this mapping (simplified excerpt):
public static class Person { @JoinColumn(name = "person_id") @OneToMany private List<Phone> phones = new ArrayList<>(); public List<Phone> getPhones() { return phones; } }
Running this code to move
phone
fromperson1
'sphones
collection toperson2
usually works:Person person1 = session.find(Person.class, 1L); Person person2 = session.find(Person.class, 2L); Phone phone = session.find(Phone.class, 1L); person1.getPhones().remove(phone); person2.getPhones().add(phone);
It executes these SQL statements both with and without batching and everything works fine:
update Phone set person_id=null where person_id=? and id=?
with?
bound to 1 and 1update Phone set person_id=? where id=?
with?
bound to 2 and 1Breaking example
However, if we re-order loading the
Person
entitiesPerson person2 = session.find(Person.class, 2L); Person person1 = session.find(Person.class, 1L); // other statements as in previous code
The SQL statement order changes to:
update Phone set person_id=? where id=?
with?
bound to 2 and 1update Phone set person_id=null where person_id=? and id=?
with?
bound to 1 and 1This still works fine in non-batched mode. This mode ignores the number of actually affected statements, which is 0 for the second statement (as the old
person_id
column value has already been replaced by the first statement).In batched mode we get an exception for the second statement instead:
For the SQL statements to look like this (and to fail),
person1.getPhones()
needs to contain anotherPhone
that stays there. See the reproduce for the full code.Reproducer
https://github.com/hibernate/hibernate-orm/pull/9967
Analysis
One reason for this is that the two
Person
s'phones
collections are flushed independently of each other despite both referring to the same database row (for the onephone
). Depending on the execution order theupdate Phone set person_id=null ...
statement does not change anything,It seems changing the independent flushing approach is very hard considering the current implementation. To me it looks easier to make Hibernate ignore that in some cases statements have fewer affected rows than usual.
The flush order of the two collections is determined by the order the
Person
entities are loaded into the session (or by their ids when usinghibernate.order_updates
).The exception happens in Hibernate 6.6 (and at least 6.4), but no longer in 7.0/main. The behavior changed as a side-product of https://hibernate.atlassian.net/browse/HHH-18586 , which now swallows the
StaleStateException
in this case.I'm not sure this behavior change was intentional and can serve as the basis for a fix in 6.6. I will add a comment in that ticket to discuss this.
If it was not intentional, it can make sense to use
org.hibernate.jdbc.Expectation.None
to ignore the number of affected rows for specific statements.Non-batched mode is not affected as DeleteRowsCoordinatorStandard.deleteRows passes a
null
OperationResultChecker
to the actual SQL execution, ignoring the number of affected rows.As a workaround one can
flush()
between theremove()
and theadd()
call in the example to prevent SQL reordering. But that is of course bad for performance.