veilidchat/lib/chat/cubits/reconciliation/author_input_queue.dart

210 lines
6.7 KiB
Dart
Raw Normal View History

2024-05-30 23:25:47 -04:00
import 'dart:async';
import 'package:veilid_support/veilid_support.dart';
import '../../../proto/proto.dart' as proto;
import 'author_input_source.dart';
2024-05-31 18:27:50 -04:00
import 'message_integrity.dart';
2024-05-30 23:25:47 -04:00
import 'output_position.dart';
class AuthorInputQueue {
2024-05-31 18:27:50 -04:00
AuthorInputQueue._({
required TypedKey author,
required AuthorInputSource inputSource,
required OutputPosition? outputPosition,
required void Function(Object, StackTrace?) onError,
required MessageIntegrity messageIntegrity,
}) : _author = author,
_onError = onError,
_inputSource = inputSource,
_outputPosition = outputPosition,
2024-06-02 11:04:19 -04:00
_lastMessage = outputPosition?.message.content,
2024-05-31 18:27:50 -04:00
_messageIntegrity = messageIntegrity,
_currentPosition = inputSource.currentWindow.last;
static Future<AuthorInputQueue?> create({
required TypedKey author,
required AuthorInputSource inputSource,
required OutputPosition? outputPosition,
required void Function(Object, StackTrace?) onError,
}) async {
final queue = AuthorInputQueue._(
author: author,
inputSource: inputSource,
outputPosition: outputPosition,
onError: onError,
messageIntegrity: await MessageIntegrity.create(author: author));
if (!await queue._findStartOfWork()) {
return null;
}
return queue;
}
2024-05-30 23:25:47 -04:00
////////////////////////////////////////////////////////////////////////////
2024-05-31 18:27:50 -04:00
// Public interface
2024-05-30 23:25:47 -04:00
2024-06-02 11:04:19 -04:00
// Check if there are no messages left in this queue to reconcile
bool get isDone => _isDone;
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// Get the current message that needs reconciliation
proto.Message? get current => _currentMessage;
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// Get the earliest output position to start inserting
OutputPosition? get outputPosition => _outputPosition;
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// Get the author of this queue
TypedKey get author => _author;
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// Remove a reconciled message and move to the next message
// Returns true if there is more work to do
Future<bool> consume() async {
2024-06-02 11:04:19 -04:00
if (_isDone) {
return false;
}
2024-05-30 23:25:47 -04:00
while (true) {
2024-05-31 18:27:50 -04:00
_lastMessage = _currentMessage;
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
_currentPosition++;
// Get more window if we need to
if (!await _updateWindow()) {
// Window is not available so this queue can't work right now
2024-06-02 11:04:19 -04:00
_isDone = true;
2024-05-31 18:27:50 -04:00
return false;
}
final nextMessage = _inputSource.currentWindow
.elements[_currentPosition - _inputSource.currentWindow.first];
// Drop the 'offline' elements because we don't reconcile
// anything until it has been confirmed to be committed to the DHT
2024-06-02 11:04:19 -04:00
// if (nextMessage.isOffline) {
// continue;
// }
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
if (_lastMessage != null) {
// Ensure the timestamp is not moving backward
if (nextMessage.value.timestamp < _lastMessage!.timestamp) {
2024-05-30 23:25:47 -04:00
continue;
}
2024-05-31 18:27:50 -04:00
}
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// Verify the id chain for the message
final matchId = await _messageIntegrity.generateMessageId(_lastMessage);
if (matchId.compare(nextMessage.value.idBytes) != 0) {
continue;
2024-05-30 23:25:47 -04:00
}
2024-05-31 18:27:50 -04:00
// Verify the signature for the message
if (!await _messageIntegrity.verifyMessage(nextMessage.value)) {
continue;
2024-05-30 23:25:47 -04:00
}
2024-05-31 18:27:50 -04:00
_currentMessage = nextMessage.value;
break;
2024-05-30 23:25:47 -04:00
}
return true;
}
2024-05-31 18:27:50 -04:00
////////////////////////////////////////////////////////////////////////////
// Internal implementation
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// Walk backward from the tail of the input queue to find the first
// message newer than our last reconcicled message from this author
// Returns false if no work is needed
Future<bool> _findStartOfWork() async {
// Iterate windows over the inputSource
outer:
while (true) {
// Iterate through current window backward
2024-06-02 11:04:19 -04:00
for (var i = _inputSource.currentWindow.elements.length - 1;
2024-05-31 18:27:50 -04:00
i >= 0 && _currentPosition >= 0;
i--, _currentPosition--) {
final elem = _inputSource.currentWindow.elements[i];
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// If we've found an input element that is older than our last
// reconciled message for this author, then we stop
if (_lastMessage != null) {
if (elem.value.timestamp < _lastMessage!.timestamp) {
break outer;
}
}
2024-05-30 23:25:47 -04:00
}
2024-05-31 18:27:50 -04:00
// If we're at the beginning of the inputSource then we stop
if (_currentPosition < 0) {
break;
2024-05-30 23:25:47 -04:00
}
2024-05-31 18:27:50 -04:00
// Get more window if we need to
if (!await _updateWindow()) {
// Window is not available or things are empty so this
// queue can't work right now
2024-06-02 11:04:19 -04:00
_isDone = true;
2024-05-30 23:25:47 -04:00
return false;
}
2024-05-31 18:27:50 -04:00
}
2024-06-02 11:04:19 -04:00
// _currentPosition points to either before the input source starts
// or the position of the previous element. We still need to set the
// _currentMessage to the previous element so consume() can compare
// against it if we can.
if (_currentPosition >= 0) {
_currentMessage = _inputSource.currentWindow
.elements[_currentPosition - _inputSource.currentWindow.first].value;
}
// After this consume(), the currentPosition and _currentMessage should
// be equal to the first message to process and the current window to
// process should not be empty
return consume();
2024-05-31 18:27:50 -04:00
}
// Slide the window toward the current position and load the batch around it
Future<bool> _updateWindow() async {
// Check if we are still in the window
if (_currentPosition >= _inputSource.currentWindow.first &&
_currentPosition <= _inputSource.currentWindow.last) {
2024-05-30 23:25:47 -04:00
return true;
2024-05-31 18:27:50 -04:00
}
// Get another input batch futher back
final avOk =
await _inputSource.updateWindow(_currentPosition, _maxWindowLength);
final asErr = avOk.asError;
if (asErr != null) {
_onError(asErr.error, asErr.stackTrace);
return false;
}
final asLoading = avOk.asLoading;
if (asLoading != null) {
// xxx: no need to block the cubit here for this
// xxx: might want to switch to a 'busy' state though
// xxx: to let the messages view show a spinner at the bottom
// xxx: while we reconcile...
// emit(const AsyncValue.loading());
return false;
}
return avOk.asData!.value;
2024-05-30 23:25:47 -04:00
}
////////////////////////////////////////////////////////////////////////////
2024-05-31 18:27:50 -04:00
final TypedKey _author;
final AuthorInputSource _inputSource;
final OutputPosition? _outputPosition;
final void Function(Object, StackTrace?) _onError;
final MessageIntegrity _messageIntegrity;
2024-05-30 23:25:47 -04:00
2024-05-31 18:27:50 -04:00
// The last message we've consumed
proto.Message? _lastMessage;
2024-05-30 23:25:47 -04:00
// The current position in the input log that we are looking at
2024-05-31 18:27:50 -04:00
int _currentPosition;
// The current message we're looking at
proto.Message? _currentMessage;
2024-06-02 11:04:19 -04:00
// If we have reached the end
bool _isDone = false;
2024-05-30 23:25:47 -04:00
// Desired maximum window length
2024-05-31 18:27:50 -04:00
static const int _maxWindowLength = 256;
2024-05-30 23:25:47 -04:00
}