【问题标题】:how do I mock sqlconnection or should I refactor the code?我如何模拟 sqlconnection 还是应该重构代码?
【发布时间】:2018-01-24 21:39:36
【问题描述】:

我有下面的代码,我看过Moq and SqlConnection?How can I stub IDBconnection,但是我仍然不知道如何模拟下面的sqlconnection。

public class SqlBulkWriter : ISqlBulkWriter
{
    private readonly string _dbConnectionString;;

    public SqlBulkWriter(string dbConnectionString)
    {
        this._dbConnectionString = dbConnectionString;
    }

    public void EmptyTable(string schema, string tableName)
    {
        using (var connection = new SqlConnection(this._dbConnectionString))
        {
            try
            {
                connection.Open();
                using (var truncate = new SqlCommand($"TRUNCATE TABLE [{schema}].[{tableName}] ", connection))
                {
                    truncate.ExecuteNonQuery();
                }
            }
            catch (Exception ex)
            {
                throw new Exception(ex);
            }
            finally
            {
                connection.Close();
            }
        }
    }
}

我要对 EmptyTable 做单元测试,我想我应该先模拟 sqlconnection 吗?或者我该如何对 EmptyTempTable 进行单元测试?

谢谢!许多人欣赏!

【问题讨论】:

  • 这是可测试的吗?还是我应该重构代码?如果是这样,我该如何重构代码?谢谢!
  • 您将代码与实现问题紧密耦合。考虑使用抽象并将它们注入到类中。
  • 好的。我改变了很多

标签: unit-testing ado.net moq xunit sqlconnection


【解决方案1】:

SqlBulkWriter 类与实现问题紧密耦合,这使得单独测试该类变得困难。

知道连接字符串并不是该类真正关心的问题,可以委托给另一个服务。

有点像

public interface IDbConnectionFactory {
   IDbConnection CreateConnection();
}

你的类的实现看起来像这样

public class SqlConnectionFactory : IDbConnectionFactory {
    private readonly string dbConnectionString;

    public SqlConnectionFactory(string dbConnectionString) {
        this.dbConnectionString = dbConnectionString;
    }

    public IDbConnection CreateConnection() {
        return new SqlConnection(this.dbConnectionString);
    }
}

SqlBulkWriter 类现在可以重构为依赖于抽象而不是具体化。

public class SqlBulkWriter : ISqlBulkWriter {
    private readonly IDbConnectionFactory connectionFactory;

    public SqlBulkWriter(IDbConnectionFactory connectionFactory) {
        this.connectionFactory = connectionFactory;
    }

    public void EmptyTable(string schema, string tableName) {
        using (var connection = connectionFactory.CreateConnection()) {
            try {
                connection.Open();
                using (var command = connection.CreateCommand()) {
                    command.CommandText = $"TRUNCATE TABLE [{schema}].[{tableName}] ";
                    command.CommandType = CommandType.Text;
                    int rowsAffected = command.ExecuteNonQuery();
                }
            } catch (Exception ex) {
                throw ex;
            } finally {
                connection.Close();
            }
        }
    }
}

这使得类更容易被单独测试,因为这些抽象可以被模拟并注入到类中。

[TestClass]
public class SqlBulkWriter_Should {
    [TestMethod]
    public void EmptyTable() {
        //Arrange
        var mock = new MockRepository(MockBehavior.Default);
        var command = mock.OneOf<IDbCommand>();
        var connection = mock.OneOf<IDbConnection>(_ => _.CreateCommand() == command);
        var factory = mock.OneOf<IDbConnectionFactory>(_ => _.CreateConnection() == connection);

        var subject = new SqlBulkWriter(factory);
        var schema = "dbo";
        var tableName = "TestTable";

        //Act
        subject.EmptyTable(schema, tableName);

        //Assert
        Mock.Get(command).Verify(_ => _.ExecuteNonQuery(), Times.Once());
    }
}

参考Moq Quickstart 以更好地了解如何使用该框架。

【讨论】:

  • 非常感谢。你是我的英雄
猜你喜欢
  • 2012-05-11
  • 2021-12-06
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多