-
Notifications
You must be signed in to change notification settings - Fork 388
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
Changes not committed by SaveChanges for single-statement updates (probably EF7-related) #1740
Comments
The Pomelo provider currently does not issue any
@nightcoder62 To get to the bottom of this issue, let's start by checking first, that select @@global.autocommit, @@session.autocommit; For example: context.Database.OpenConnection();
var connection = context.Database.GetDbConnection();
using var command = connection.CreateCommand();
command.CommandText = "select @@global.autocommit, @@session.autocommit;";
using var dataReader = command.ExecuteReader();
if (dataReader.Read())
{
var globalAutoCommit = dataReader[0];
var sessionAutoCommit = dataReader[1];
Logger.LogInformation($"@@global.autocommit = {globalAutoCommit}, @@session.autocommit = {sessionAutoCommit}");
} Then check the log, whether it is the global or session |
@lauxjpn yes, I mentioned such suspicions (configuration or app issue - it's a big app) in one of the linked threads, so reproducing the problem in a cleanroom setting is next on my list. I'm a bit pressed for time though, but using your suggestion to check the current value of autocommit is quicker. I'll try to find some time to do that. Thanks! |
Spot on, @lauxjpn , ping @roji - your (and my) suspicion was true. Someone, somewhere, years ago, has turned autocommit off globally server-side. I may not ever find the reason, and I'm not the server admin so it'll take some time until I can get it fixed - but since the workaround (reverting EF7 to EF6 behaviour) makes the problem go away, we're fine. Most likely, we won't even change the setting - there are old legacy systems working with the database too, and it could be a workaround for something. I leave it for you to decide whether or not to close this issue, but might I suggest mentioning the workaround in a readme for the EF7 support, since everything works perfectly with global autocommit = 0 in EF6 but fails in EF7 (without the workaround)? Fabulous adventure ... 😄 Oh, and @roji, this (the workaround, mainly) should perhaps be mentioned and highlighted in the "what's new" docs for EF7 as well? I don't know if the cause for someone changing that setting here is a common thing, but I'm seeing old articles on SO about how to do it, so we may not be alone. It's pretty hairy, as it could result in data loss. |
@nightcoder62 Glad you found the issue! Using your The other way would be to use a connection interceptor (overriding We might explicitly set There are a couple of assumptions that Pomelo makes about the database server configuration for all features to work as expected, which we should document. |
This problem is specific to MySQL, since we disable implicit transactions for SQL Server and most other databases (PG, SQLite...) don't have such a mode. So I'm not sure it's appropriate to mention in the general EF what's new docs. |
@roji wrote:
I agree that this specific problem may be a bit too obscure for the docs, but a sentence that mentions the possibility to revert to the old transaction behaviour (using AutoTransactionBehaviour) would be prudent, I think. There could be similar interactions in other systems and situations. But that's just my 2 cents, I trust your team to make the right decision - you have the bigger picture. Remember, you gave me the solution before I found it myself. OK, I had only just started looking for it, but I suspect it could have taken me some time. Had it been mentioned in the what's new docs (which I had read carefully - more than once), we might have solved the issue even quicker. In any case, we're extremely grateful to the one of you guys who suggested that it should be possible to revert to the old behaviour in the first case - it saved our day. And the decision for now is to leave the setting as is, because we can't risk some legacy system breaking due to us changing it - that's life in the in-house trenches*. 😢 *) The solution the problem happened in is, in fact, a bunch of apps and services that are, one by one, replacing all those legacy systems, so we're on it. But that doesn't happen overnight. 😄 |
@lauxjpn I'd love that documentation. I have a pilot friend who sometimes let me ride shotgun in a Cessna or something, and in the club, there's a big poster on the door that says "Pilots". The poster reads: "The flight isn't over until the paperwork is done". The paperwork is dull, but yes, it has to be done. If you forget to close your flight plan (as in "Sweden Control, I'm home"), it can get expensive, depending on the amount of search and rescue alerted. I tend to think it's the same with code. Unless it's my code, of course ... |
For me adding that Still
Though it says What else should we do to solve this? I'm using When I just call However, in our architecture, sometimes we insert a record into a table using |
@Nefcanto a minimal code repro showing this behavior would, as always, be the best way forward. |
@roji, since creating a repro is time-consuming, can you please tell me when the auto-commit would do the job of committing? I mean, would it do it when the |
Auto-commit (the default) just means that when you send a SQL statement to the database without having started an (explicit) transaction beforehand, that statement is executed and committed immediately (as if it were wrapped inside an explicit transaction containing only itself). When auto-commit is off, that generally means that sending a SQL statement starts a new transaction, which you later much explicitly commit or rollback. I'm not aware of any timer-based mechanics around this in any database. |
@Nefcanto We will indeed need a repro for that. You can use the Also, please share with us the database version you are using (and any setup out of the ordinary). |
Well, after we prepared a MRE, we realized that it works just fine. As we investigated, it turned out to be a bug in our view that one colleague caused. The view did not return the record because of a wronly configured join. However, when we put "SaveChanges" and "Find" after each other and from two different instantiated contexts, it returns null. I'm still investigating this. |
so what is default behaviour of Pomelo provider when we call SaveChanges()? |
Summary
After upgrading a project to EF7 and Pomelo.EntityFrameworkCore.MySql 7.0.0-silver.1, certain changes to the database stopped working, without any errors. Everything seemed to work, but the changes were never committed to the database. As far as we can see, this affected only changes that generated single statements (as in deleting a record from a table).
The details haven't been fully investigated, but we finally found a workaround, and I'm creating this issue in case others encounter the same problem - there's also a chance that the cause is obvious to the maintainers when they see this. I'll try to do some further testing and research over the weekend.
Steps to reproduce
In a .NET 7.0 (Windows x64) project, we use these nuget packages (and whatever else is added automagically):
We connect to a database that runs MySQL version 5.6.21-log, using InnoDB. Yes, it's old but that's out of our control at the moment.
Configuration of our DbContext:
The issue
Single-statement database changes (SaveChanges) seem to work according to all logging, but the changes never get committed.
Running this on a freshly injected context results in one affected record, but it's still in the database (note that CurrentTransaction on context.Database is null, as expected, before the second line is executed):
No exceptions, no error messages. The log looks like this:
Workaround
Add this in the constructor for the context:
This makes the problem go away.
Suspicion
When AutoTransactionBehaviour is the default (AutoTransactionBehaviour.WhenNeeded), autocommit (MySql "SET autocommit=0") is set to 0 somewhere, which is wrong (it doesn't turn off implicit transactions, it just turns off committing them automatically). InnoDB starts implicit transactions ALWAYS, and it can't be turned off - a commit is ALWAYS required, disabling autocommit just stops it from happening automatically.
Further details
See my comments on #1620 - as you can see, I got some help from @roji on this.
This is the change in EF7 that this bug is most likely caused by:
https://learn.microsoft.com/en-us/ef/core/what-is-new/ef-core-7.0/whatsnew#unneeded-transactions-are-eliminated
The text was updated successfully, but these errors were encountered: