Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Closing prepared statement after error in nested transaction prevents transaction rollback in v0.1.6 #49

Closed
ahobson opened this issue Mar 13, 2023 · 9 comments
Assignees

Comments

@ahobson
Copy link

ahobson commented Mar 13, 2023

First of all, thank you for go-txdb! It's been a huge help for us testing our project.

v0.1.6 introduced some new behavior that I believe to be a bug. The following test passes on v0.1.5 but not v0.1.6. When using postgresql, it fails. It seems closing a prepared statement automatically rolls back the nested transaction?

I took a look at the changes in v0.1.6 and wasn't able to untangle the behavior, but I was hoping this test case might help someone else figure it out.

Please let me know if I can provide any more information.

func TestShouldAllowClosingPreparedStatementAndRollback(t *testing.T) {
	for _, driver := range drivers() {
		db, err := sql.Open(driver, "rollback")
		if err != nil {
			t.Fatalf(driver+": failed to open a connection, have you run 'make test'? err: %s", err)
		}
		defer db.Close()

		// do a query prior to starting a nested transaction to
		// reproduce the error
		var count int
		err = db.QueryRow("SELECT COUNT(id) FROM users").Scan(&count)
		if err != nil {
			t.Fatalf(driver+": prepared statement count err %v", err)
		}
		if count != 3 {
			t.Logf("Count not 3: %d", count)
			t.FailNow()
		}

		// start a nested transaction
		tx, err := db.Begin()
		if err != nil {
			t.Fatalf(driver+": failed to start transaction: %s", err)
		}
		// need a prepared statement to reproduce the error
		insertSQL := "INSERT INTO users (username, email) VALUES(?, ?)"
		if strings.Index(driver, "psql_") == 0 {
			insertSQL = "INSERT INTO users (username, email) VALUES($1, $2)"
		}
		stmt, err := tx.Prepare(insertSQL)
		if err != nil {
			t.Fatalf(driver+": failed to prepare named statement: %s", err)
		}

		// try to insert already existing username/email
		_, err = stmt.Exec("gopher", "gopher@go.com")
		if err == nil {
			t.Fatalf(driver + ": double insert?")
		}
		// The insert failed, so we need to close the prepared statement
		err = stmt.Close()
		if err != nil {
			t.Fatalf(driver+": error closing prepared statement: %s", err)
		}
		// rollback the transaction now that it has failed
		err = tx.Rollback()
		if err != nil {
			t.Logf(driver+": failed rollback of failed transaction: %s", err)
			t.FailNow()
		}
	}
}
@arthurspa
Copy link

arthurspa commented May 5, 2023

The problem seems to be that when we open a Tx, go will reuse an existing connection (https://github.com/golang/go/blob/release-branch.go1.20/src/database/sql/sql.go#L1302) and never release it, since there was no commit nor rollback in this scenario.

Then, when we call Close() method, db.freeConn will be empty here (https://github.com/golang/go/blob/release-branch.go1.20/src/database/sql/sql.go#L887) and it will not call dc.closeDBLocked()(https://github.com/golang/go/blob/release-branch.go1.20/src/database/sql/sql.go#L888). Therefore go-txdb Close() method will never be called.

I'm not sure how to solve this problem though.

Another way to reproduce it:

  1. open a DB connection
  2. interact with any table, insert or select, etc
  3. open a transaction
  4. close the connection

After step 4, the transaction will still be alive.

@mdestagnol
Copy link

Same issue here with v0.1.6, we have to rollback to v0.1.5

@l3pp4rd
Copy link
Member

l3pp4rd commented May 18, 2023

Hi, not using go for over few years now, would be really helpful if someone could resolve the issue and raise a pull request with a test to prevent regression.

@stefafafan stefafafan self-assigned this Jun 7, 2023
@Yiling-J
Copy link
Collaborator

@stefafafan Any luck on this issue? If you are too busy I can also take a look

@stefafafan
Copy link
Collaborator

@Yiling-J I took a look at this just a little, but got a bit busy with other stuff. It will help if you would be able to take a look at it too, thanks!

@Yiling-J
Copy link
Collaborator

Yiling-J commented Aug 1, 2023

@ahobson I've merge PR to the main branch that I think might fix the bug. Could you please run your tests against it and let me know if you still see the issue?

@ahobson
Copy link
Author

ahobson commented Aug 1, 2023

@Yiling-J Thank you! I tried it with our application and the tests that used to fail with v0.1.6 pass with v0.1.7-0.20230731141835-240c3c2c17cd. This is great!

@Yiling-J Yiling-J closed this as completed Aug 2, 2023
@ahobson
Copy link
Author

ahobson commented Aug 7, 2023

@Yiling-J If you don't mind, let me know when you get a chance to make a release with this fix. Thanks again!

@Yiling-J
Copy link
Collaborator

Yiling-J commented Aug 8, 2023

@ahobson v0.1.7 released

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

6 participants