Delen via


Storingen simuleren tijdens servicewerkbelastingen

Met de testbaarheidsscenario's in Azure Service Fabric kunnen ontwikkelaars zich geen zorgen maken over het omgaan met afzonderlijke fouten. Er zijn echter scenario's waarin mogelijk een expliciete interleaving van clientworkloads en fouten nodig is. De interleaving van clientworkloads en fouten zorgt ervoor dat de service daadwerkelijk een actie uitvoert wanneer er een fout optreedt. Gezien het controleniveau dat testbaarheid biedt, kunnen deze zich op precieze punten van de uitvoering van de workload bevinden. Deze inductie van fouten in verschillende statussen in de toepassing kan fouten vinden en de kwaliteit verbeteren.

Voorbeeld van aangepast scenario

Deze test toont een scenario waarin de bedrijfsworkload wordt verbonden met foutieve en ondankbaar fouten. De fouten moeten worden veroorzaakt in het midden van servicebewerkingen of berekeningen voor de beste resultaten.

Laten we een voorbeeld bekijken van een service die vier werkbelastingen beschikbaar maakt: A, B, C en D. Elk komt overeen met een set werkstromen en kan rekenkracht, opslag of een combinatie zijn. Omwille van de eenvoud zullen we de werkbelastingen in ons voorbeeld abstraheren. De verschillende fouten die in dit voorbeeld worden uitgevoerd, zijn:

  • RestartNode: Ungraceful fout om het opnieuw opstarten van een machine te simuleren.
  • RestartDeployedCodePackage: Ungraceful fault to simulate service host process crashes.
  • RemoveReplica: Probleemloos bij het simuleren van het verwijderen van replica's.
  • MovePrimary: Probleemloze fout bij het simuleren van replica-verplaatsingen die worden geactiveerd door de Service Fabric-load balancer.
// Add a reference to System.Fabric.Testability.dll and System.Fabric.dll.

using System;
using System.Fabric;
using System.Fabric.Testability.Scenario;
using System.Threading;
using System.Threading.Tasks;

class Test
{
    public static int Main(string[] args)
    {
        // Replace these strings with the actual version for your cluster and application.
        string clusterConnection = "localhost:19000";
        Uri applicationName = new Uri("fabric:/samples/PersistentToDoListApp");
        Uri serviceName = new Uri("fabric:/samples/PersistentToDoListApp/PersistentToDoListService");

        Console.WriteLine("Starting Workload Test...");
        try
        {
            RunTestAsync(clusterConnection, applicationName, serviceName).Wait();
        }
        catch (AggregateException ae)
        {
            Console.WriteLine("Workload Test failed: ");
            foreach (Exception ex in ae.InnerExceptions)
            {
                if (ex is FabricException)
                {
                    Console.WriteLine("HResult: {0} Message: {1}", ex.HResult, ex.Message);
                }
            }
            return -1;
        }

        Console.WriteLine("Workload Test completed successfully.");
        return 0;
    }

    public enum ServiceWorkloads
    {
        A,
        B,
        C,
        D
    }

    public enum ServiceFabricFaults
    {
        RestartNode,
        RestartCodePackage,
        RemoveReplica,
        MovePrimary,
    }

    public static async Task RunTestAsync(string clusterConnection, Uri applicationName, Uri serviceName)
    {
        // Create FabricClient with connection and security information here.
        FabricClient fabricClient = new FabricClient(clusterConnection);
        // Maximum time to wait for a service to stabilize.
        TimeSpan maxServiceStabilizationTime = TimeSpan.FromSeconds(120);

        // How many loops of faults you want to execute.
        uint testLoopCount = 20;
        Random random = new Random();

        for (var i = 0; i < testLoopCount; ++i)
        {
            var workload = SelectRandomValue<ServiceWorkloads>(random);
            // Start the workload.
            var workloadTask = RunWorkloadAsync(workload);

            // While the task is running, induce faults into the service. They can be ungraceful faults like
            // RestartNode and RestartDeployedCodePackage or graceful faults like RemoveReplica or MovePrimary.
            var fault = SelectRandomValue<ServiceFabricFaults>(random);

            // Create a replica selector, which will select a primary replica from the given service to test.
            var replicaSelector = ReplicaSelector.PrimaryOf(PartitionSelector.RandomOf(serviceName));
            // Run the selected random fault.
            await RunFaultAsync(applicationName, fault, replicaSelector, fabricClient);
            // Validate the health and stability of the service.
            await fabricClient.TestManager.ValidateServiceAsync(serviceName, maxServiceStabilizationTime);

            // Wait for the workload to finish successfully.
            await workloadTask;
        }
    }

    private static async Task RunFaultAsync(Uri applicationName, ServiceFabricFaults fault, ReplicaSelector selector, FabricClient client)
    {
        switch (fault)
        {
            case ServiceFabricFaults.RestartNode:
                await client.FaultManager.RestartNodeAsync(selector, CompletionMode.Verify);
                break;
            case ServiceFabricFaults.RestartCodePackage:
                await client.FaultManager.RestartDeployedCodePackageAsync(applicationName, selector, CompletionMode.Verify);
                break;
            case ServiceFabricFaults.RemoveReplica:
                await client.FaultManager.RemoveReplicaAsync(selector, CompletionMode.Verify, false);
                break;
            case ServiceFabricFaults.MovePrimary:
                await client.FaultManager.MovePrimaryAsync(selector.PartitionSelector);
                break;
        }
    }

    private static Task RunWorkloadAsync(ServiceWorkloads workload)
    {
        throw new NotImplementedException();
        // This is where you trigger and complete your service workload.
        // Note that the faults induced while your service workload is running will
        // fault the primary service. Hence, you will need to reconnect to complete or check
        // the status of the workload.
    }

    private static T SelectRandomValue<T>(Random random)
    {
        Array values = Enum.GetValues(typeof(T));
        T workload = (T)values.GetValue(random.Next(values.Length));
        return workload;
    }
}