c# - 使用 Moq 模拟插入查询到 MySQL 数据库

标签 c# mysql unit-testing mocking moq

我目前正在尝试使用 Moq 学习 Mocking,我想在我现有的数据库上尝试一下,但是我不确定如何才是正确的方法。

在我的数据层中,我有一个处理连接到数据库的类,并具有用于插入、选择等的各种方法。我想测试一个 Actor 是否被正确插入到数据库中。

我的 Insert 方法目前看起来像这样:

public void Insert(string firstname, string lastname)
{
    string query = $"INSERT INTO `sakila`.`actor`(`first_name`,`last_name`) VALUES('" + firstname + "','" + lastname + "')";
    Console.WriteLine(query);
    //open connection
    if (this.OpenConnection() == true)
    {
        Console.WriteLine("Established connection");
        //create command and assign the query and connection from the constructor
        MySqlCommand cmd = new MySqlCommand(query, connection);

        //Execute command
        cmd.ExecuteNonQuery();
        Console.WriteLine("Insert query succesfully executed.");

        //close connection
        this.CloseConnection();
    }
}

我将如何使用 Mocks 来做到这一点?我是否为 Actor 实体创建一个类?我应该为我的 DbConnection 类创建一个接口(interface)吗? 很抱歉提出所有问题,但我真的不知道如何解决这个问题。

最佳答案

目前,被测方法与实现问题的耦合过于紧密,以至于无法轻松地对其进行单独单元测试。尝试抽象出这些实现问题,以便可以轻松地模拟它们以进行隔离测试。

public interface IDbConnectionFactory {
    IDbConnection CreateConnection();
}

上述连接工厂抽象可用于访问您的 MySql 数据存储的其他必要的 System.Data 抽象。

public class MyDataAccessClass {
    private IDbConnectionFactory connectionFactory;

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

    public void Insert(string firstname, string lastname) {
        var query = $"INSERT INTO `sakila`.`actor`(`first_name`,`last_name`) VALUES('" + firstname + "','" + lastname + "')";
        Console.WriteLine(query);
        using(var connection = connectionFactory.CreateConnection() {
            //Creates and returns a MySqlCommand object associated with the MySqlConnection. 
            using(var command = connection.CreateCommand()) {
                command.CommandText = query;
                Console.WriteLine("Established connection");
                connection.Open();
                command.ExecuteNonQuery();
                Console.WriteLine("Insert query succesfully executed.");
                connection.Close();//is not actually necessary as the using statement will make sure to close the connection.
            }
        }
    }
}

工厂的生产实现将返回一个实际的 MySqlConnection

public class MySqlConnectionFactory: IDbConnectionFactory {
    public IDbConnection CreateConnection() {
        return new MySqlConnection("connection string");
    }
}

可以通过依赖注入(inject)传递到数据层

为了测试,您可以使用您选择的模拟框架来模拟接口(interface),或者创建您自己的模拟来注入(inject)和测试您的方法。

[TestClass]
public class DataAccessLayerUnitTest {
    [TestMethod]
    public void TestInsert() {
        //Arrange
        var commandMock = new Mock<IDbCommand>();
        commandMock
            .Setup(m => m.ExecuteNonQuery())
            .Verifiable();

        var connectionMock = new Mock<IDbConnection>();
        connectionMock
            .Setup(m => m.CreateCommand())
            .Returns(commandMock.Object);

        var connectionFactoryMock = new Mock<IDbConnectionFactory>();
        connectionFactoryMock
            .Setup(m => m.CreateConnection())
            .Returns(connectionMock.Object);

        var sut = new MyDataAccessClass(connectionFactoryMock.Object);
        var firstName = "John";
        var lastName = "Doe";

        //Act
        sut.Insert(firstName, lastName);

        //Assert
        commandMock.Verify();
    }
}

最后,建议您在命令文本中使用命令参数,因为手动构造查询字符串会使代码易受 SQL 注入(inject)攻击。

为了更好地了解如何使用 Moq,请查看他们的 Quickstart

关于c# - 使用 Moq 模拟插入查询到 MySQL 数据库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42795338/

相关文章:

c# - Azure Function 无法使用表绑定(bind)执行 : connection refused

mysql - 将 utf8_general_ci 表和字段转换为 utf8_unicode_ci

mysql - 带有子查询和 "HAVING"的极慢 SQL 查询

c# - 如何发现 DataColumn 的底层数据类型是 xml

c# - 创建配置或 INI 文件以存储用户设置首选项

php - PHP 循环数组逻辑错误

ios - 使用 xcodebuild 或 Xcode 运行选定的测试类

java - 如何在 Play Framework 中声明第二个数据库进行测试?

unit-testing - 使用 ODataQueryOptions 编写单元测试

c# - 当 .Net Framework 从 3.5 更改为 4.5 时,WPF Interop Control 呈现黑色