// A Conversation is a type of Chat that is 1:1 between two Contacts only // Each Contact in the ContactList has at most one Conversation between the // remote contact and the local account import 'dart:async'; import 'dart:convert'; import 'package:async_tools/async_tools.dart'; import 'package:equatable/equatable.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:meta/meta.dart'; import 'package:protobuf/protobuf.dart'; import 'package:veilid_support/veilid_support.dart'; import '../../account_manager/account_manager.dart'; import '../../proto/proto.dart' as proto; import '../../tools/tools.dart'; const _sfUpdateAccountChange = 'updateAccountChange'; @immutable class ConversationState extends Equatable { const ConversationState( {required this.localConversation, required this.remoteConversation}); final proto.Conversation? localConversation; final proto.Conversation? remoteConversation; @override List get props => [localConversation, remoteConversation]; } /// Represents the control channel between two contacts /// Used to pass profile, identity and status changes, and the messages key for /// 1-1 chats class ConversationCubit extends Cubit> { ConversationCubit( {required AccountInfo accountInfo, required TypedKey remoteIdentityPublicKey, TypedKey? localConversationRecordKey, TypedKey? remoteConversationRecordKey}) : _accountInfo = accountInfo, _localConversationRecordKey = localConversationRecordKey, _remoteIdentityPublicKey = remoteIdentityPublicKey, _remoteConversationRecordKey = remoteConversationRecordKey, super(const AsyncValue.loading()) { _identityWriter = _accountInfo.identityWriter; if (_localConversationRecordKey != null) { _initWait.add(() async { await _setLocalConversation(() async { // Open local record key if it is specified final pool = DHTRecordPool.instance; final crypto = await _cachedConversationCrypto(); final writer = _identityWriter; final record = await pool.openRecordWrite( _localConversationRecordKey!, writer, debugName: 'ConversationCubit::LocalConversation', parent: accountInfo.accountRecordKey, crypto: crypto); return record; }); }); } if (_remoteConversationRecordKey != null) { _initWait.add(() async { await _setRemoteConversation(() async { // Open remote record key if it is specified final pool = DHTRecordPool.instance; final crypto = await _cachedConversationCrypto(); final record = await pool.openRecordRead(_remoteConversationRecordKey, debugName: 'ConversationCubit::RemoteConversation', parent: pool.getParentRecordKey(_remoteConversationRecordKey) ?? accountInfo.accountRecordKey, crypto: crypto); return record; }); }); } } @override Future close() async { await _initWait(); await _accountSubscription?.cancel(); await _localSubscription?.cancel(); await _remoteSubscription?.cancel(); await _localConversationCubit?.close(); await _remoteConversationCubit?.close(); await super.close(); } //////////////////////////////////////////////////////////////////////////// // Public Interface /// Initialize a local conversation /// If we were the initiator of the conversation there may be an /// incomplete 'existingConversationRecord' that we need to fill /// in now that we have the remote identity key /// The ConversationCubit must not already have a local conversation /// The callback allows for more initialization to occur and for /// cleanup to delete records upon failure of the callback Future initLocalConversation( {required proto.Profile profile, required FutureOr Function(DHTRecord) callback, TypedKey? existingConversationRecordKey}) async { assert(_localConversationRecordKey == null, 'must not have a local conversation yet'); final pool = DHTRecordPool.instance; final crypto = await _cachedConversationCrypto(); final accountRecordKey = _accountInfo.accountRecordKey; final writer = _accountInfo.identityWriter; // Open with SMPL scheme for identity writer late final DHTRecord localConversationRecord; if (existingConversationRecordKey != null) { localConversationRecord = await pool.openRecordWrite( existingConversationRecordKey, writer, debugName: 'ConversationCubit::initLocalConversation::LocalConversation', parent: accountRecordKey, crypto: crypto); } else { localConversationRecord = await pool.createRecord( debugName: 'ConversationCubit::initLocalConversation::LocalConversation', parent: accountRecordKey, crypto: crypto, writer: writer, schema: DHTSchema.smpl( oCnt: 0, members: [DHTSchemaMember(mKey: writer.key, mCnt: 1)])); } final out = localConversationRecord // ignore: prefer_expression_function_bodies .deleteScope((localConversation) async { // Make messages log return _initLocalMessages( localConversationKey: localConversation.key, callback: (messages) async { // Create initial local conversation key contents final conversation = proto.Conversation() ..profile = profile ..superIdentityJson = jsonEncode(_accountInfo.localAccount.superIdentity.toJson()) ..messages = messages.recordKey.toProto(); // Write initial conversation to record final update = await localConversation.tryWriteProtobuf( proto.Conversation.fromBuffer, conversation); if (update != null) { throw Exception('Failed to write local conversation'); } final out = await callback(localConversation); // Upon success emit the local conversation record to the state _updateLocalConversationState(AsyncValue.data(conversation)); return out; }); }); // If success, save the new local conversation record key in this object _localConversationRecordKey = localConversationRecord.key; await _setLocalConversation(() async => localConversationRecord); return out; } /// Delete the conversation keys associated with this conversation Future delete() async { final pool = DHTRecordPool.instance; await _initWait(); final localConversationCubit = _localConversationCubit; final remoteConversationCubit = _remoteConversationCubit; final deleteSet = DelayedWaitSet(); if (localConversationCubit != null) { final data = localConversationCubit.state.asData; if (data == null) { log.warning('could not delete local conversation'); return false; } deleteSet.add(() async { _localConversationCubit = null; await localConversationCubit.close(); final conversation = data.value; final messagesKey = conversation.messages.toVeilid(); await pool.deleteRecord(messagesKey); await pool.deleteRecord(_localConversationRecordKey!); _localConversationRecordKey = null; }); } if (remoteConversationCubit != null) { final data = remoteConversationCubit.state.asData; if (data == null) { log.warning('could not delete remote conversation'); return false; } deleteSet.add(() async { _remoteConversationCubit = null; await remoteConversationCubit.close(); final conversation = data.value; final messagesKey = conversation.messages.toVeilid(); await pool.deleteRecord(messagesKey); await pool.deleteRecord(_remoteConversationRecordKey!); }); } // Commit the delete futures await deleteSet(); return true; } /// Force refresh of conversation keys Future refresh() async { await _initWait(); final lcc = _localConversationCubit; final rcc = _remoteConversationCubit; if (lcc != null) { await lcc.refreshDefault(); } if (rcc != null) { await rcc.refreshDefault(); } } /// Watch for account record changes and update the conversation void watchAccountChanges(Stream> accountStream, AsyncValue currentState) { assert(_accountSubscription == null, 'only watch account once'); _accountSubscription = accountStream.listen(_updateAccountChange); _updateAccountChange(currentState); } //////////////////////////////////////////////////////////////////////////// // Private Implementation void _updateAccountChange(AsyncValue avaccount) { final account = avaccount.asData?.value; if (account == null) { return; } final cubit = _localConversationCubit; if (cubit == null) { return; } serialFuture((this, _sfUpdateAccountChange), () async { await cubit.record.eventualUpdateProtobuf(proto.Conversation.fromBuffer, (old) async { if (old == null || old.profile == account.profile) { return null; } return old.deepCopy()..profile = account.profile; }); }); } void _updateLocalConversationState(AsyncValue avconv) { final newState = avconv.when( data: (conv) { _incrementalState = ConversationState( localConversation: conv, remoteConversation: _incrementalState.remoteConversation); // return loading still if state isn't complete if ((_localConversationRecordKey != null && _incrementalState.localConversation == null) || (_remoteConversationRecordKey != null && _incrementalState.remoteConversation == null)) { return const AsyncValue.loading(); } // state is complete, all required keys are open return AsyncValue.data(_incrementalState); }, loading: AsyncValue.loading, error: AsyncValue.error, ); emit(newState); } void _updateRemoteConversationState(AsyncValue avconv) { final newState = avconv.when( data: (conv) { _incrementalState = ConversationState( localConversation: _incrementalState.localConversation, remoteConversation: conv); // return loading still if state isn't complete if ((_localConversationRecordKey != null && _incrementalState.localConversation == null) || (_remoteConversationRecordKey != null && _incrementalState.remoteConversation == null)) { return const AsyncValue.loading(); } // state is complete, all required keys are open return AsyncValue.data(_incrementalState); }, loading: AsyncValue.loading, error: AsyncValue.error, ); emit(newState); } // Open local converation key Future _setLocalConversation(Future Function() open) async { assert(_localConversationCubit == null, 'shoud not set local conversation twice'); _localConversationCubit = DefaultDHTRecordCubit( open: open, decodeState: proto.Conversation.fromBuffer); _localSubscription = _localConversationCubit!.stream.listen(_updateLocalConversationState); } // Open remote converation key Future _setRemoteConversation(Future Function() open) async { assert(_remoteConversationCubit == null, 'shoud not set remote conversation twice'); _remoteConversationCubit = DefaultDHTRecordCubit( open: open, decodeState: proto.Conversation.fromBuffer); _remoteSubscription = _remoteConversationCubit!.stream.listen(_updateRemoteConversationState); } // Initialize local messages Future _initLocalMessages({ required TypedKey localConversationKey, required FutureOr Function(DHTLog) callback, }) async { final crypto = await _cachedConversationCrypto(); final writer = _identityWriter; return (await DHTLog.create( debugName: 'ConversationCubit::initLocalMessages::LocalMessages', parent: localConversationKey, crypto: crypto, writer: writer)) .deleteScope((messages) async => await callback(messages)); } Future _cachedConversationCrypto() async { var conversationCrypto = _conversationCrypto; if (conversationCrypto != null) { return conversationCrypto; } conversationCrypto = await _accountInfo.makeConversationCrypto(_remoteIdentityPublicKey); _conversationCrypto = conversationCrypto; return conversationCrypto; } //////////////////////////////////////////////////////////////////////////// // Fields TypedKey get remoteIdentityPublicKey => _remoteIdentityPublicKey; final AccountInfo _accountInfo; late final KeyPair _identityWriter; final TypedKey _remoteIdentityPublicKey; TypedKey? _localConversationRecordKey; final TypedKey? _remoteConversationRecordKey; DefaultDHTRecordCubit? _localConversationCubit; DefaultDHTRecordCubit? _remoteConversationCubit; StreamSubscription>? _localSubscription; StreamSubscription>? _remoteSubscription; StreamSubscription>? _accountSubscription; ConversationState _incrementalState = const ConversationState( localConversation: null, remoteConversation: null); VeilidCrypto? _conversationCrypto; final WaitSet _initWait = WaitSet(); }