Bug in NonBatchingBatch makes session ununsable after failed transaction

Description

In NonBatchingBatch.addToBatch() are some bugs (missing try...finally) that leaves the class in an undefined condition if the statement there fails. In addition, statements are not closed on error.

I have attached a project that shows the bug and contains a fix.

Here's my complete description (which you will also find in ShowBug.java):

/**

  • Description of the problem:

  • - we have a pre-bound session (as in OpenSessionInViewFilter) - there

  • are multiple transactions - one transaction fails - the transaction

  • manager disconnects the session (this is also true if the transaction

  • succeeds - so it should not be a problem) - according the docs

  • reconnecting should be done by transaction management - all following

  • transactions fail

  • In AbstractBatchImpl: try { doExecuteBatch(); } finally {

  • releaseStatements(); }

  • Statements are released within the same Transaction (ensured via

  • try...finally)

  • In NonBatchingBatch: try { [...] final int rowCount =

  • statement.executeUpdate(); [...] } catch ( SQLException e ) {

  • LOG.debug( "SQLException escaped proxy", e ); throw

  • sqlExceptionHelper().convert( e, "could not execute batch statement",

  • entry.getKey() ); } getStatements().clear();

  • Problems with this implementation: - Statements are not released here

  • - Statements are not closed if the query fails (resource leak!) -

  • statement.clearBatch() is never called (not sure if this matters. at

  • least in AbstractBatchImpl it is called) - Statements are not removed

  • from the list

  • In the next transaction (which succeeds), releaseStatements() is

  • called which fails because the proxy is not valid anymore.

  • Solution:

  • Fix the method NonBatchingBatch.addToBatch() as in attached NonBatchingBatch_fixed.java

  • Details:

  • - move all the close/release stuff outside of the loop

  • - put the loop in a "try" block

  • - in the "finally" block call super.releaseStatements()

  • - in AbstractBatchImpl make releaseStatements() protected instead of private

  • (this should really be done to avoid dupplicate code)

  • */

Attachments

1

Activity

Show:

Brett MeyerMarch 7, 2014 at 5:30 PM

Bulk closing rejected tickets in "resolved" state.

Steve EbersoleOctober 4, 2013 at 6:42 PM

Also, I just committed some work on NonBatchingBatch as part of HHH-7689. Plus as I said the JDBC proxying that is specifically mentioned here is no longer done. At this point I lean towards rejecting this as out of date.

I did look at your code. Most of the changes are functionally the same as the changes done under HHH-7689. One change that is not there, that I think is unnecessary is your call to Statement#clearBatch (you specifically mention that above too). This is specifically for non-batching batches, addBatch is never called on the underlying JDBC objects so calling clearBatch is at best no-op.

Like I said, going to close this. Please give the changes a try. If it still does not work as expected attach a test case that is simplified (aka, does not use Spring) and reproduces the condition you think is in error. Thanks.

Steve EbersoleOctober 3, 2013 at 11:02 PM

Even though the logs show "On release of batch it still contained JDBC statements", look at where/how that is used in the new code. It is used in a code block cleaning up the statements. Its just a warning to you that this is happening.

Really I just need a test case showing this happening, ideally in a supported means. Spring is breaking the explicit contract of Session here based on your explanation. If I can support them in doing how they try to do it, I will. But that means getting me a test case showing this behavior sans any Spring stuff; just a plain Hibernate usage

Steve EbersoleOctober 3, 2013 at 1:15 PM

This code significantly changed in 4.2 (we no longer proxy the JDBC objects ourselves). Do you see this behavior in 4.2 as well?

Bogdan FluerasOctober 3, 2013 at 12:04 PM

@Michale Wyraz: Your workaround of increasing batch size works for me too.
Hoping for this bug to be fixed though.

Out of Date

Details

Assignee

Reporter

Original estimate

Time tracking

No time logged2h remaining

Components

Affects versions

Priority

Created October 11, 2012 at 5:16 PM
Updated March 7, 2014 at 5:30 PM
Resolved October 4, 2013 at 6:42 PM