Basit Bir Mock Framework Geliştirelim 4

2023-05-11

Mocklama kütüphanemizi daha da geliştirelim. Şu anda, bir metodun belirli parametrelerle çağrılmasını ve belirli bir sonuç döndürmesini bekliyoruz. Ancak, bazen belirli bir parametre türüne sahip bir metodu çağrıldığında belirli bir sonuç döndürmesini isteyebiliriz, parametrenin kendisinin ne olduğu önemli olmayabilir. Bu özellik, genellikle mocklama kütüphanelerinde “It.IsAny” olarak adlandırılır.

Ayrıca, bir mock nesnesinin bir metodu birden fazla kez çağrıldığında farklı sonuçlar döndürmesini isteyebiliriz. Bu, genellikle “SetupSequence” olarak adlandırılır.

Bu iki özelliği ekleyelim.

using System;
using System.Collections.Concurrent;
using System.Linq.Expressions;
using System.Reflection;
using System.Collections.Generic;

public class Mock<T> : DispatchProxy
{
    private ConcurrentDictionary<string, object> _methodResponses = new();
    private ConcurrentDictionary<string, Queue<object>> _methodSequences = new();
    private ConcurrentDictionary<string, int> _methodCalls = new();

    protected override object Invoke(MethodInfo targetMethod, object[] args)
    {
        var key = GetKey(targetMethod, args);
        var wildcardKey = $"{targetMethod.Name}_*";

        if (_methodSequences.TryGetValue(key, out var sequence) && sequence.Count > 0)
        {
            _methodCalls[key]++;
            return sequence.Dequeue();
        }
        else if (_methodSequences.TryGetValue(wildcardKey, out sequence) && sequence.Count > 0)
        {
            _methodCalls[wildcardKey]++;
            return sequence.Dequeue();
        }
        else if (_methodResponses.TryGetValue(key, out var value))
        {
            _methodCalls[key]++;
            return value;
        }
        else if (_methodResponses.TryGetValue(wildcardKey, out value))
        {
            _methodCalls[wildcardKey]++;
            return value;
        }

        throw new Exception($"No response set up for method {targetMethod.Name}");
    }

    private string GetKey(MethodInfo methodInfo, object[] args)
    {
        return $"{methodInfo.Name}_{string.Join("_", args)}";
    }

    public void Setup(Expression<Action<T>> expression, object response)
    {
        SetupMethod(expression, response, _methodResponses);
    }

    public void SetupSequence(Expression<Action<T>> expression, params object[] responses)
    {
        var queue = new Queue<object>(responses);
        SetupMethod(expression, queue, _methodSequences);
    }

    private void SetupMethod(Expression<Action<T>> expression, object response, ConcurrentDictionary<string, object> dictionary)
    {
        var methodCall = expression.Body as MethodCallExpression;
        var methodName = methodCall.Method.Name;
        var arguments = new object[methodCall.Arguments.Count];
        for (int i = 0; i < methodCall.Arguments.Count; i++)
        {
            var arg = methodCall.Arguments[i];
            if (arg.NodeType == ExpressionType.Constant)
            {
                arguments[i] = ((ConstantExpression)arg).Value;
            }
            else if (arg.NodeType == ExpressionType.Call && ((MethodCallExpression)arg).Method.Name == "IsAny" && ((MethodCallExpression)arg).Object.Type.Name == "It")
            {
                arguments[i] = "*";
            }
        }
        var key = GetKey(methodCall.Method, arguments);
        dictionary.TryAdd(key, response);
    }

    public void Verify(Expression<Action<T>> expression, int expectedInvocationCount = 1)
    {
        var methodCall = expression.Body as MethodCallExpression;
        var methodName = methodCall.Method.Name;
        var arguments = new object[methodCall.Arguments.Count];
        for (int i = 0; i < methodCall.Arguments.Count; i++)
        {
            var arg = methodCall.Arguments[i];
            if (arg.NodeType == ExpressionType.Constant)
            {
                arguments[i] = ((ConstantExpression)arg).Value;
            }
        }
        var key = GetKey(methodCall.Method, arguments);
        if (_methodCalls.TryGetValue(key, out var actualInvocationCount))
        {
            if (actualInvocationCount != expectedInvocationCount)
            {
                throw new Exception($"Method {methodName} was expected to be called {expectedInvocationCount} times but was called {actualInvocationCount} times.");
            }
        }
        else
        {
            throw new Exception($"Method {methodName} was expected to be called {expectedInvocationCount} times but was not called.");
        }
    }

    public static T Create()
    {
        return Create<T, Mock<T>>();
    }
}

public static class It
{
    public static T IsAny<T>()
    {
        return default;
    }
}

