Reliable Services 알림
알림을 사용하면 클라이언트에서 관심 있는 개체에 대한 변경 내용을 추적할 수 있습니다. 신뢰할 수 있는 상태 관리자 및 신뢰할 수 있는 사전의 두 가지 개체 유형에서 알림을 지원합니다.
알림을 사용하는 일반적인 이유:
- 보조 인덱스 또는 복제본 상태에 대한 집계된 필터링 보기와 같은 구체화된 보기를 구축하기 위해서입니다. 신뢰할 수 있는 사전에 있는 모든 키의 정렬된 인덱스가 그 예입니다.
- 지난 1시간 동안 추가된 사용자 수와 같은 모니터링 데이터를 전송하기 위해서입니다.
알림은 작업 적용의 일부로 실행됩니다. 주 복제본에서 작업은 transaction.CommitAsync()
또는 this.StateManager.GetOrAddAsync()
의 일부로 쿼럼 승인 후에 적용됩니다. 보조 복제본에서 작업은 복제 큐 데이터 처리 시 적용됩니다. 이 때문에 알림은 가능한 한 빨리 처리되어야 하며 동기 이벤트는 광범위한 작업을 포함하지 않아야 합니다. 그렇지 않으면 복제본 빌드업뿐만 아니라 트랜잭션 처리 시간에 부정적인 영향을 미칠 수 있습니다.
신뢰할 수 있는 상태 관리자 알림
신뢰할 수 있는 상태 관리자는 다음 이벤트에 대한 알림을 제공합니다.
- 트랜잭션
- Commit
- 상태 관리자
- 다시 빌드
- 신뢰할 수 있는 상태 추가
- 신뢰할 수 있는 상태 제거
신뢰할 수 있는 상태 관리자는 현재 처리 중인 트랜잭션을 추적합니다. 알림이 실행되도록 하는 유일한 트랜잭션 상태 변경은 커밋 중인 트랜잭션입니다.
신뢰할 수 있는 상태 관리자는 신뢰할 수 있는 사전 및 신뢰할 수 있는 큐와 같은 신뢰할 수 있는 상태 컬렉션을 유지 관리합니다. 신뢰할 수 있는 상태 관리자는 이 컬렉션이 변경될 때, 즉 신뢰할 수 있는 상태가 추가 또는 제거되거나 전체 컬렉션이 다시 작성될 때 알림을 실행합니다. 신뢰할 수 있는 상태 관리자 컬렉션은 다음 세 가지 경우에 다시 작성됩니다.
- 복구: 복제본이 시작되면 디스크에서 이전 상태를 복구합니다. 복구가 끝나면 NotifyStateManagerChangedEventArgs 를 사용하여 복구된 신뢰할 수 있는 상태 집합이 포함된 이벤트를 실행합니다.
- 전체 복사: 복제본을 구성 집합에 포함하려면 빌드해야 합니다. 경우에 따라 주 복제본의 신뢰할 수 있는 상태 관리자 상태의 전체 복사본을 유휴 보조 복제본에 적용해야 합니다. 보조 복제본의 신뢰할 수 있는 상태 관리자는 NotifyStateManagerChangedEventArgs 를 사용하여 주 복제본에서 획득한 신뢰할 수 있는 상태 집합이 포함된 이벤트를 실행합니다.
- 복원: 재해 복구 시나리오에서 RestoreAsync를 사용하여 백업에서 복제본의 상태를 복원할 수 있습니다. 이러한 경우 주 복제본의 신뢰할 수 있는 상태 관리자는 NotifyStateManagerChangedEventArgs 를 사용하여 백업에서 복원한 신뢰할 수 있는 상태 집합이 포함된 이벤트를 실행합니다.
트랜잭션 알림 및/또는 상태 관리자 알림을 등록하려면 신뢰할 수 있는 상태 관리자의 TransactionChanged 또는 StateManagerChanged 이벤트에 등록해야 합니다. 이러한 이벤트 처리기에 등록하는 일반적인 위치는 상태 저장 서비스의 생성자입니다. 생성자에 등록하면 IReliableStateManager의 수명 동안 변경 내용으로 발생되는 모든 알림을 놓치지 않게 됩니다.
public MyService(StatefulServiceContext context)
: base(MyService.EndpointName, context, CreateReliableStateManager(context))
{
this.StateManager.TransactionChanged += this.OnTransactionChangedHandler;
this.StateManager.StateManagerChanged += this.OnStateManagerChangedHandler;
}
TransactionChanged 이벤트 처리기는 NotifyTransactionChangedEventArgs를 사용하여 이벤트에 대한 세부 정보를 제공합니다. 변경 유형을 지정하는 작업 속성(예: NotifyTransactionChangedAction.Commit)을 포함합니다. 또한 변경된 트랜잭션에 대한 참조를 제공하는 트랜잭션 속성을 포함합니다.
참고 항목
오늘날 TransactionChanged 이벤트는 트랜잭션이 커밋되는 경우에만 발생합니다. 작업은 NotifyTransactionChangedAction.Commit와 동일합니다. 하지만 나중에 다른 유형의 트랜잭션 상태 변경에 대해 이벤트가 발생할 수 있습니다. 작업을 확인하고, 예상되는 작업인 경우에만 이벤트를 처리하는 것이 좋습니다.
다음은 예제 TransactionChanged 이벤트 처리기입니다.
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);
}
}
StateManagerChanged 이벤트 처리기는 NotifyStateManagerChangedEventArgs를 사용하여 이벤트에 대한 세부 정보를 제공합니다. NotifyStateManagerChangedEventArgs에는 두 가지 하위 클래스인 NotifyStateManagerRebuildEventArgs와 NotifyStateManagerSingleEntityChangedEventArgs가 있습니다. NotifyStateManagerChangedEventArgs의 작업 속성을 사용하여 NotifyStateManagerChangedEventArgs를 올바른 하위 클래스로 캐스트합니다.
- NotifyStateManagerChangedAction.Rebuild: NotifyStateManagerRebuildEventArgs
- NotifyStateManagerChangedAction.Add 및 NotifyStateManagerChangedAction.Remove: NotifyStateManagerSingleEntityChangedEventArgs
다음은 예제 StateManagerChanged 알림 처리기입니다.
public void OnStateManagerChangedHandler(object sender, NotifyStateManagerChangedEventArgs e)
{
if (e.Action == NotifyStateManagerChangedAction.Rebuild)
{
this.ProcessStateManagerRebuildNotification(e);
return;
}
this.ProcessStateManagerSingleEntityNotification(e);
}
신뢰할 수 있는 사전 알림
신뢰할 수 있는 사전은 다음 이벤트에 대한 알림을 제공합니다.
- 다시 빌드: ReliableDictionary가 복구 또는 복사된 로컬 상태 또는 백업에서 읽기 가능한 중계 복구 상태로 복구될 때 호출됩니다. 이 복구 상태를 만든 이후의 트랜잭션 레코드는 다시 빌드가 완료되기 전에 적용됩니다. 이러한 레코드를 적용하면 알림의 지우기, 추가, 업데이트 및/또는 제거를 수행할 수 있습니다.
- Clear: ClearAsync 메서드를 통해 ReliableDictionary의 상태를 지울 때 호출됩니다.
- Add: 항목이 ReliableDictionary에 추가될 때 호출됩니다.
- Update: IReliableDictionary 의 항목이 업데이트되었을 때 호출됩니다.
- Remove: IReliableDictionary 의 항목이 삭제되었을 때 호출됩니다.
신뢰할 수 있는 사전 알림을 가져오려면 IReliableDictionary에서 DictionaryChanaged 이벤트 처리기에 등록해야 합니다. 이러한 이벤트 처리기를 등록하는 일반적인 위치는 ReliableStateManager.StateManagerChanged 추가 알림입니다. IReliableDictionary가 IReliableStateManager에 추가되었을 때 등록하면 알림을 놓치지 않게 됩니다.
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;
}
}
}
참고 항목
ProcessStateManagerSingleEntityNotification은 위의 OnStateManagerChangedHandler 예제에서 호출하는 샘플 메서드입니다.
위의 코드는 DictionaryChanged와 함께 IReliableNotificationAsyncCallback 인터페이스도 설정합니다. NotifyDictionaryRebuildEventArgs에는 비동기식으로 열거되어야 하는 IAsyncEnumerable 인터페이스가 포함되어 있으므로 OnDictionaryChangedHandler 대신 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);
}
}
참고 항목
위의 코드에서 다시 작성 알림 처리의 일부로, 먼저 유지 관리되던 집계된 상태가 지워집니다. 신뢰할 수 있는 컬렉션은 새 상태로 다시 작성되므로 모든 이전 알림은 관련이 없어지기 때문입니다.
DictionaryChanged 이벤트 처리기는 NotifyDictionaryChangedEventArgs를 사용하여 이벤트에 대한 세부 정보를 제공합니다. NotifyDictionaryChangedEventArgs 에는 5개의 하위 클래스가 있습니다. NotifyDictionaryChangedEventArgs의 작업 속성을 사용하여 NotifyDictionaryChangedEventArgs를 올바른 하위 클래스로 캐스트합니다.
- 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;
}
}
권장 사항
- 하세요 .
- 마세요 .
- 하세요 . 새 작업 형식이 나중에 추가될 수 있습니다.
몇 가지 주의할 사항은 다음과 같습니다.
- 알림이 작업 실행의 일부로 실행됩니다. 예를 들어 복원 작업의 단계로 복원 알림이 실행됩니다. 알림 이벤트가 처리될 때까지는 복원이 계속되지 않습니다.
- 알림이 작업 적용의 일부로 실행되므로 클라이언트에는 로컬로 커밋된 작업에 대한 알림만 표시됩니다. 작업은 로컬로만 커밋(즉, 로깅)되도록 보장되므로 나중에 실행을 취소할 수도 있고 그렇지 않을 수도 있습니다.
- 다시 실행 경로에서는 적용된 각 작업에 대해 단일 알림이 실행됩니다. 즉, 트랜잭션 T1에 Create(X), Delete(X), Create(X)가 포함되는 경우 이 순서대로 X 생성에 대한 알림을 한 번 받고 삭제 알림을 받은 후 생성 알림을 한 번 더 받습니다.
- 여러 작업을 포함하는 트랜잭션의 경우 작업이 사용자로부터 주 복제본에 수신된 순서대로 적용됩니다.
- 거짓 진행률 처리의 일부로, 일부 작업이 보조 복제본에서 실행 취소될 수 있습니다. 이러한 실행 취소 작업에 대해 알림이 발생하고 복제본의 상태가 안정적인 지점으로 롤백됩니다. 실행 취소 알림의 중요한 차이점은 중복 키가 있는 이벤트가 집계된다는 것입니다. 예를 들어 트랜잭션 T1이 실행 취소되는 경우 Delete(X)에 대한 단일 알림이 표시됩니다.
다음 단계
알려진 문제
- 특정 상황에서는 다시 빌드하는 동안 일부 트랜잭션 알림을 건너뛸 수 있습니다. 이 경우 알림만 누락되었기 때문에 올바른 값은 여전히 존재하며 읽거나 반복할 수 있습니다. 메모리 내 상태를 복구하기 위해 신뢰할 수 있는 컬렉션은 검사점 파일로 주기적으로 압축되는 미리 쓰기 로그를 사용합니다. 복원하는 동안 먼저 기본 상태가 다시 빌드 알림을 트리거하는 검사점 파일에서 로드됩니다. 그런 다음 로그에 저장된 트랜잭션이 적용되고 각 트랜잭션이 자체 지우기, 추가, 업데이트 또는 제거 알림을 트리거합니다. 이 문제는 복원 후 새 검사점이 신속하게 수행되는 경합 상태에서 발생할 수 있습니다. 로그를 적용하기 전에 검사점이 완료되면 메모리 내 상태가 새 검사점의 상태로 설정됩니다. 이로 인해 올바른 상태가 되지만 아직 적용되지 않은 로그의 트랜잭션은 알림을 보내지 않게 됩니다.