Freigeben über


Reliable Services – Benachrichtigungen

Mit Benachrichtigungen können Clients Änderungen an einem Objekt verfolgen, an dem sie interessiert sind. Zwei Arten von Objekten unterstützen Benachrichtigungen: Reliable State Manager und Reliable Dictionary.

Häufige Gründe für die Verwendung von Benachrichtigungen:

  • Erstellen von materialisierten Sichten, z.B. sekundäre Indizes oder aggregierte, gefilterte Sichten mit dem Replikatstatus. Ein Beispiel hierfür ist ein sortierter Index aller Schlüssel in Reliable Dictionary.
  • Senden von Überwachungsdaten, z.B. die Anzahl von in der letzten Stunde hinzugefügten Benutzern.

Benachrichtigungen werden während der Anwendung von Vorgängen ausgelöst. Auf einem primären Replikat werden Vorgänge nach der Bestätigung des Quorums im Rahmen von transaction.CommitAsync() oder this.StateManager.GetOrAddAsync() angewendet. Bei sekundären Replikaten werden Vorgänge bei der Verarbeitung von Replikationswarteschlangendaten angewendet. Aus diesem Grund sollten Benachrichtigungen so schnell wie möglich behandelt werden und synchrone Ereignisse keine aufwändigen Vorgänge enthalten. Andernfalls könnte sich dies negativ auf die Transaktionsverarbeitungszeit und den Replikataufbau auswirken.

Reliable State Manager-Benachrichtigungen

Der Reliable State Manager liefert Benachrichtigungen für folgende Ereignisse:

  • Transaktion
    • Commit
  • Zustands-Manager
    • Neu erstellen
    • Hinzufügen eines Reliable State
    • Entfernen eines Reliable State

Der Reliable State Manager verfolgt die aktuellen In-Flight-Transaktionen. Die einzige Änderung des Transaktionsstatus, bei der eine Benachrichtigung ausgelöst wird, ist der Commit einer Transaktion.

Der Reliable State Manager verwaltet eine Sammlung von Reliable States, z.B. Reliable Dictionary und Reliable Queue. Der Reliable State Manager löst Benachrichtigungen aus, wenn sich diese Auflistung wie folgt verändert: Ein Reliable State wird hinzugefügt oder entfernt, oder die gesamte Sammlung wird neu erstellt. Die Sammlung des Reliable State Manager wird in drei Fällen neu erstellt:

  • Zustandswiederherstellung: Wenn ein Replikat gestartet wird, stellt es den vorherigen Zustand vom Datenträger wieder her. Nach Abschluss der Wiederherstellung wird NotifyStateManagerChangedEventArgs zum Auslösen eines Ereignisses verwendet, das den Satz mit den wiederhergestellten Reliable States enthält.
  • Vollständige Kopie: Bevor ein Replikat dem Konfigurationssatz hinzugefügt werden kann, muss es erstellt werden. In einigen Fällen muss eine vollständige Kopie des Reliable State Manager-Zustands vom primären Replikat auf das inaktive sekundäre Replikat angewendet werden. Reliable State Manager auf dem sekundären Replikat nutzt NotifyStateManagerChangedEventArgs zum Auslösen eines Ereignisses, das den Satz mit den Reliable States enthält, die über das primäre Replikat beschafft wurden.
  • Wiederherstellung: Bei einer Notfallwiederherstellung kann der Replikatzustand mit RestoreAsyncaus einer Sicherung wiederhergestellt werden. In solchen Fällen nutzt Reliable State Manager auf dem primären Replikat NotifyStateManagerChangedEventArgs zum Auslösen eines Ereignisses, das den Satz mit den Reliable States enthält, die aus der Sicherung wiederhergestellt wurden.

Um sich für Transaktionsbenachrichtigungen und/oder Zustands-Manager-Benachrichtigungen zu registrieren, müssen Sie sich bei den Ereignissen TransactionChanged oder StateManagerChanged im Reliable State Manager registrieren. Üblicherweise wird die Registrierung bei diesen Ereignishandlern im Konstruktor Ihres zustandsbehafteten Diensts durchgeführt. Wenn Sie sich auf dem Konstruktor registrieren, entgeht Ihnen keine Benachrichtigung, die durch eine Änderung während der Lebensdauer von IReliableStateManager verursacht wird.

public MyService(StatefulServiceContext context)
    : base(MyService.EndpointName, context, CreateReliableStateManager(context))
{
    this.StateManager.TransactionChanged += this.OnTransactionChangedHandler;
    this.StateManager.StateManagerChanged += this.OnStateManagerChangedHandler;
}

Der TransactionChanged-Ereignishandler verwendet NotifyTransactionChangedEventArgs, um Details zum Ereignis bereitzustellen. Darin ist die Action-Eigenschaft enthalten (z.B. NotifyTransactionChangedAction.Commit), mit der die Art der Änderung angegeben wird. Außerdem ist die Transaction-Eigenschaft enthalten, mit der ein Verweis auf die geänderte Transaktion bereitgestellt wird.

Hinweis

TransactionChanged -Ereignisse werden derzeit nur ausgelöst, wenn für die Transaktion ein Commit durchgeführt wird. Die Aktion ist dann mit NotifyTransactionChangedAction.Commitidentisch. In Zukunft können Ereignisse aber auch für andere Arten von Änderungen des Transaktionsstatus ausgelöst werden. Wir empfehlen Ihnen, die Aktion zu überprüfen und das Ereignis nur zu verarbeiten, wenn es sich um das erwartete Ereignis handelt.

Es folgt ein Beispiel für den TransactionChanged -Ereignishandler.

private void OnTransactionChangedHandler(object sender, NotifyTransactionChangedEventArgs e)
{
    if (e.Action == NotifyTransactionChangedAction.Commit)
    {
        this.lastCommitLsn = e.Transaction.CommitSequenceNumber;
        this.lastTransactionId = e.Transaction.TransactionId;

        this.lastCommittedTransactionList.Add(e.Transaction.TransactionId);
    }
}

Der StateManagerChanged-Ereignishandler verwendet NotifyStateManagerChangedEventArgs, um Details zum Ereignis bereitzustellen. NotifyStateManagerChangedEventArgs verfügt über zwei Unterklassen: NotifyStateManagerRebuildEventArgs und NotifyStateManagerSingleEntityChangedEventArgs. Sie verwenden die Action-Eigenschaft in NotifyStateManagerChangedEventArgs, um NotifyStateManagerChangedEventArgs in die richtige Unterklasse umzuwandeln:

  • NotifyStateManagerChangedAction.Rebuild: NotifyStateManagerRebuildEventArgs
  • NotifyStateManagerChangedAction.Add und NotifyStateManagerChangedAction.Remove: NotifyStateManagerSingleEntityChangedEventArgs

Es folgt ein Beispiel für einen StateManagerChanged -Benachrichtigungshandler.

public void OnStateManagerChangedHandler(object sender, NotifyStateManagerChangedEventArgs e)
{
    if (e.Action == NotifyStateManagerChangedAction.Rebuild)
    {
        this.ProcessStateManagerRebuildNotification(e);

        return;
    }

    this.ProcessStateManagerSingleEntityNotification(e);
}

Reliable Dictionary-Benachrichtigungen

Reliable Dictionary liefert Benachrichtigungen für folgende Ereignisse:

  • Rebuild (Neu erstellen): Wird aufgerufen, wenn für ReliableDictionary ein lesbarer Zwischenzustand aus einem wiederhergestellten oder kopierten lokalen Status oder einer Sicherung wiederhergestellt wurde. Ein Datensatz von Transaktionen seit der Erstellung dieses Wiederherstellungszustands wird dann angewendet, bevor die Neuerstellung abgeschlossen ist. Die Anwendung dieses Datensatzes stellt Benachrichtigungen für Löschen, Hinzufügen, Aktualisieren und/oder Entfernen bereit.
  • Clear (Löschen): Wird aufgerufen, wenn der Status von ReliableDictionary über die ClearAsync-Methode gelöscht wurde.
  • Add (Hinzufügen): Wird aufgerufen, wenn ReliableDictionaryein Element hinzugefügt wurde.
  • Update (Aktualisieren): Wird aufgerufen, wenn ein Element in IReliableDictionary aktualisiert wurde.
  • Remove (Entfernen): Wird aufgerufen, wenn ein Element aus IReliableDictionary gelöscht wurde.