Ve kullanalım

public interface ITestInterface
{
    string TestMethod(int parameter);
}

public class Test
{
    public void TestMethod()
    {
        var mock = Mock<ITestInterface>.Create();
        (mock as Mock<ITestInterface>).Setup(x => x.TestMethod(It.IsAny<int>()), "Test response for any int");

        var result1 = mock.TestMethod(5);
        var result2 = mock.TestMethod(10);

        Console.WriteLine(result1); // Outputs: Test response for any int
        Console.WriteLine(result2); // Outputs: Test response for any int

        (mock as Mock<ITestInterface>).Verify(x => x.TestMethod(5)); // Throws no exception
        (mock as Mock<ITestInterface>).Verify(x => x.TestMethod(10)); // Throws no exception

        (mock as Mock<ITestInterface>).SetupSequence(x => x.TestMethod(It.IsAny<int>()), "First call", "Second call");

        var result3 = mock.TestMethod(20);
        var result4 = mock.TestMethod(30);

        Console.WriteLine(result3); // Outputs: First call
        Console.WriteLine(result4); // Outputs: Second call
    }
}

Bu yeni versiyonda, It.IsAny<T> kullanarak belirli bir parametre türüne sahip bir metodu çağrıldığında belirli bir sonuç döndürebiliriz. Ayrıca, SetupSequence kullanarak bir metodu birden çok kez çağırdığımızda farklı sonuçlar döndürebiliriz. Bu, bir dizi çağrının farklı yanıtlar vermesini simüle etmek için kullanılabilir.



Bu gibi daha fazla gönderi...

SemaphoreSlim Sınıfı: C#'ta Çoklu Görevlere Dayalı Programlama

2023-06-10 | #net #semaphoreslim

Genel Bakış SemaphoreSlim sınıfı, bir veya daha fazla threadin aynı anda belirli bir kaynağı veya işlemi kullanmasını kontrol etmek için C# ‘ta kullanılan bir yapıdır. SemaphoreSlim, aynı anda kaynağa erişebilecek thread sayısını sınırlar. SemaphoreSlim kullanımı, genellikle çok threadli uygulamalarda deadlock durumlarını önlemek ve belirli bir kaynağın aynı anda yalnızca bir veya daha fazla thread tarafından kullanılmasını sağlamak için kullanılır. SemaphoreSlim Kullanımı SemaphoreSlim sınıfı, aşağıdaki gibi kullanılır: SemaphoreSlim semaphoreSlim = new SemaphoreSlim(1, 1); Burada ilk parametre, aynı anda kaç threadin kaynağı kullanabileceğini belirler.

Devamı 


Basit Bir Rest Api Client Geliştirelim 9

2023-05-14 | #diy-rest-api-client #do-it-yourself #net #rest-api

HTTP isteklerini izlemek ve teşhis etmek için daha fazla yetenek ekleyerek, kütüphanemizi daha da geliştirebiliriz. Bu amaçla, isteklerin ve yanıtların bazı temel bilgilerini loga kaydeden bir middleware ekleyelim. Bu özellik, HTTP isteklerinin ve yanıtlarının kaydını tutmak için HttpClient‘in DelegatingHandler sınıfını kullanır. Bu, bir HTTP isteği gönderildiğinde ve bir yanıt alındığında çalışan bir kod parçasıdır. Bu durumda, HTTP isteklerinin ve yanıtlarının bazı temel bilgilerini loga kaydeder. Aşağıda, bu özelliği eklemek için gereken kod parçacığı bulunmaktadır:

Devamı 


Basit Bir Rest Api Client Geliştirelim 8

2023-05-12 | #diy-rest-api-client #do-it-yourself #net #rest-api

Bu noktada, geliştirmemiz gereken bir diğer önemli konu da güvenlikle ilgili olabilir. Özellikle, API çağrıları genellikle belirli bir kimlik doğrulama yöntemi gerektirir. Bunu sağlamak için, HTTP istemcimizi çok kullanılan iki kimlik doğrulama yöntemiyle, yani Basic Authentication ve Bearer (Token) Authentication ile uyumlu hale getirebiliriz. Bu amaçla, istemcimize iki yeni metot ekleyelim: public void SetBasicAuthentication(string username, string password) { var basicAuthValue = Convert.ToBase64String(Encoding.UTF8.GetBytes($"{username}:{password}")); SetDefaultHeader("Authorization", $"Basic {basicAuthValue}"); } public void SetBearerToken(string token) { SetDefaultHeader("Authorization", $"Bearer {token}"); } Bu metodlar, Authorization başlığını ayarlar, böylece her HTTP talebi kimlik doğrulama bilgilerini içerir.

Devamı 