sql >> Databáze >  >> RDS >> Mysql

Zesměšňující vložení dotazu do databáze MySQL pomocí Moq

V současné době je testovaná metoda příliš úzce spojena s implementací, aby byla snadno jednotně testovatelná v izolaci. Zkuste tyto obavy týkající se implementace abstrahovat, aby je bylo možné snadno zesměšnit pro izolované testy.

public interface IDbConnectionFactory {
    IDbConnection CreateConnection();
}

Výše uvedenou tovární abstrakci připojení lze použít pro přístup k dalším nezbytným System.Data abstrakce vašeho úložiště dat 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.
            }
        }
    }
}

Produkční implementace továrny vrátí skutečné MySqlConnection

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

které lze předat do datové vrstvy prostřednictvím vkládání závislostí

Pro testování zesměšňujete rozhraní pomocí vašeho zesměšňujícího rámce, který si vyberete, nebo si vytvoříte vlastní padělky, které vložíte a otestujete svou metodu.

[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();
    }
}

Nakonec je vhodné použít parametry příkazu v textu příkazu, protože ruční vytvoření řetězce dotazu otevře kód před útoky SQL injection.

Chcete-li lépe pochopit, jak používat Moq, podívejte se na jejich Rychlý start




  1. PHP/Apache:PHP Závažná chyba:Volání nedefinované funkce mysql_connect()

  2. SQLSTATE[HY000] [2002] Žádný takový soubor nebo adresář

  3. connect ECONNREFUSED - node js , sql

  4. Dotaz k nalezení cizích klíčů