Um Reliable Dictionary-Benachrichtigungen zu erhalten, müssen Sie sich mit dem DictionaryChanged-Ereignishandler bei IReliableDictionary registrieren. Üblicherweise wird die Registrierung bei diesen Ereignishandlern in der ReliableStateManager.StateManagerChanged -Hinzufügungsbenachrichtigung durchgeführt. Durch die Registrierung beim Hinzufügen von IReliableDictionary zu IReliableStateManager wird sichergestellt, dass Ihnen keine Benachrichtigungen entgehen.

private void ProcessStateManagerSingleEntityNotification(NotifyStateManagerChangedEventArgs e)
{
    var operation = e as NotifyStateManagerSingleEntityChangedEventArgs;

    if (operation.Action == NotifyStateManagerChangedAction.Add)
    {
        if (operation.ReliableState is IReliableDictionary<TKey, TValue>)
        {
            var dictionary = (IReliableDictionary<TKey, TValue>)operation.ReliableState;
            dictionary.RebuildNotificationAsyncCallback = this.OnDictionaryRebuildNotificationHandlerAsync;
            dictionary.DictionaryChanged += this.OnDictionaryChangedHandler;
        }
    }
}

Hinweis

ProcessStateManagerSingleEntityNotification ist die Beispielmethode, die im vorherigen OnStateManagerChangedHandler-Beispiel aufgerufen wird.

Im vorherigen Code wird die IReliableNotificationAsyncCallback-Schnittstelle zusammen mit DictionaryChanged festgelegt. Da NotifyDictionaryRebuildEventArgs eine IAsyncEnumerable-Schnittstelle enthält, die asynchron aufgelistet werden muss, werden Neuerstellungsbenachrichtigungen nicht mit OnDictionaryChangedHandler ausgelöst, sondern mit RebuildNotificationAsyncCallback.

public async Task OnDictionaryRebuildNotificationHandlerAsync(
    IReliableDictionary<TKey, TValue> origin,
    NotifyDictionaryRebuildEventArgs<TKey, TValue> rebuildNotification)
{
    this.secondaryIndex.Clear();

    var enumerator = e.State.GetAsyncEnumerator();
    while (await enumerator.MoveNextAsync(CancellationToken.None))
    {
        this.secondaryIndex.Add(enumerator.Current.Key, enumerator.Current.Value);
    }
}

Hinweis

Als Teil der Verarbeitung der Neuerstellungsbenachrichtigung wird im obigen Code zuerst der beibehaltene aggregierte Zustand gelöscht. Alle vorherigen Benachrichtigungen sind irrelevant, da die Reliable Collection mit einem neuen Zustand wiederhergestellt wird.

Der DictionaryChanged-Ereignishandler verwendet NotifyDictionaryChangedEventArgs, um Details zum Ereignis bereitzustellen. NotifyDictionaryChangedEventArgs verfügt über fünf Unterklassen. Verwenden Sie die Action-Eigenschaft in NotifyDictionaryChangedEventArgs, um NotifyDictionaryChangedEventArgs in die richtige Unterklasse umzuwandeln:

  • NotifyDictionaryChangedAction.Rebuild: NotifyDictionaryRebuildEventArgs
  • NotifyDictionaryChangedAction.Clear: NotifyDictionaryClearEventArgs
  • NotifyDictionaryChangedAction.Add: NotifyDictionaryItemAddedEventArgs
  • NotifyDictionaryChangedAction.Update: NotifyDictionaryItemUpdatedEventArgs
  • NotifyDictionaryChangedAction.Remove: NotifyDictionaryItemRemovedEventArgs
public void OnDictionaryChangedHandler(object sender, NotifyDictionaryChangedEventArgs<TKey, TValue> e)
{
    switch (e.Action)
    {
        case NotifyDictionaryChangedAction.Clear:
            var clearEvent = e as NotifyDictionaryClearEventArgs<TKey, TValue>;
            this.ProcessClearNotification(clearEvent);
            return;

        case NotifyDictionaryChangedAction.Add:
            var addEvent = e as NotifyDictionaryItemAddedEventArgs<TKey, TValue>;
            this.ProcessAddNotification(addEvent);
            return;

        case NotifyDictionaryChangedAction.Update:
            var updateEvent = e as NotifyDictionaryItemUpdatedEventArgs<TKey, TValue>;
            this.ProcessUpdateNotification(updateEvent);
            return;

        case NotifyDictionaryChangedAction.Remove:
            var deleteEvent = e as NotifyDictionaryItemRemovedEventArgs<TKey, TValue>;
            this.ProcessRemoveNotification(deleteEvent);
            return;

        default:
            break;
    }
}

Empfehlungen

  • so schnell wie möglich durch.
  • Führen Sie keine aufwändigen Vorgänge (z.B. E/A-Vorgänge) im Rahmen von synchronen Ereignissen aus.
  • so schnell wie möglich der Bearbeitung des Ereignisses den Aktionstyp. Es kann sein, dass in Zukunft neue Aktionstypen hinzugefügt werden.

Hier folgen einige Punkte, die es zu beachten gilt:

  • Benachrichtigungen werden während der Ausführung eines Vorgangs ausgelöst. Als Schritt eines Wiederherstellungsvorgangs wird beispielsweise eine Wiederherstellungsbenachrichtigung ausgelöst. Eine Wiederherstellung wird erst fortgesetzt, nachdem das Benachrichtigungsereignis verarbeitet wurde.
  • Da Benachrichtigungen als Teil der Anwendung von Vorgängen ausgelöst werden, werden für Clients nur Benachrichtigungen für Vorgänge mit lokaler Commitausführung angezeigt. Vorgänge können in Zukunft eventuell rückgängig gemacht werden, da für sie nur ein lokaler Commit (d.h. Protokollierung) garantiert wird.
  • Im Wiederholungspfad wird eine einzelne Benachrichtigung für jeden angewendeten Vorgang ausgelöst. Dies bedeutet Folgendes: Wenn Transaktion T1 „Create(X)2“, „Delete(X)“ und „Create(X)“ enthält, erhalten Sie eine Benachrichtigung für die Erstellung von X, eine für den Löschvorgang und eine für die erneute Erstellung (in dieser Reihenfolge).
  • Für Transaktionen, die mehrere Vorgänge enthalten, werden die Vorgänge in der Reihenfolge angewendet, in der sie auf dem primären Replikat vom Benutzer empfangen wurden.
  • Im Rahmen der Verarbeitung falscher Fortschritte können einige Vorgänge auf sekundären Replikaten rückgängig gemacht werden. Für solche Vorgänge zum Rückgängigmachen, die den Zustand des Replikats auf einen stabilen Punkt zurücksetzen, werden Benachrichtigungen ausgelöst. Ein wichtiger Unterschied bei Benachrichtigungen zu Vorgängen zum Rückgängigmachen ist, dass Ereignisse mit doppelten Schlüsseln aggregiert werden. Wird beispielsweise die Transaktion T1 rückgängig gemacht, wird eine einzelne Löschbenachrichtigung „Delete(X)“ angezeigt.

Nächste Schritte

Bekannte Probleme

  • In bestimmten Situationen können einige Transaktionsbenachrichtigungen während einer Neuerstellung übersprungen werden. In diesem Fall ist der richtige Wert noch vorhanden und kann weiterhin gelesen oder iteriert werden. Nur die Benachrichtigung fehlt. Um den In-Memory-Zustand wiederherzustellen, verwendet „Zuverlässige Sammlung“ ein Write-Ahead-Protokoll, das in regelmäßigen Abständen in eine Prüfpunktdatei komprimiert wird. Während der Wiederherstellung wird zuerst der Basiszustand aus Prüfpunktdateien geladen. Dadurch wird eine Neuerstellungsbenachrichtigung ausgelöst. Anschließend werden die im Protokoll gespeicherten Transaktionen angewendet, die jeweils eine eigene Benachrichtigung zum Löschen, Hinzufügen, Aktualisieren oder Entfernen auslösen. Dieses Problem kann sich aus einer Racebedingung ergeben, bei der ein neuer Prüfpunkt schnell nach einer Wiederherstellung übernommen wird. Wenn der Prüfpunkt vor der Anwendung des Protokolls abgeschlossen wird, wird der In-Memory-Zustand auf den Zustand des neuen Prüfpunkts festgelegt. Dies führt zwar zum richtigen Zustand, bedeutet aber, dass Transaktionen im Protokoll, die noch nicht angewendet wurden, keine Benachrichtigungen senden.