sql >> データベース >  >> RDS >> Mysql

Moqを使用してMySQLデータベースに挿入クエリをモックする

    現在、テスト中のメソッドは、実装上の懸念と密接に関連しているため、単独でユニットテストを簡単に行うことはできません。これらの実装の懸念事項を抽象化して、分離されたテストのために簡単にモックできるようにしてください。

    public interface IDbConnectionFactory {
        IDbConnection CreateConnection();
    }
    

    上記の接続ファクトリの抽象化を使用して、他の必要なSystem.Dataにアクセスできます。 MySqlデータストアの抽象化。

    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");
        }
    }
    

    これは、依存性注入を介してデータレイヤーに渡すことができます

    テストでは、選択したモックフレームワークを使用してインターフェースをモックするか、独自の偽物を作成してメソッドを注入してテストします。

    [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インジェクション攻撃に対してコードが開かれるためです。

    Moqの使用方法をよりよく理解するには、クイックスタート を確認してください。




    1. mySql REGEXPの論理AND演算子?

    2. MySQLのGroupByとの1対多結合は、1つの観測値のみを返します

    3. PostgreSQL9.1での自律型トランザクション

    4. Amazon RDS Aurora vs RDS MySQL vs MySQL on EC2?