简体   繁体   中英

unit Testing EF DbUpdateConcurrencyException handling

I need help testing the following code

public virtual void Update(T entity)
    {
        if (entity == null)
        {
            throw new ArgumentNullException("entity");
        }

        int iretries = 0;
        bool success = false;

        do
        {
            try
            {
                this.context.SaveChanges();
                success = true;
            }
            catch (DbUpdateConcurrencyException ex)
            {
                // Get the current entity values and the values in the database
                // as instances of the entity type
                var entry = ex.Entries.Single();
                var databaseValues = entry.GetDatabaseValues();

                // Choose an initial set of resolved values. In this case we
                // make the default be the values currently in the database: StoreWins
                object resolvedValues = ResolveConcurrency(databaseValues.ToObject());

                // Update the original values with the database values and
                // the current values with whatever the user choose.
                entry.OriginalValues.SetValues(databaseValues);
                entry.CurrentValues.SetValues(resolvedValues);

                // give up after n retries
                if (++iretries == NUMBER_OF_CONC_RETRIES)
                    throw;
            }
            catch (Exception)
            {
                //rethrow 
                throw;
            }
        } while (!success);
    }

I want to unit test the DbUpdateConcurrencyException branch.

So, one simple test scenario would be:

  • Creating a new DbUpdateConcurrencyException
  • Mock the SaveChanges to throw the above exception
  • Verify that SaveChanges was called a number of NUMBER_OF_CONC_RETRIES
  • Assert that the Update method re-throws the exception

In the current state, the above test scenario cannot be tested, I cannot mock the exception to contain a IEnumerable<DbEntityEntry> with a single DbEntityEntry ; I cannot mock the GetDatabaseValues() , etc.

A simple solution would be to insert a new layer of abstraction; let's say using an interface to abstract the entire code that currently sits in the catch block, and to provide a mock that does nothing.

But then I would end up in the situation when I would want to test the implementation of that interface, and would end up having the same questions as above. How can I mock the DbUpdateConcurrencyException , the GetDatabaseValues , etc.

I am using moq for mocking.

Thank you for your input

If you cannot mock something you must hide it behind something else you can mock or override in test. Your test actually doesn't need to use all that stuff for loading values and setting them in the entry - that is all dependent on EF and you will not test it when mocking the context because that would mean to re-implement EF's logic behind SaveChanges . All you need to do is:

catch (DbUpdateConcurrencyException ex) {
    RefreshValues(ex);

    // give up after n retries
    if (++iretries == NUMBER_OF_CONC_RETRIES)
        throw;
}

Where RefreshValues can be protected virtual method and you can override it in your test either by providing test version of the class (you can even achieve this with Moq) or by inheriting the test from this class and overriding the method directly in test class.

To setup Moq you need interface for your context exposing SaveChanges method:

var contextMock = new Mock<IContext>();
contextMock.Setup(m => m.SaveChanges())
           .Callback(m => throw new DbUpdateConcurrencyException());

If you need to test that it works for few throws as well you need to keep counter in the test and use it in the callback to decide if throw or not.

I have a pretty good solution to this, DbContext SaveChanges is an override method: so you can make a sub class of your normal DbConext like this:

public class TestDbContext : MyDbContext
{
    private Exception _exception;

    // your ctor of choice        

    public void OnSaveChangesThrow(Exception exception)
    {
        _exception = exception;
    }

    public override Task<int> SaveChangesAsync(CancellationToken cancellationToken = default)
    {
        if (_exception != null)
        {
            throw _exception;
        }
        return base.SaveChangesAsync(cancellationToken);
    }
}

In my unit test, I can create an InMemory DbContext, Seed the database with a few values and then call OnSaveChangesThrow to prepare for my exception.

[TestMethod]
public async Task MarkAsSubmitted_DbUpdateConcurrencyException()
{
    // Arrange
    var db = DbContextFactory.GetTestDbContext();
    await MyFixture.Seed(db);
    var submissionId = 1;
    var sut = new SubmissionCommands(db);
    db.OnSaveChangesThrow(new DbUpdateConcurrencyException());

    // Act
    var success = await sut.MarkAsSubmitted(submissionId);

    // Assert
    Assert.AreEqual(false, success);
}

It is not much extra code to cover this interesting test case. TestDbContext can be declared in the test project.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM