mirror of
https://gitlab.com/veilid/veilidchat.git
synced 2025-01-11 15:49:29 -05:00
more refactor and dhtrecord multiple-open support
This commit is contained in:
parent
c4c7b264aa
commit
e262b0f777
@ -147,7 +147,7 @@ class ContactInvitationListCubit
|
|||||||
|
|
||||||
Future<void> deleteInvitation(
|
Future<void> deleteInvitation(
|
||||||
{required bool accepted,
|
{required bool accepted,
|
||||||
required proto.ContactInvitationRecord contactInvitationRecord}) async {
|
required TypedKey contactRequestInboxRecordKey}) async {
|
||||||
final pool = DHTRecordPool.instance;
|
final pool = DHTRecordPool.instance;
|
||||||
final accountRecordKey =
|
final accountRecordKey =
|
||||||
_activeAccountInfo.userLogin.accountRecordInfo.accountRecord.recordKey;
|
_activeAccountInfo.userLogin.accountRecordInfo.accountRecord.recordKey;
|
||||||
@ -159,14 +159,11 @@ class ContactInvitationListCubit
|
|||||||
if (item == null) {
|
if (item == null) {
|
||||||
throw Exception('Failed to get contact invitation record');
|
throw Exception('Failed to get contact invitation record');
|
||||||
}
|
}
|
||||||
if (item.contactRequestInbox.recordKey ==
|
if (item.contactRequestInbox.recordKey.toVeilid() ==
|
||||||
contactInvitationRecord.contactRequestInbox.recordKey) {
|
contactRequestInboxRecordKey) {
|
||||||
await shortArray.tryRemoveItem(i);
|
await shortArray.tryRemoveItem(i);
|
||||||
break;
|
|
||||||
}
|
await (await pool.openOwned(item.contactRequestInbox.toVeilid(),
|
||||||
}
|
|
||||||
await (await pool.openOwned(
|
|
||||||
contactInvitationRecord.contactRequestInbox.toVeilid(),
|
|
||||||
parent: accountRecordKey))
|
parent: accountRecordKey))
|
||||||
.scope((contactRequestInbox) async {
|
.scope((contactRequestInbox) async {
|
||||||
// Wipe out old invitation so it shows up as invalid
|
// Wipe out old invitation so it shows up as invalid
|
||||||
@ -174,11 +171,13 @@ class ContactInvitationListCubit
|
|||||||
await contactRequestInbox.delete();
|
await contactRequestInbox.delete();
|
||||||
});
|
});
|
||||||
if (!accepted) {
|
if (!accepted) {
|
||||||
await (await pool.openRead(
|
await (await pool.openRead(item.localConversationRecordKey.toVeilid(),
|
||||||
contactInvitationRecord.localConversationRecordKey.toVeilid(),
|
|
||||||
parent: accountRecordKey))
|
parent: accountRecordKey))
|
||||||
.delete();
|
.delete();
|
||||||
}
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<ValidContactInvitation?> validateInvitation(
|
Future<ValidContactInvitation?> validateInvitation(
|
||||||
|
@ -10,7 +10,7 @@ import 'cubits.dart';
|
|||||||
typedef WaitingInvitationsBlocMapState
|
typedef WaitingInvitationsBlocMapState
|
||||||
= BlocMapState<TypedKey, AsyncValue<InvitationStatus>>;
|
= BlocMapState<TypedKey, AsyncValue<InvitationStatus>>;
|
||||||
|
|
||||||
// Map of contactInvitationListRecordKey to WaitingInvitationCubit
|
// Map of contactRequestInboxRecordKey to WaitingInvitationCubit
|
||||||
// Wraps a contact invitation cubit to watch for accept/reject
|
// Wraps a contact invitation cubit to watch for accept/reject
|
||||||
// Automatically follows the state of a ContactInvitationListCubit.
|
// Automatically follows the state of a ContactInvitationListCubit.
|
||||||
class WaitingInvitationsBlocMapCubit extends BlocMapCubit<TypedKey,
|
class WaitingInvitationsBlocMapCubit extends BlocMapCubit<TypedKey,
|
||||||
@ -20,6 +20,7 @@ class WaitingInvitationsBlocMapCubit extends BlocMapCubit<TypedKey,
|
|||||||
TypedKey, proto.ContactInvitationRecord> {
|
TypedKey, proto.ContactInvitationRecord> {
|
||||||
WaitingInvitationsBlocMapCubit(
|
WaitingInvitationsBlocMapCubit(
|
||||||
{required this.activeAccountInfo, required this.account});
|
{required this.activeAccountInfo, required this.account});
|
||||||
|
|
||||||
Future<void> addWaitingInvitation(
|
Future<void> addWaitingInvitation(
|
||||||
{required proto.ContactInvitationRecord
|
{required proto.ContactInvitationRecord
|
||||||
contactInvitationRecord}) async =>
|
contactInvitationRecord}) async =>
|
||||||
|
@ -53,7 +53,9 @@ class ContactInvitationItemWidget extends StatelessWidget {
|
|||||||
context.read<ContactInvitationListCubit>();
|
context.read<ContactInvitationListCubit>();
|
||||||
await contactInvitationListCubit.deleteInvitation(
|
await contactInvitationListCubit.deleteInvitation(
|
||||||
accepted: false,
|
accepted: false,
|
||||||
contactInvitationRecord: contactInvitationRecord);
|
contactRequestInboxRecordKey: contactInvitationRecord
|
||||||
|
.contactRequestInbox.recordKey
|
||||||
|
.toVeilid());
|
||||||
},
|
},
|
||||||
backgroundColor: scale.tertiaryScale.background,
|
backgroundColor: scale.tertiaryScale.background,
|
||||||
foregroundColor: scale.tertiaryScale.text,
|
foregroundColor: scale.tertiaryScale.text,
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
import 'package:async_tools/async_tools.dart';
|
import 'package:async_tools/async_tools.dart';
|
||||||
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
||||||
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:provider/provider.dart';
|
import 'package:provider/provider.dart';
|
||||||
|
import 'package:veilid_support/veilid_support.dart';
|
||||||
|
|
||||||
import '../../../account_manager/account_manager.dart';
|
import '../../../account_manager/account_manager.dart';
|
||||||
import '../../../chat/chat.dart';
|
import '../../../chat/chat.dart';
|
||||||
@ -13,65 +15,143 @@ import '../../../router/router.dart';
|
|||||||
import '../../../tools/tools.dart';
|
import '../../../tools/tools.dart';
|
||||||
|
|
||||||
class HomeAccountReadyShell extends StatefulWidget {
|
class HomeAccountReadyShell extends StatefulWidget {
|
||||||
const HomeAccountReadyShell({required this.child, super.key});
|
factory HomeAccountReadyShell(
|
||||||
|
{required BuildContext context, required Widget child, Key? key}) {
|
||||||
@override
|
// These must exist in order for the account to
|
||||||
HomeAccountReadyShellState createState() => HomeAccountReadyShellState();
|
// be considered 'ready' for this widget subtree
|
||||||
|
|
||||||
final Widget child;
|
|
||||||
}
|
|
||||||
|
|
||||||
class HomeAccountReadyShellState extends State<HomeAccountReadyShell> {
|
|
||||||
//
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
// These must be valid already before making this widget,
|
|
||||||
// per the ShellRoute above it
|
|
||||||
final activeLocalAccount = context.read<ActiveLocalAccountCubit>().state!;
|
final activeLocalAccount = context.read<ActiveLocalAccountCubit>().state!;
|
||||||
final accountInfo =
|
final accountInfo =
|
||||||
AccountRepository.instance.getAccountInfo(activeLocalAccount);
|
AccountRepository.instance.getAccountInfo(activeLocalAccount);
|
||||||
final activeAccountInfo = accountInfo.activeAccountInfo!;
|
final activeAccountInfo = accountInfo.activeAccountInfo!;
|
||||||
final routerCubit = context.read<RouterCubit>();
|
final routerCubit = context.read<RouterCubit>();
|
||||||
|
|
||||||
return Provider<ActiveAccountInfo>.value(
|
return HomeAccountReadyShell._(
|
||||||
value: activeAccountInfo,
|
activeLocalAccount: activeLocalAccount,
|
||||||
|
accountInfo: accountInfo,
|
||||||
|
activeAccountInfo: activeAccountInfo,
|
||||||
|
routerCubit: routerCubit,
|
||||||
|
key: key,
|
||||||
|
child: child);
|
||||||
|
}
|
||||||
|
const HomeAccountReadyShell._(
|
||||||
|
{required this.activeLocalAccount,
|
||||||
|
required this.accountInfo,
|
||||||
|
required this.activeAccountInfo,
|
||||||
|
required this.routerCubit,
|
||||||
|
required this.child,
|
||||||
|
super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
HomeAccountReadyShellState createState() => HomeAccountReadyShellState();
|
||||||
|
|
||||||
|
final Widget child;
|
||||||
|
final TypedKey activeLocalAccount;
|
||||||
|
final AccountInfo accountInfo;
|
||||||
|
final ActiveAccountInfo activeAccountInfo;
|
||||||
|
final RouterCubit routerCubit;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
||||||
|
super.debugFillProperties(properties);
|
||||||
|
properties
|
||||||
|
..add(DiagnosticsProperty<TypedKey>(
|
||||||
|
'activeLocalAccount', activeLocalAccount))
|
||||||
|
..add(DiagnosticsProperty<AccountInfo>('accountInfo', accountInfo))
|
||||||
|
..add(DiagnosticsProperty<ActiveAccountInfo>(
|
||||||
|
'activeAccountInfo', activeAccountInfo))
|
||||||
|
..add(DiagnosticsProperty<RouterCubit>('routerCubit', routerCubit));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class HomeAccountReadyShellState extends State<HomeAccountReadyShell> {
|
||||||
|
final SingleStateProcessor<WaitingInvitationsBlocMapState>
|
||||||
|
_singleInvitationStatusProcessor = SingleStateProcessor();
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Process all accepted or rejected invitations
|
||||||
|
void _invitationStatusListener(
|
||||||
|
BuildContext context, WaitingInvitationsBlocMapState state) {
|
||||||
|
_singleInvitationStatusProcessor.updateState(state,
|
||||||
|
closure: (newState) async {
|
||||||
|
final contactListCubit = context.read<ContactListCubit>();
|
||||||
|
final contactInvitationListCubit =
|
||||||
|
context.read<ContactInvitationListCubit>();
|
||||||
|
|
||||||
|
for (final entry in newState.entries) {
|
||||||
|
final contactRequestInboxRecordKey = entry.key;
|
||||||
|
final invStatus = entry.value.data?.value;
|
||||||
|
// Skip invitations that have not yet been accepted or rejected
|
||||||
|
if (invStatus == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete invitation and process the accepted or rejected contact
|
||||||
|
final acceptedContact = invStatus.acceptedContact;
|
||||||
|
if (acceptedContact != null) {
|
||||||
|
await contactInvitationListCubit.deleteInvitation(
|
||||||
|
accepted: true,
|
||||||
|
contactRequestInboxRecordKey: contactRequestInboxRecordKey);
|
||||||
|
|
||||||
|
// Accept
|
||||||
|
await contactListCubit.createContact(
|
||||||
|
remoteProfile: acceptedContact.remoteProfile,
|
||||||
|
remoteIdentity: acceptedContact.remoteIdentity,
|
||||||
|
remoteConversationRecordKey:
|
||||||
|
acceptedContact.remoteConversationRecordKey,
|
||||||
|
localConversationRecordKey:
|
||||||
|
acceptedContact.localConversationRecordKey,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// Reject
|
||||||
|
await contactInvitationListCubit.deleteInvitation(
|
||||||
|
accepted: false,
|
||||||
|
contactRequestInboxRecordKey: contactRequestInboxRecordKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) => Provider<ActiveAccountInfo>.value(
|
||||||
|
value: widget.activeAccountInfo,
|
||||||
child: BlocProvider(
|
child: BlocProvider(
|
||||||
create: (context) =>
|
create: (context) => AccountRecordCubit(
|
||||||
AccountRecordCubit(record: activeAccountInfo.accountRecord),
|
record: widget.activeAccountInfo.accountRecord),
|
||||||
child: Builder(builder: (context) {
|
child: Builder(builder: (context) {
|
||||||
final account =
|
final account =
|
||||||
context.watch<AccountRecordCubit>().state.data?.value;
|
context.watch<AccountRecordCubit>().state.data?.value;
|
||||||
if (account == null) {
|
if (account == null) {
|
||||||
return waitingPage();
|
return waitingPage();
|
||||||
}
|
}
|
||||||
return MultiBlocProvider(providers: [
|
return MultiBlocProvider(
|
||||||
|
providers: [
|
||||||
BlocProvider(
|
BlocProvider(
|
||||||
create: (context) => ContactInvitationListCubit(
|
create: (context) => ContactInvitationListCubit(
|
||||||
activeAccountInfo: activeAccountInfo,
|
activeAccountInfo: widget.activeAccountInfo,
|
||||||
account: account)),
|
account: account)),
|
||||||
BlocProvider(
|
BlocProvider(
|
||||||
create: (context) => ContactListCubit(
|
create: (context) => ContactListCubit(
|
||||||
activeAccountInfo: activeAccountInfo,
|
activeAccountInfo: widget.activeAccountInfo,
|
||||||
account: account)),
|
account: account)),
|
||||||
BlocProvider(
|
BlocProvider(
|
||||||
create: (context) => ChatListCubit(
|
create: (context) => ChatListCubit(
|
||||||
activeAccountInfo: activeAccountInfo,
|
activeAccountInfo: widget.activeAccountInfo,
|
||||||
account: account)),
|
account: account)),
|
||||||
BlocProvider(
|
BlocProvider(
|
||||||
create: (context) => ActiveConversationsBlocMapCubit(
|
create: (context) => ActiveConversationsBlocMapCubit(
|
||||||
activeAccountInfo: activeAccountInfo,
|
activeAccountInfo: widget.activeAccountInfo,
|
||||||
contactListCubit: context.read<ContactListCubit>())
|
contactListCubit: context.read<ContactListCubit>())
|
||||||
..follow(
|
..follow(
|
||||||
initialInputState: const AsyncValue.loading(),
|
initialInputState: const AsyncValue.loading(),
|
||||||
stream: context.read<ChatListCubit>().stream)),
|
stream: context.read<ChatListCubit>().stream)),
|
||||||
BlocProvider(
|
BlocProvider(
|
||||||
create: (context) => ActiveConversationMessagesBlocMapCubit(
|
create: (context) =>
|
||||||
activeAccountInfo: activeAccountInfo,
|
ActiveConversationMessagesBlocMapCubit(
|
||||||
|
activeAccountInfo: widget.activeAccountInfo,
|
||||||
)..follow(
|
)..follow(
|
||||||
initialInputState: IMap(),
|
initialInputState: IMap(),
|
||||||
stream: context
|
stream: context
|
||||||
@ -80,17 +160,23 @@ class HomeAccountReadyShellState extends State<HomeAccountReadyShell> {
|
|||||||
BlocProvider(
|
BlocProvider(
|
||||||
create: (context) => ActiveChatCubit(null)
|
create: (context) => ActiveChatCubit(null)
|
||||||
..withStateListen((event) {
|
..withStateListen((event) {
|
||||||
routerCubit.setHasActiveChat(event != null);
|
widget.routerCubit.setHasActiveChat(event != null);
|
||||||
})),
|
})),
|
||||||
BlocProvider(
|
BlocProvider(
|
||||||
create: (context) => WaitingInvitationsBlocMapCubit(
|
create: (context) => WaitingInvitationsBlocMapCubit(
|
||||||
activeAccountInfo: activeAccountInfo, account: account)
|
activeAccountInfo: widget.activeAccountInfo,
|
||||||
|
account: account)
|
||||||
..follow(
|
..follow(
|
||||||
initialInputState: const AsyncValue.loading(),
|
initialInputState: const AsyncValue.loading(),
|
||||||
stream: context
|
stream: context
|
||||||
.read<ContactInvitationListCubit>()
|
.read<ContactInvitationListCubit>()
|
||||||
.stream))
|
.stream))
|
||||||
], child: widget.child);
|
],
|
||||||
|
child: MultiBlocListener(listeners: [
|
||||||
|
BlocListener<WaitingInvitationsBlocMapCubit,
|
||||||
|
WaitingInvitationsBlocMapState>(
|
||||||
|
listener: _invitationStatusListener,
|
||||||
|
)
|
||||||
|
], child: widget.child));
|
||||||
})));
|
})));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -10,12 +10,12 @@ import 'home_account_missing.dart';
|
|||||||
import 'home_no_active.dart';
|
import 'home_no_active.dart';
|
||||||
|
|
||||||
class HomeShell extends StatefulWidget {
|
class HomeShell extends StatefulWidget {
|
||||||
const HomeShell({required this.child, super.key});
|
const HomeShell({required this.accountReadyBuilder, super.key});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
HomeShellState createState() => HomeShellState();
|
HomeShellState createState() => HomeShellState();
|
||||||
|
|
||||||
final Widget child;
|
final Builder accountReadyBuilder;
|
||||||
}
|
}
|
||||||
|
|
||||||
class HomeShellState extends State<HomeShell> {
|
class HomeShellState extends State<HomeShell> {
|
||||||
@ -32,7 +32,7 @@ class HomeShellState extends State<HomeShell> {
|
|||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget buildWithLogin(BuildContext context, Widget child) {
|
Widget buildWithLogin(BuildContext context) {
|
||||||
final activeLocalAccount = context.watch<ActiveLocalAccountCubit>().state;
|
final activeLocalAccount = context.watch<ActiveLocalAccountCubit>().state;
|
||||||
|
|
||||||
if (activeLocalAccount == null) {
|
if (activeLocalAccount == null) {
|
||||||
@ -56,7 +56,7 @@ class HomeShellState extends State<HomeShell> {
|
|||||||
child: BlocProvider(
|
child: BlocProvider(
|
||||||
create: (context) => AccountRecordCubit(
|
create: (context) => AccountRecordCubit(
|
||||||
record: accountInfo.activeAccountInfo!.accountRecord),
|
record: accountInfo.activeAccountInfo!.accountRecord),
|
||||||
child: child));
|
child: widget.accountReadyBuilder));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -72,6 +72,6 @@ class HomeShellState extends State<HomeShell> {
|
|||||||
child: DecoratedBox(
|
child: DecoratedBox(
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: scale.primaryScale.activeElementBackground),
|
color: scale.primaryScale.activeElementBackground),
|
||||||
child: buildWithLogin(context, widget.child))));
|
child: buildWithLogin(context))));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@ import 'dart:async';
|
|||||||
import 'package:bloc/bloc.dart';
|
import 'package:bloc/bloc.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter/widgets.dart';
|
import 'package:flutter/widgets.dart';
|
||||||
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:freezed_annotation/freezed_annotation.dart';
|
import 'package:freezed_annotation/freezed_annotation.dart';
|
||||||
import 'package:go_router/go_router.dart';
|
import 'package:go_router/go_router.dart';
|
||||||
import 'package:stream_transform/stream_transform.dart';
|
import 'package:stream_transform/stream_transform.dart';
|
||||||
@ -68,8 +69,10 @@ class RouterCubit extends Cubit<RouterState> {
|
|||||||
),
|
),
|
||||||
ShellRoute(
|
ShellRoute(
|
||||||
navigatorKey: _homeNavKey,
|
navigatorKey: _homeNavKey,
|
||||||
builder: (context, state, child) =>
|
builder: (context, state, child) => HomeShell(
|
||||||
HomeShell(child: HomeAccountReadyShell(child: child)),
|
accountReadyBuilder: Builder(
|
||||||
|
builder: (context) =>
|
||||||
|
HomeAccountReadyShell(context: context, child: child))),
|
||||||
routes: [
|
routes: [
|
||||||
GoRoute(
|
GoRoute(
|
||||||
path: '/home',
|
path: '/home',
|
||||||
|
@ -10,41 +10,20 @@ class AsyncTransformerCubit<T, S> extends Cubit<AsyncValue<T>> {
|
|||||||
_subscription = input.stream.listen(_asyncTransform);
|
_subscription = input.stream.listen(_asyncTransform);
|
||||||
}
|
}
|
||||||
void _asyncTransform(AsyncValue<S> newInputState) {
|
void _asyncTransform(AsyncValue<S> newInputState) {
|
||||||
// Use a singlefuture here to ensure we get dont lose any updates
|
_singleStateProcessor.updateState(newInputState, closure: (newState) async {
|
||||||
// If the input stream gives us an update while we are
|
|
||||||
// still processing the last update, the most recent input state will
|
|
||||||
// be saved and processed eventually.
|
|
||||||
singleFuture(this, () async {
|
|
||||||
var newState = newInputState;
|
|
||||||
var done = false;
|
|
||||||
while (!done) {
|
|
||||||
// Emit the transformed state
|
// Emit the transformed state
|
||||||
try {
|
try {
|
||||||
if (newState is AsyncLoading) {
|
if (newState is AsyncLoading<S>) {
|
||||||
return AsyncValue<T>.loading();
|
emit(const AsyncValue.loading());
|
||||||
}
|
} else if (newState is AsyncError<S>) {
|
||||||
if (newState is AsyncError) {
|
emit(AsyncValue.error(newState.error, newState.stackTrace));
|
||||||
final newStateError = newState as AsyncError<S>;
|
} else {
|
||||||
return AsyncValue<T>.error(
|
|
||||||
newStateError.error, newStateError.stackTrace);
|
|
||||||
}
|
|
||||||
final transformedState = await transform(newState.data!.value);
|
final transformedState = await transform(newState.data!.value);
|
||||||
emit(transformedState);
|
emit(transformedState);
|
||||||
|
}
|
||||||
} on Exception catch (e, st) {
|
} on Exception catch (e, st) {
|
||||||
emit(AsyncValue.error(e, st));
|
emit(AsyncValue.error(e, st));
|
||||||
}
|
}
|
||||||
// See if there's another state change to process
|
|
||||||
final next = _nextInputState;
|
|
||||||
_nextInputState = null;
|
|
||||||
if (next != null) {
|
|
||||||
newState = next;
|
|
||||||
} else {
|
|
||||||
done = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, onBusy: () {
|
|
||||||
// Keep this state until we process again
|
|
||||||
_nextInputState = newInputState;
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -56,7 +35,8 @@ class AsyncTransformerCubit<T, S> extends Cubit<AsyncValue<T>> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Cubit<AsyncValue<S>> input;
|
Cubit<AsyncValue<S>> input;
|
||||||
AsyncValue<S>? _nextInputState;
|
final SingleStateProcessor<AsyncValue<S>> _singleStateProcessor =
|
||||||
|
SingleStateProcessor();
|
||||||
Future<AsyncValue<T>> Function(S) transform;
|
Future<AsyncValue<T>> Function(S) transform;
|
||||||
late final StreamSubscription<AsyncValue<S>> _subscription;
|
late final StreamSubscription<AsyncValue<S>> _subscription;
|
||||||
}
|
}
|
||||||
|
@ -30,16 +30,8 @@ abstract mixin class StateFollower<S extends Object, K, V> {
|
|||||||
Future<void> updateState(K key, V value);
|
Future<void> updateState(K key, V value);
|
||||||
|
|
||||||
void _updateFollow(S newInputState) {
|
void _updateFollow(S newInputState) {
|
||||||
// Use a singlefuture here to ensure we get dont lose any updates
|
_singleStateProcessor.updateState(getStateMap(newInputState),
|
||||||
// If the input stream gives us an update while we are
|
closure: (newStateMap) async {
|
||||||
// still processing the last update, the most recent input state will
|
|
||||||
// be saved and processed eventually.
|
|
||||||
final newInputStateMap = getStateMap(newInputState);
|
|
||||||
|
|
||||||
singleFuture(this, () async {
|
|
||||||
var newStateMap = newInputStateMap;
|
|
||||||
var done = false;
|
|
||||||
while (!done) {
|
|
||||||
for (final k in _lastInputStateMap.keys) {
|
for (final k in _lastInputStateMap.keys) {
|
||||||
if (!newStateMap.containsKey(k)) {
|
if (!newStateMap.containsKey(k)) {
|
||||||
// deleted
|
// deleted
|
||||||
@ -56,23 +48,11 @@ abstract mixin class StateFollower<S extends Object, K, V> {
|
|||||||
|
|
||||||
// Keep this state map for the next time
|
// Keep this state map for the next time
|
||||||
_lastInputStateMap = newStateMap;
|
_lastInputStateMap = newStateMap;
|
||||||
|
|
||||||
// See if there's another state change to process
|
|
||||||
final next = _nextInputStateMap;
|
|
||||||
_nextInputStateMap = null;
|
|
||||||
if (next != null) {
|
|
||||||
newStateMap = next;
|
|
||||||
} else {
|
|
||||||
done = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, onBusy: () {
|
|
||||||
// Keep this state until we process again
|
|
||||||
_nextInputStateMap = newInputStateMap;
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
late IMap<K, V> _lastInputStateMap;
|
late IMap<K, V> _lastInputStateMap;
|
||||||
IMap<K, V>? _nextInputStateMap;
|
final SingleStateProcessor<IMap<K, V>> _singleStateProcessor =
|
||||||
|
SingleStateProcessor();
|
||||||
late final StreamSubscription<S> _subscription;
|
late final StreamSubscription<S> _subscription;
|
||||||
}
|
}
|
||||||
|
@ -3,4 +3,6 @@ library;
|
|||||||
|
|
||||||
export 'src/async_tag_lock.dart';
|
export 'src/async_tag_lock.dart';
|
||||||
export 'src/async_value.dart';
|
export 'src/async_value.dart';
|
||||||
export 'src/single_async.dart';
|
export 'src/serial_future.dart';
|
||||||
|
export 'src/single_future.dart';
|
||||||
|
export 'src/single_state_processor.dart';
|
||||||
|
57
packages/async_tools/lib/src/serial_future.dart
Normal file
57
packages/async_tools/lib/src/serial_future.dart
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
// Process a single future at a time per tag queued serially
|
||||||
|
//
|
||||||
|
// The closure function is called to produce the future that is to be executed.
|
||||||
|
// If a future with a particular tag is still executing, it is queued serially
|
||||||
|
// and executed when the previous tagged future completes.
|
||||||
|
// When a tagged serialFuture finishes executing, the onDone callback is called.
|
||||||
|
// If an unhandled exception happens in the closure future, the onError callback
|
||||||
|
// is called.
|
||||||
|
|
||||||
|
import 'dart:async';
|
||||||
|
import 'dart:collection';
|
||||||
|
|
||||||
|
import 'async_tag_lock.dart';
|
||||||
|
|
||||||
|
AsyncTagLock<Object> _keys = AsyncTagLock();
|
||||||
|
typedef SerialFutureQueueItem = Future<void> Function();
|
||||||
|
Map<Object, Queue<SerialFutureQueueItem>> _queues = {};
|
||||||
|
|
||||||
|
SerialFutureQueueItem _makeSerialFutureQueueItem<T>(
|
||||||
|
Future<T> Function() closure,
|
||||||
|
void Function(T)? onDone,
|
||||||
|
void Function(Object e, StackTrace? st)? onError) =>
|
||||||
|
() async {
|
||||||
|
try {
|
||||||
|
final out = await closure();
|
||||||
|
if (onDone != null) {
|
||||||
|
onDone(out);
|
||||||
|
}
|
||||||
|
// ignore: avoid_catches_without_on_clauses
|
||||||
|
} catch (e, sp) {
|
||||||
|
if (onError != null) {
|
||||||
|
onError(e, sp);
|
||||||
|
} else {
|
||||||
|
rethrow;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void serialFuture<T>(Object tag, Future<T> Function() closure,
|
||||||
|
{void Function(T)? onDone,
|
||||||
|
void Function(Object e, StackTrace? st)? onError}) {
|
||||||
|
final queueItem = _makeSerialFutureQueueItem(closure, onDone, onError);
|
||||||
|
if (!_keys.tryLock(tag)) {
|
||||||
|
final queue = _queues[tag];
|
||||||
|
queue!.add(queueItem);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
final queue = _queues[tag] = Queue.from([queueItem]);
|
||||||
|
unawaited(() async {
|
||||||
|
do {
|
||||||
|
final queueItem = queue.removeFirst();
|
||||||
|
await queueItem();
|
||||||
|
} while (queue.isNotEmpty);
|
||||||
|
_queues.remove(tag);
|
||||||
|
_keys.unlockTag(tag);
|
||||||
|
}());
|
||||||
|
}
|
@ -1,25 +0,0 @@
|
|||||||
import 'dart:async';
|
|
||||||
|
|
||||||
import 'async_tag_lock.dart';
|
|
||||||
|
|
||||||
AsyncTagLock<Object> _keys = AsyncTagLock();
|
|
||||||
|
|
||||||
void singleFuture<T>(Object tag, Future<T> Function() closure,
|
|
||||||
{void Function()? onBusy, void Function(T)? onDone}) {
|
|
||||||
if (!_keys.tryLock(tag)) {
|
|
||||||
if (onBusy != null) {
|
|
||||||
onBusy();
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
unawaited(() async {
|
|
||||||
try {
|
|
||||||
final out = await closure();
|
|
||||||
if (onDone != null) {
|
|
||||||
onDone(out);
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
_keys.unlockTag(tag);
|
|
||||||
}
|
|
||||||
}());
|
|
||||||
}
|
|
42
packages/async_tools/lib/src/single_future.dart
Normal file
42
packages/async_tools/lib/src/single_future.dart
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
|
||||||
|
import 'async_tag_lock.dart';
|
||||||
|
|
||||||
|
AsyncTagLock<Object> _keys = AsyncTagLock();
|
||||||
|
|
||||||
|
// Process a single future at a time per tag
|
||||||
|
//
|
||||||
|
// The closure function is called to produce the future that is to be executed.
|
||||||
|
// If a future with a particular tag is still executing, the onBusy callback
|
||||||
|
// is called.
|
||||||
|
// When a tagged singleFuture finishes executing, the onDone callback is called.
|
||||||
|
// If an unhandled exception happens in the closure future, the onError callback
|
||||||
|
// is called.
|
||||||
|
void singleFuture<T>(Object tag, Future<T> Function() closure,
|
||||||
|
{void Function()? onBusy,
|
||||||
|
void Function(T)? onDone,
|
||||||
|
void Function(Object e, StackTrace? st)? onError}) {
|
||||||
|
if (!_keys.tryLock(tag)) {
|
||||||
|
if (onBusy != null) {
|
||||||
|
onBusy();
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
unawaited(() async {
|
||||||
|
try {
|
||||||
|
final out = await closure();
|
||||||
|
if (onDone != null) {
|
||||||
|
onDone(out);
|
||||||
|
}
|
||||||
|
// ignore: avoid_catches_without_on_clauses
|
||||||
|
} catch (e, sp) {
|
||||||
|
if (onError != null) {
|
||||||
|
onError(e, sp);
|
||||||
|
} else {
|
||||||
|
rethrow;
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
_keys.unlockTag(tag);
|
||||||
|
}
|
||||||
|
}());
|
||||||
|
}
|
46
packages/async_tools/lib/src/single_state_processor.dart
Normal file
46
packages/async_tools/lib/src/single_state_processor.dart
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
|
||||||
|
import '../async_tools.dart';
|
||||||
|
|
||||||
|
// Process a single state update at a time ensuring the most
|
||||||
|
// recent state gets processed asynchronously, possibly skipping
|
||||||
|
// states that happen while a previous state is still being processed.
|
||||||
|
//
|
||||||
|
// Eventually this will always process the most recent state passed to
|
||||||
|
// updateState.
|
||||||
|
//
|
||||||
|
// This is useful for processing state changes asynchronously without waiting
|
||||||
|
// from a synchronous execution context
|
||||||
|
class SingleStateProcessor<State> {
|
||||||
|
SingleStateProcessor();
|
||||||
|
|
||||||
|
void updateState(State newInputState,
|
||||||
|
{required Future<void> Function(State) closure}) {
|
||||||
|
// Use a singlefuture here to ensure we get dont lose any updates
|
||||||
|
// If the input stream gives us an update while we are
|
||||||
|
// still processing the last update, the most recent input state will
|
||||||
|
// be saved and processed eventually.
|
||||||
|
|
||||||
|
singleFuture(this, () async {
|
||||||
|
var newState = newInputState;
|
||||||
|
var done = false;
|
||||||
|
while (!done) {
|
||||||
|
await closure(newState);
|
||||||
|
|
||||||
|
// See if there's another state change to process
|
||||||
|
final next = _nextState;
|
||||||
|
_nextState = null;
|
||||||
|
if (next != null) {
|
||||||
|
newState = next;
|
||||||
|
} else {
|
||||||
|
done = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, onBusy: () {
|
||||||
|
// Keep this state until we process again
|
||||||
|
_nextState = newInputState;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
State? _nextState;
|
||||||
|
}
|
@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
library dht_support;
|
library dht_support;
|
||||||
|
|
||||||
export 'src/dht_record.dart';
|
|
||||||
export 'src/dht_record_crypto.dart';
|
export 'src/dht_record_crypto.dart';
|
||||||
export 'src/dht_record_cubit.dart';
|
export 'src/dht_record_cubit.dart';
|
||||||
export 'src/dht_record_pool.dart';
|
export 'src/dht_record_pool.dart';
|
||||||
|
@ -1,12 +1,4 @@
|
|||||||
import 'dart:async';
|
part of 'dht_record_pool.dart';
|
||||||
import 'dart:typed_data';
|
|
||||||
|
|
||||||
import 'package:equatable/equatable.dart';
|
|
||||||
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
|
||||||
import 'package:meta/meta.dart';
|
|
||||||
import 'package:protobuf/protobuf.dart';
|
|
||||||
|
|
||||||
import '../../../veilid_support.dart';
|
|
||||||
|
|
||||||
@immutable
|
@immutable
|
||||||
class DHTRecordWatchChange extends Equatable {
|
class DHTRecordWatchChange extends Equatable {
|
||||||
@ -14,7 +6,7 @@ class DHTRecordWatchChange extends Equatable {
|
|||||||
{required this.local, required this.data, required this.subkeys});
|
{required this.local, required this.data, required this.subkeys});
|
||||||
|
|
||||||
final bool local;
|
final bool local;
|
||||||
final Uint8List data;
|
final Uint8List? data;
|
||||||
final List<ValueSubkeyRange> subkeys;
|
final List<ValueSubkeyRange> subkeys;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -26,46 +18,41 @@ class DHTRecordWatchChange extends Equatable {
|
|||||||
class DHTRecord {
|
class DHTRecord {
|
||||||
DHTRecord(
|
DHTRecord(
|
||||||
{required VeilidRoutingContext routingContext,
|
{required VeilidRoutingContext routingContext,
|
||||||
required DHTRecordDescriptor recordDescriptor,
|
required SharedDHTRecordData sharedDHTRecordData,
|
||||||
int defaultSubkey = 0,
|
required int defaultSubkey,
|
||||||
KeyPair? writer,
|
required KeyPair? writer,
|
||||||
DHTRecordCrypto crypto = const DHTRecordCryptoPublic()})
|
required DHTRecordCrypto crypto})
|
||||||
: _crypto = crypto,
|
: _crypto = crypto,
|
||||||
_routingContext = routingContext,
|
_routingContext = routingContext,
|
||||||
_recordDescriptor = recordDescriptor,
|
|
||||||
_defaultSubkey = defaultSubkey,
|
_defaultSubkey = defaultSubkey,
|
||||||
_writer = writer,
|
_writer = writer,
|
||||||
_open = true,
|
_open = true,
|
||||||
_valid = true,
|
_valid = true,
|
||||||
_subkeySeqCache = {},
|
_sharedDHTRecordData = sharedDHTRecordData;
|
||||||
needsWatchStateUpdate = false,
|
|
||||||
inWatchStateUpdate = false;
|
|
||||||
|
|
||||||
|
final SharedDHTRecordData _sharedDHTRecordData;
|
||||||
final VeilidRoutingContext _routingContext;
|
final VeilidRoutingContext _routingContext;
|
||||||
final DHTRecordDescriptor _recordDescriptor;
|
|
||||||
final int _defaultSubkey;
|
final int _defaultSubkey;
|
||||||
final KeyPair? _writer;
|
final KeyPair? _writer;
|
||||||
final Map<int, int> _subkeySeqCache;
|
|
||||||
final DHTRecordCrypto _crypto;
|
final DHTRecordCrypto _crypto;
|
||||||
|
|
||||||
bool _open;
|
bool _open;
|
||||||
bool _valid;
|
bool _valid;
|
||||||
@internal
|
@internal
|
||||||
StreamController<DHTRecordWatchChange>? watchController;
|
StreamController<DHTRecordWatchChange>? watchController;
|
||||||
@internal
|
@internal
|
||||||
bool needsWatchStateUpdate;
|
|
||||||
@internal
|
|
||||||
bool inWatchStateUpdate;
|
|
||||||
@internal
|
|
||||||
WatchState? watchState;
|
WatchState? watchState;
|
||||||
|
|
||||||
int subkeyOrDefault(int subkey) => (subkey == -1) ? _defaultSubkey : subkey;
|
int subkeyOrDefault(int subkey) => (subkey == -1) ? _defaultSubkey : subkey;
|
||||||
|
|
||||||
VeilidRoutingContext get routingContext => _routingContext;
|
VeilidRoutingContext get routingContext => _routingContext;
|
||||||
TypedKey get key => _recordDescriptor.key;
|
TypedKey get key => _sharedDHTRecordData.recordDescriptor.key;
|
||||||
PublicKey get owner => _recordDescriptor.owner;
|
PublicKey get owner => _sharedDHTRecordData.recordDescriptor.owner;
|
||||||
KeyPair? get ownerKeyPair => _recordDescriptor.ownerKeyPair();
|
KeyPair? get ownerKeyPair =>
|
||||||
DHTSchema get schema => _recordDescriptor.schema;
|
_sharedDHTRecordData.recordDescriptor.ownerKeyPair();
|
||||||
int get subkeyCount => _recordDescriptor.schema.subkeyCount();
|
DHTSchema get schema => _sharedDHTRecordData.recordDescriptor.schema;
|
||||||
|
int get subkeyCount =>
|
||||||
|
_sharedDHTRecordData.recordDescriptor.schema.subkeyCount();
|
||||||
KeyPair? get writer => _writer;
|
KeyPair? get writer => _writer;
|
||||||
DHTRecordCrypto get crypto => _crypto;
|
DHTRecordCrypto get crypto => _crypto;
|
||||||
OwnedDHTRecordPointer get ownedDHTRecordPointer =>
|
OwnedDHTRecordPointer get ownedDHTRecordPointer =>
|
||||||
@ -79,22 +66,16 @@ class DHTRecord {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
await watchController?.close();
|
await watchController?.close();
|
||||||
await _routingContext.closeDHTRecord(_recordDescriptor.key);
|
await DHTRecordPool.instance._recordClosed(this);
|
||||||
DHTRecordPool.instance.recordClosed(_recordDescriptor.key);
|
|
||||||
_open = false;
|
_open = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> delete() async {
|
void _markDeleted() {
|
||||||
if (!_valid) {
|
|
||||||
throw StateError('already deleted');
|
|
||||||
}
|
|
||||||
if (_open) {
|
|
||||||
await close();
|
|
||||||
}
|
|
||||||
await DHTRecordPool.instance.deleteDeep(key);
|
|
||||||
_valid = false;
|
_valid = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> delete() => DHTRecordPool.instance.delete(key);
|
||||||
|
|
||||||
Future<T> scope<T>(Future<T> Function(DHTRecord) scopeFunction) async {
|
Future<T> scope<T>(Future<T> Function(DHTRecord) scopeFunction) async {
|
||||||
try {
|
try {
|
||||||
return await scopeFunction(this);
|
return await scopeFunction(this);
|
||||||
@ -134,17 +115,17 @@ class DHTRecord {
|
|||||||
bool forceRefresh = false,
|
bool forceRefresh = false,
|
||||||
bool onlyUpdates = false}) async {
|
bool onlyUpdates = false}) async {
|
||||||
subkey = subkeyOrDefault(subkey);
|
subkey = subkeyOrDefault(subkey);
|
||||||
final valueData = await _routingContext.getDHTValue(
|
final valueData = await _routingContext.getDHTValue(key, subkey,
|
||||||
_recordDescriptor.key, subkey, forceRefresh);
|
forceRefresh: forceRefresh);
|
||||||
if (valueData == null) {
|
if (valueData == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
final lastSeq = _subkeySeqCache[subkey];
|
final lastSeq = _sharedDHTRecordData.subkeySeqCache[subkey];
|
||||||
if (onlyUpdates && lastSeq != null && valueData.seq <= lastSeq) {
|
if (onlyUpdates && lastSeq != null && valueData.seq <= lastSeq) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
final out = _crypto.decrypt(valueData.data, subkey);
|
final out = _crypto.decrypt(valueData.data, subkey);
|
||||||
_subkeySeqCache[subkey] = valueData.seq;
|
_sharedDHTRecordData.subkeySeqCache[subkey] = valueData.seq;
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -176,17 +157,16 @@ class DHTRecord {
|
|||||||
Future<Uint8List?> tryWriteBytes(Uint8List newValue,
|
Future<Uint8List?> tryWriteBytes(Uint8List newValue,
|
||||||
{int subkey = -1}) async {
|
{int subkey = -1}) async {
|
||||||
subkey = subkeyOrDefault(subkey);
|
subkey = subkeyOrDefault(subkey);
|
||||||
final lastSeq = _subkeySeqCache[subkey];
|
final lastSeq = _sharedDHTRecordData.subkeySeqCache[subkey];
|
||||||
final encryptedNewValue = await _crypto.encrypt(newValue, subkey);
|
final encryptedNewValue = await _crypto.encrypt(newValue, subkey);
|
||||||
|
|
||||||
// Set the new data if possible
|
// Set the new data if possible
|
||||||
var newValueData = await _routingContext.setDHTValue(
|
var newValueData =
|
||||||
_recordDescriptor.key, subkey, encryptedNewValue);
|
await _routingContext.setDHTValue(key, subkey, encryptedNewValue);
|
||||||
if (newValueData == null) {
|
if (newValueData == null) {
|
||||||
// A newer value wasn't found on the set, but
|
// A newer value wasn't found on the set, but
|
||||||
// we may get a newer value when getting the value for the sequence number
|
// we may get a newer value when getting the value for the sequence number
|
||||||
newValueData = await _routingContext.getDHTValue(
|
newValueData = await _routingContext.getDHTValue(key, subkey);
|
||||||
_recordDescriptor.key, subkey, false);
|
|
||||||
if (newValueData == null) {
|
if (newValueData == null) {
|
||||||
assert(newValueData != null, "can't get value that was just set");
|
assert(newValueData != null, "can't get value that was just set");
|
||||||
return null;
|
return null;
|
||||||
@ -195,13 +175,13 @@ class DHTRecord {
|
|||||||
|
|
||||||
// Record new sequence number
|
// Record new sequence number
|
||||||
final isUpdated = newValueData.seq != lastSeq;
|
final isUpdated = newValueData.seq != lastSeq;
|
||||||
_subkeySeqCache[subkey] = newValueData.seq;
|
_sharedDHTRecordData.subkeySeqCache[subkey] = newValueData.seq;
|
||||||
|
|
||||||
// See if the encrypted data returned is exactly the same
|
// See if the encrypted data returned is exactly the same
|
||||||
// if so, shortcut and don't bother decrypting it
|
// if so, shortcut and don't bother decrypting it
|
||||||
if (newValueData.data.equals(encryptedNewValue)) {
|
if (newValueData.data.equals(encryptedNewValue)) {
|
||||||
if (isUpdated) {
|
if (isUpdated) {
|
||||||
addLocalValueChange(newValue, subkey);
|
_addLocalValueChange(newValue, subkey);
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -209,36 +189,35 @@ class DHTRecord {
|
|||||||
// Decrypt value to return it
|
// Decrypt value to return it
|
||||||
final decryptedNewValue = await _crypto.decrypt(newValueData.data, subkey);
|
final decryptedNewValue = await _crypto.decrypt(newValueData.data, subkey);
|
||||||
if (isUpdated) {
|
if (isUpdated) {
|
||||||
addLocalValueChange(decryptedNewValue, subkey);
|
_addLocalValueChange(decryptedNewValue, subkey);
|
||||||
}
|
}
|
||||||
return decryptedNewValue;
|
return decryptedNewValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> eventualWriteBytes(Uint8List newValue, {int subkey = -1}) async {
|
Future<void> eventualWriteBytes(Uint8List newValue, {int subkey = -1}) async {
|
||||||
subkey = subkeyOrDefault(subkey);
|
subkey = subkeyOrDefault(subkey);
|
||||||
final lastSeq = _subkeySeqCache[subkey];
|
final lastSeq = _sharedDHTRecordData.subkeySeqCache[subkey];
|
||||||
final encryptedNewValue = await _crypto.encrypt(newValue, subkey);
|
final encryptedNewValue = await _crypto.encrypt(newValue, subkey);
|
||||||
|
|
||||||
ValueData? newValueData;
|
ValueData? newValueData;
|
||||||
do {
|
do {
|
||||||
do {
|
do {
|
||||||
// Set the new data
|
// Set the new data
|
||||||
newValueData = await _routingContext.setDHTValue(
|
newValueData =
|
||||||
_recordDescriptor.key, subkey, encryptedNewValue);
|
await _routingContext.setDHTValue(key, subkey, encryptedNewValue);
|
||||||
|
|
||||||
// Repeat if newer data on the network was found
|
// Repeat if newer data on the network was found
|
||||||
} while (newValueData != null);
|
} while (newValueData != null);
|
||||||
|
|
||||||
// Get the data to check its sequence number
|
// Get the data to check its sequence number
|
||||||
newValueData = await _routingContext.getDHTValue(
|
newValueData = await _routingContext.getDHTValue(key, subkey);
|
||||||
_recordDescriptor.key, subkey, false);
|
|
||||||
if (newValueData == null) {
|
if (newValueData == null) {
|
||||||
assert(newValueData != null, "can't get value that was just set");
|
assert(newValueData != null, "can't get value that was just set");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Record new sequence number
|
// Record new sequence number
|
||||||
_subkeySeqCache[subkey] = newValueData.seq;
|
_sharedDHTRecordData.subkeySeqCache[subkey] = newValueData.seq;
|
||||||
|
|
||||||
// The encrypted data returned should be exactly the same
|
// The encrypted data returned should be exactly the same
|
||||||
// as what we are trying to set,
|
// as what we are trying to set,
|
||||||
@ -247,7 +226,7 @@ class DHTRecord {
|
|||||||
|
|
||||||
final isUpdated = newValueData.seq != lastSeq;
|
final isUpdated = newValueData.seq != lastSeq;
|
||||||
if (isUpdated) {
|
if (isUpdated) {
|
||||||
addLocalValueChange(newValue, subkey);
|
_addLocalValueChange(newValue, subkey);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -258,8 +237,7 @@ class DHTRecord {
|
|||||||
|
|
||||||
// Get the existing data, do not allow force refresh here
|
// Get the existing data, do not allow force refresh here
|
||||||
// because if we need a refresh the setDHTValue will fail anyway
|
// because if we need a refresh the setDHTValue will fail anyway
|
||||||
var oldValue =
|
var oldValue = await get(subkey: subkey);
|
||||||
await get(subkey: subkey, forceRefresh: false, onlyUpdates: false);
|
|
||||||
|
|
||||||
do {
|
do {
|
||||||
// Update the data
|
// Update the data
|
||||||
@ -314,16 +292,16 @@ class DHTRecord {
|
|||||||
int? count}) async {
|
int? count}) async {
|
||||||
// Set up watch requirements which will get picked up by the next tick
|
// Set up watch requirements which will get picked up by the next tick
|
||||||
final oldWatchState = watchState;
|
final oldWatchState = watchState;
|
||||||
watchState = WatchState(
|
watchState =
|
||||||
subkeys: subkeys?.lock, expiration: expiration, count: count);
|
WatchState(subkeys: subkeys, expiration: expiration, count: count);
|
||||||
if (oldWatchState != watchState) {
|
if (oldWatchState != watchState) {
|
||||||
needsWatchStateUpdate = true;
|
_sharedDHTRecordData.needsWatchStateUpdate = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<StreamSubscription<DHTRecordWatchChange>> listen(
|
Future<StreamSubscription<DHTRecordWatchChange>> listen(
|
||||||
Future<void> Function(
|
Future<void> Function(
|
||||||
DHTRecord record, Uint8List data, List<ValueSubkeyRange> subkeys)
|
DHTRecord record, Uint8List? data, List<ValueSubkeyRange> subkeys)
|
||||||
onUpdate,
|
onUpdate,
|
||||||
{bool localChanges = true}) async {
|
{bool localChanges = true}) async {
|
||||||
// Set up watch requirements
|
// Set up watch requirements
|
||||||
@ -339,14 +317,16 @@ class DHTRecord {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Future.delayed(Duration.zero, () async {
|
Future.delayed(Duration.zero, () async {
|
||||||
final Uint8List data;
|
final Uint8List? data;
|
||||||
if (change.local) {
|
if (change.local) {
|
||||||
// local changes are not encrypted
|
// local changes are not encrypted
|
||||||
data = change.data;
|
data = change.data;
|
||||||
} else {
|
} else {
|
||||||
// incoming/remote changes are encrypted
|
// incoming/remote changes are encrypted
|
||||||
data =
|
final changeData = change.data;
|
||||||
await _crypto.decrypt(change.data, change.subkeys.first.low);
|
data = changeData == null
|
||||||
|
? null
|
||||||
|
: await _crypto.decrypt(changeData, change.subkeys.first.low);
|
||||||
}
|
}
|
||||||
await onUpdate(this, data, change.subkeys);
|
await onUpdate(this, data, change.subkeys);
|
||||||
});
|
});
|
||||||
@ -362,17 +342,48 @@ class DHTRecord {
|
|||||||
// Tear down watch requirements
|
// Tear down watch requirements
|
||||||
if (watchState != null) {
|
if (watchState != null) {
|
||||||
watchState = null;
|
watchState = null;
|
||||||
needsWatchStateUpdate = true;
|
_sharedDHTRecordData.needsWatchStateUpdate = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void addLocalValueChange(Uint8List data, int subkey) {
|
void _addValueChange(
|
||||||
|
{required bool local,
|
||||||
|
required Uint8List data,
|
||||||
|
required List<ValueSubkeyRange> subkeys}) {
|
||||||
|
final ws = watchState;
|
||||||
|
if (ws != null) {
|
||||||
|
final watchedSubkeys = ws.subkeys;
|
||||||
|
if (watchedSubkeys == null) {
|
||||||
|
// Report all subkeys
|
||||||
|
watchController?.add(
|
||||||
|
DHTRecordWatchChange(local: false, data: data, subkeys: subkeys));
|
||||||
|
} else {
|
||||||
|
// Only some subkeys are being watched, see if the reported update
|
||||||
|
// overlaps the subkeys being watched
|
||||||
|
final overlappedSubkeys = watchedSubkeys.intersectSubkeys(subkeys);
|
||||||
|
// If the reported data isn't within the
|
||||||
|
// range we care about, don't pass it through
|
||||||
|
final overlappedFirstSubkey = overlappedSubkeys.firstSubkey;
|
||||||
|
final updateFirstSubkey = subkeys.firstSubkey;
|
||||||
|
final updatedData = (overlappedFirstSubkey != null &&
|
||||||
|
updateFirstSubkey != null &&
|
||||||
|
overlappedFirstSubkey == updateFirstSubkey)
|
||||||
|
? data
|
||||||
|
: null;
|
||||||
|
// Report only wathced subkeys
|
||||||
watchController?.add(DHTRecordWatchChange(
|
watchController?.add(DHTRecordWatchChange(
|
||||||
local: true, data: data, subkeys: [ValueSubkeyRange.single(subkey)]));
|
local: local, data: updatedData, subkeys: overlappedSubkeys));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _addLocalValueChange(Uint8List data, int subkey) {
|
||||||
|
_addValueChange(
|
||||||
|
local: true, data: data, subkeys: [ValueSubkeyRange.single(subkey)]);
|
||||||
}
|
}
|
||||||
|
|
||||||
void addRemoteValueChange(VeilidUpdateValueChange update) {
|
void addRemoteValueChange(VeilidUpdateValueChange update) {
|
||||||
watchController?.add(DHTRecordWatchChange(
|
_addValueChange(
|
||||||
local: false, data: update.valueData.data, subkeys: update.subkeys));
|
local: false, data: update.valueData.data, subkeys: update.subkeys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,8 +3,8 @@ import 'dart:typed_data';
|
|||||||
import '../../../../veilid_support.dart';
|
import '../../../../veilid_support.dart';
|
||||||
|
|
||||||
abstract class DHTRecordCrypto {
|
abstract class DHTRecordCrypto {
|
||||||
FutureOr<Uint8List> encrypt(Uint8List data, int subkey);
|
Future<Uint8List> encrypt(Uint8List data, int subkey);
|
||||||
FutureOr<Uint8List> decrypt(Uint8List data, int subkey);
|
Future<Uint8List> decrypt(Uint8List data, int subkey);
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////
|
////////////////////////////////////
|
||||||
@ -32,11 +32,11 @@ class DHTRecordCryptoPrivate implements DHTRecordCrypto {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
FutureOr<Uint8List> encrypt(Uint8List data, int subkey) =>
|
Future<Uint8List> encrypt(Uint8List data, int subkey) =>
|
||||||
_cryptoSystem.encryptNoAuthWithNonce(data, _secretKey);
|
_cryptoSystem.encryptNoAuthWithNonce(data, _secretKey);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
FutureOr<Uint8List> decrypt(Uint8List data, int subkey) =>
|
Future<Uint8List> decrypt(Uint8List data, int subkey) =>
|
||||||
_cryptoSystem.decryptNoAuthWithNonce(data, _secretKey);
|
_cryptoSystem.decryptNoAuthWithNonce(data, _secretKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -46,8 +46,8 @@ class DHTRecordCryptoPublic implements DHTRecordCrypto {
|
|||||||
const DHTRecordCryptoPublic();
|
const DHTRecordCryptoPublic();
|
||||||
|
|
||||||
@override
|
@override
|
||||||
FutureOr<Uint8List> encrypt(Uint8List data, int subkey) => data;
|
Future<Uint8List> encrypt(Uint8List data, int subkey) async => data;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
FutureOr<Uint8List> decrypt(Uint8List data, int subkey) => data;
|
Future<Uint8List> decrypt(Uint8List data, int subkey) async => data;
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ import '../../veilid_support.dart';
|
|||||||
|
|
||||||
typedef InitialStateFunction<T> = Future<T?> Function(DHTRecord);
|
typedef InitialStateFunction<T> = Future<T?> Function(DHTRecord);
|
||||||
typedef StateFunction<T> = Future<T?> Function(
|
typedef StateFunction<T> = Future<T?> Function(
|
||||||
DHTRecord, List<ValueSubkeyRange>, Uint8List);
|
DHTRecord, List<ValueSubkeyRange>, Uint8List?);
|
||||||
|
|
||||||
class DHTRecordCubit<T> extends Cubit<AsyncValue<T>> {
|
class DHTRecordCubit<T> extends Cubit<AsyncValue<T>> {
|
||||||
DHTRecordCubit({
|
DHTRecordCubit({
|
||||||
@ -28,9 +28,8 @@ class DHTRecordCubit<T> extends Cubit<AsyncValue<T>> {
|
|||||||
|
|
||||||
DHTRecordCubit.value({
|
DHTRecordCubit.value({
|
||||||
required DHTRecord record,
|
required DHTRecord record,
|
||||||
required Future<T?> Function(DHTRecord) initialStateFunction,
|
required InitialStateFunction<T> initialStateFunction,
|
||||||
required Future<T?> Function(DHTRecord, List<ValueSubkeyRange>, Uint8List)
|
required StateFunction<T> stateFunction,
|
||||||
stateFunction,
|
|
||||||
}) : _record = record,
|
}) : _record = record,
|
||||||
_stateFunction = stateFunction,
|
_stateFunction = stateFunction,
|
||||||
_wantsCloseRecord = false,
|
_wantsCloseRecord = false,
|
||||||
@ -41,9 +40,8 @@ class DHTRecordCubit<T> extends Cubit<AsyncValue<T>> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _init(
|
Future<void> _init(
|
||||||
Future<T?> Function(DHTRecord) initialStateFunction,
|
InitialStateFunction<T> initialStateFunction,
|
||||||
Future<T?> Function(DHTRecord, List<ValueSubkeyRange>, Uint8List)
|
StateFunction<T> stateFunction,
|
||||||
stateFunction,
|
|
||||||
) async {
|
) async {
|
||||||
// Make initial state update
|
// Make initial state update
|
||||||
try {
|
try {
|
||||||
@ -142,7 +140,7 @@ class DefaultDHTRecordCubit<T> extends DHTRecordCubit<T> {
|
|||||||
if (subkeys.containsSubkey(defaultSubkey)) {
|
if (subkeys.containsSubkey(defaultSubkey)) {
|
||||||
final Uint8List data;
|
final Uint8List data;
|
||||||
final firstSubkey = subkeys.firstOrNull!.low;
|
final firstSubkey = subkeys.firstOrNull!.low;
|
||||||
if (firstSubkey != defaultSubkey) {
|
if (firstSubkey != defaultSubkey || updatedata == null) {
|
||||||
final maybeData = await record.get(forceRefresh: true);
|
final maybeData = await record.get(forceRefresh: true);
|
||||||
if (maybeData == null) {
|
if (maybeData == null) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -1,15 +1,21 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
import 'dart:math';
|
||||||
|
import 'dart:typed_data';
|
||||||
|
|
||||||
import 'package:async_tools/async_tools.dart';
|
import 'package:async_tools/async_tools.dart';
|
||||||
import 'package:equatable/equatable.dart';
|
import 'package:equatable/equatable.dart';
|
||||||
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
||||||
import 'package:freezed_annotation/freezed_annotation.dart';
|
import 'package:freezed_annotation/freezed_annotation.dart';
|
||||||
|
import 'package:mutex/mutex.dart';
|
||||||
|
import 'package:protobuf/protobuf.dart';
|
||||||
|
|
||||||
import '../../../../veilid_support.dart';
|
import '../../../../veilid_support.dart';
|
||||||
|
|
||||||
part 'dht_record_pool.freezed.dart';
|
part 'dht_record_pool.freezed.dart';
|
||||||
part 'dht_record_pool.g.dart';
|
part 'dht_record_pool.g.dart';
|
||||||
|
|
||||||
|
part 'dht_record.dart';
|
||||||
|
|
||||||
/// Record pool that managed DHTRecords and allows for tagged deletion
|
/// Record pool that managed DHTRecords and allows for tagged deletion
|
||||||
@freezed
|
@freezed
|
||||||
class DHTRecordPoolAllocations with _$DHTRecordPoolAllocations {
|
class DHTRecordPoolAllocations with _$DHTRecordPoolAllocations {
|
||||||
@ -39,13 +45,14 @@ class OwnedDHTRecordPointer with _$OwnedDHTRecordPointer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Watch state
|
/// Watch state
|
||||||
|
@immutable
|
||||||
class WatchState extends Equatable {
|
class WatchState extends Equatable {
|
||||||
const WatchState(
|
const WatchState(
|
||||||
{required this.subkeys,
|
{required this.subkeys,
|
||||||
required this.expiration,
|
required this.expiration,
|
||||||
required this.count,
|
required this.count,
|
||||||
this.realExpiration});
|
this.realExpiration});
|
||||||
final IList<ValueSubkeyRange>? subkeys;
|
final List<ValueSubkeyRange>? subkeys;
|
||||||
final Timestamp? expiration;
|
final Timestamp? expiration;
|
||||||
final int? count;
|
final int? count;
|
||||||
final Timestamp? realExpiration;
|
final Timestamp? realExpiration;
|
||||||
@ -54,23 +61,51 @@ class WatchState extends Equatable {
|
|||||||
List<Object?> get props => [subkeys, expiration, count, realExpiration];
|
List<Object?> get props => [subkeys, expiration, count, realExpiration];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Data shared amongst all DHTRecord instances
|
||||||
|
class SharedDHTRecordData {
|
||||||
|
SharedDHTRecordData(
|
||||||
|
{required this.recordDescriptor,
|
||||||
|
required this.defaultWriter,
|
||||||
|
required this.defaultRoutingContext});
|
||||||
|
DHTRecordDescriptor recordDescriptor;
|
||||||
|
KeyPair? defaultWriter;
|
||||||
|
VeilidRoutingContext defaultRoutingContext;
|
||||||
|
Map<int, int> subkeySeqCache = {};
|
||||||
|
bool inWatchStateUpdate = false;
|
||||||
|
bool needsWatchStateUpdate = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Per opened record data
|
||||||
|
class OpenedRecordInfo {
|
||||||
|
OpenedRecordInfo(
|
||||||
|
{required DHTRecordDescriptor recordDescriptor,
|
||||||
|
required KeyPair? defaultWriter,
|
||||||
|
required VeilidRoutingContext defaultRoutingContext})
|
||||||
|
: shared = SharedDHTRecordData(
|
||||||
|
recordDescriptor: recordDescriptor,
|
||||||
|
defaultWriter: defaultWriter,
|
||||||
|
defaultRoutingContext: defaultRoutingContext);
|
||||||
|
SharedDHTRecordData shared;
|
||||||
|
Set<DHTRecord> records = {};
|
||||||
|
}
|
||||||
|
|
||||||
class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
||||||
DHTRecordPool._(Veilid veilid, VeilidRoutingContext routingContext)
|
DHTRecordPool._(Veilid veilid, VeilidRoutingContext routingContext)
|
||||||
: _state = DHTRecordPoolAllocations(
|
: _state = DHTRecordPoolAllocations(
|
||||||
childrenByParent: IMap(),
|
childrenByParent: IMap(),
|
||||||
parentByChild: IMap(),
|
parentByChild: IMap(),
|
||||||
rootRecords: ISet()),
|
rootRecords: ISet()),
|
||||||
_opened = <TypedKey, DHTRecord>{},
|
_mutex = Mutex(),
|
||||||
_locks = AsyncTagLock(),
|
_opened = <TypedKey, OpenedRecordInfo>{},
|
||||||
_routingContext = routingContext,
|
_routingContext = routingContext,
|
||||||
_veilid = veilid;
|
_veilid = veilid;
|
||||||
|
|
||||||
// Persistent DHT record list
|
// Persistent DHT record list
|
||||||
DHTRecordPoolAllocations _state;
|
DHTRecordPoolAllocations _state;
|
||||||
// Lock table to ensure we don't open the same record more than once
|
// Create/open Mutex
|
||||||
final AsyncTagLock<TypedKey> _locks;
|
final Mutex _mutex;
|
||||||
// Which DHT records are currently open
|
// Which DHT records are currently open
|
||||||
final Map<TypedKey, DHTRecord> _opened;
|
final Map<TypedKey, OpenedRecordInfo> _opened;
|
||||||
// Default routing context to use for new keys
|
// Default routing context to use for new keys
|
||||||
final VeilidRoutingContext _routingContext;
|
final VeilidRoutingContext _routingContext;
|
||||||
// Convenience accessor
|
// Convenience accessor
|
||||||
@ -107,30 +142,106 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
|
|
||||||
Veilid get veilid => _veilid;
|
Veilid get veilid => _veilid;
|
||||||
|
|
||||||
void _recordOpened(DHTRecord record) {
|
Future<OpenedRecordInfo> _recordCreateInner(
|
||||||
if (_opened.containsKey(record.key)) {
|
{required VeilidRoutingContext dhtctx,
|
||||||
throw StateError('record already opened');
|
required DHTSchema schema,
|
||||||
|
KeyPair? writer,
|
||||||
|
TypedKey? parent}) async {
|
||||||
|
assert(_mutex.isLocked, 'should be locked here');
|
||||||
|
|
||||||
|
// Create the record
|
||||||
|
final recordDescriptor = await dhtctx.createDHTRecord(schema);
|
||||||
|
|
||||||
|
// Reopen if a writer is specified to ensure
|
||||||
|
// we switch the default writer
|
||||||
|
if (writer != null) {
|
||||||
|
await dhtctx.openDHTRecord(recordDescriptor.key, writer: writer);
|
||||||
}
|
}
|
||||||
_opened[record.key] = record;
|
final openedRecordInfo = OpenedRecordInfo(
|
||||||
|
recordDescriptor: recordDescriptor,
|
||||||
|
defaultWriter: writer ?? recordDescriptor.ownerKeyPair(),
|
||||||
|
defaultRoutingContext: dhtctx);
|
||||||
|
_opened[recordDescriptor.key] = openedRecordInfo;
|
||||||
|
|
||||||
|
// Register the dependency
|
||||||
|
await _addDependencyInner(parent, recordDescriptor.key);
|
||||||
|
|
||||||
|
return openedRecordInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
void recordClosed(TypedKey key) {
|
Future<OpenedRecordInfo> _recordOpenInner(
|
||||||
final rec = _opened.remove(key);
|
{required VeilidRoutingContext dhtctx,
|
||||||
if (rec == null) {
|
required TypedKey recordKey,
|
||||||
|
KeyPair? writer,
|
||||||
|
TypedKey? parent}) async {
|
||||||
|
assert(_mutex.isLocked, 'should be locked here');
|
||||||
|
|
||||||
|
// If we are opening a key that already exists
|
||||||
|
// make sure we are using the same parent if one was specified
|
||||||
|
_validateParent(parent, recordKey);
|
||||||
|
|
||||||
|
// See if this has been opened yet
|
||||||
|
final openedRecordInfo = _opened[recordKey];
|
||||||
|
if (openedRecordInfo == null) {
|
||||||
|
// Fresh open, just open the record
|
||||||
|
final recordDescriptor =
|
||||||
|
await dhtctx.openDHTRecord(recordKey, writer: writer);
|
||||||
|
final newOpenedRecordInfo = OpenedRecordInfo(
|
||||||
|
recordDescriptor: recordDescriptor,
|
||||||
|
defaultWriter: writer,
|
||||||
|
defaultRoutingContext: dhtctx);
|
||||||
|
_opened[recordDescriptor.key] = newOpenedRecordInfo;
|
||||||
|
|
||||||
|
// Register the dependency
|
||||||
|
await _addDependencyInner(parent, recordKey);
|
||||||
|
|
||||||
|
return newOpenedRecordInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Already opened
|
||||||
|
|
||||||
|
// See if we need to reopen the record with a default writer and possibly
|
||||||
|
// a different routing context
|
||||||
|
if (writer != null && openedRecordInfo.shared.defaultWriter == null) {
|
||||||
|
final newRecordDescriptor =
|
||||||
|
await dhtctx.openDHTRecord(recordKey, writer: writer);
|
||||||
|
openedRecordInfo.shared.defaultWriter = writer;
|
||||||
|
openedRecordInfo.shared.defaultRoutingContext = dhtctx;
|
||||||
|
if (openedRecordInfo.shared.recordDescriptor.ownerSecret == null) {
|
||||||
|
openedRecordInfo.shared.recordDescriptor = newRecordDescriptor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Register the dependency
|
||||||
|
await _addDependencyInner(parent, recordKey);
|
||||||
|
|
||||||
|
return openedRecordInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _recordClosed(DHTRecord record) async {
|
||||||
|
await _mutex.protect(() async {
|
||||||
|
final key = record.key;
|
||||||
|
final openedRecordInfo = _opened[key];
|
||||||
|
if (openedRecordInfo == null ||
|
||||||
|
!openedRecordInfo.records.remove(record)) {
|
||||||
throw StateError('record already closed');
|
throw StateError('record already closed');
|
||||||
}
|
}
|
||||||
_locks.unlockTag(key);
|
if (openedRecordInfo.records.isEmpty) {
|
||||||
|
await _routingContext.closeDHTRecord(key);
|
||||||
|
_opened.remove(key);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> deleteDeep(TypedKey parent) async {
|
Future<void> delete(TypedKey recordKey) async {
|
||||||
// Collect all dependencies
|
// Collect all dependencies (including the record itself)
|
||||||
final allDeps = <TypedKey>[];
|
final allDeps = <TypedKey>[];
|
||||||
final currentDeps = [parent];
|
final currentDeps = [recordKey];
|
||||||
while (currentDeps.isNotEmpty) {
|
while (currentDeps.isNotEmpty) {
|
||||||
final nextDep = currentDeps.removeLast();
|
final nextDep = currentDeps.removeLast();
|
||||||
|
|
||||||
// Remove this child from its parent
|
// Remove this child from its parent
|
||||||
await _removeDependency(nextDep);
|
await _removeDependencyInner(nextDep);
|
||||||
|
|
||||||
allDeps.add(nextDep);
|
allDeps.add(nextDep);
|
||||||
final childDeps =
|
final childDeps =
|
||||||
@ -138,18 +249,27 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
currentDeps.addAll(childDeps);
|
currentDeps.addAll(childDeps);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete all dependent records in parallel
|
// Delete all dependent records in parallel (including the record itself)
|
||||||
final allFutures = <Future<void>>[];
|
final allDeleteFutures = <Future<void>>[];
|
||||||
|
final allCloseFutures = <Future<void>>[];
|
||||||
|
final allDeletedRecords = <DHTRecord>{};
|
||||||
for (final dep in allDeps) {
|
for (final dep in allDeps) {
|
||||||
// If record is opened, close it first
|
// If record is opened, close it first
|
||||||
final rec = _opened[dep];
|
final openinfo = _opened[dep];
|
||||||
if (rec != null) {
|
if (openinfo != null) {
|
||||||
await rec.close();
|
for (final rec in openinfo.records) {
|
||||||
|
allCloseFutures.add(rec.close());
|
||||||
|
allDeletedRecords.add(rec);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Then delete
|
// Then delete
|
||||||
allFutures.add(_routingContext.deleteDHTRecord(dep));
|
allDeleteFutures.add(_routingContext.deleteDHTRecord(dep));
|
||||||
|
}
|
||||||
|
await Future.wait(allCloseFutures);
|
||||||
|
await Future.wait(allDeleteFutures);
|
||||||
|
for (final deletedRecord in allDeletedRecords) {
|
||||||
|
deletedRecord._markDeleted();
|
||||||
}
|
}
|
||||||
await Future.wait(allFutures);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void _validateParent(TypedKey? parent, TypedKey child) {
|
void _validateParent(TypedKey? parent, TypedKey child) {
|
||||||
@ -169,7 +289,8 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _addDependency(TypedKey? parent, TypedKey child) async {
|
Future<void> _addDependencyInner(TypedKey? parent, TypedKey child) async {
|
||||||
|
assert(_mutex.isLocked, 'should be locked here');
|
||||||
if (parent == null) {
|
if (parent == null) {
|
||||||
if (_state.rootRecords.contains(child)) {
|
if (_state.rootRecords.contains(child)) {
|
||||||
// Dependency already added
|
// Dependency already added
|
||||||
@ -191,7 +312,8 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _removeDependency(TypedKey child) async {
|
Future<void> _removeDependencyInner(TypedKey child) async {
|
||||||
|
assert(_mutex.isLocked, 'should be locked here');
|
||||||
if (_state.rootRecords.contains(child)) {
|
if (_state.rootRecords.contains(child)) {
|
||||||
_state = await store(
|
_state = await store(
|
||||||
_state.copyWith(rootRecords: _state.rootRecords.remove(child)));
|
_state.copyWith(rootRecords: _state.rootRecords.remove(child)));
|
||||||
@ -226,57 +348,52 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
int defaultSubkey = 0,
|
int defaultSubkey = 0,
|
||||||
DHTRecordCrypto? crypto,
|
DHTRecordCrypto? crypto,
|
||||||
KeyPair? writer,
|
KeyPair? writer,
|
||||||
}) async {
|
}) async =>
|
||||||
|
_mutex.protect(() async {
|
||||||
final dhtctx = routingContext ?? _routingContext;
|
final dhtctx = routingContext ?? _routingContext;
|
||||||
final recordDescriptor = await dhtctx.createDHTRecord(schema);
|
|
||||||
|
|
||||||
await _locks.lockTag(recordDescriptor.key);
|
final openedRecordInfo = await _recordCreateInner(
|
||||||
|
dhtctx: dhtctx, schema: schema, writer: writer, parent: parent);
|
||||||
|
|
||||||
final rec = DHTRecord(
|
final rec = DHTRecord(
|
||||||
routingContext: dhtctx,
|
routingContext: dhtctx,
|
||||||
recordDescriptor: recordDescriptor,
|
|
||||||
defaultSubkey: defaultSubkey,
|
defaultSubkey: defaultSubkey,
|
||||||
writer: writer ?? recordDescriptor.ownerKeyPair(),
|
sharedDHTRecordData: openedRecordInfo.shared,
|
||||||
|
writer: writer ??
|
||||||
|
openedRecordInfo.shared.recordDescriptor.ownerKeyPair(),
|
||||||
crypto: crypto ??
|
crypto: crypto ??
|
||||||
await DHTRecordCryptoPrivate.fromTypedKeyPair(
|
await DHTRecordCryptoPrivate.fromTypedKeyPair(openedRecordInfo
|
||||||
recordDescriptor.ownerTypedKeyPair()!));
|
.shared.recordDescriptor
|
||||||
|
.ownerTypedKeyPair()!));
|
||||||
|
|
||||||
await _addDependency(parent, rec.key);
|
openedRecordInfo.records.add(rec);
|
||||||
|
|
||||||
_recordOpened(rec);
|
|
||||||
|
|
||||||
return rec;
|
return rec;
|
||||||
}
|
});
|
||||||
|
|
||||||
/// Open a DHTRecord readonly
|
/// Open a DHTRecord readonly
|
||||||
Future<DHTRecord> openRead(TypedKey recordKey,
|
Future<DHTRecord> openRead(TypedKey recordKey,
|
||||||
{VeilidRoutingContext? routingContext,
|
{VeilidRoutingContext? routingContext,
|
||||||
TypedKey? parent,
|
TypedKey? parent,
|
||||||
int defaultSubkey = 0,
|
int defaultSubkey = 0,
|
||||||
DHTRecordCrypto? crypto}) async {
|
DHTRecordCrypto? crypto}) async =>
|
||||||
await _locks.lockTag(recordKey);
|
_mutex.protect(() async {
|
||||||
|
|
||||||
final dhtctx = routingContext ?? _routingContext;
|
final dhtctx = routingContext ?? _routingContext;
|
||||||
|
|
||||||
late final DHTRecord rec;
|
final openedRecordInfo = await _recordOpenInner(
|
||||||
// If we are opening a key that already exists
|
dhtctx: dhtctx, recordKey: recordKey, parent: parent);
|
||||||
// make sure we are using the same parent if one was specified
|
|
||||||
_validateParent(parent, recordKey);
|
|
||||||
|
|
||||||
// Open from the veilid api
|
final rec = DHTRecord(
|
||||||
final recordDescriptor = await dhtctx.openDHTRecord(recordKey, null);
|
|
||||||
rec = DHTRecord(
|
|
||||||
routingContext: dhtctx,
|
routingContext: dhtctx,
|
||||||
recordDescriptor: recordDescriptor,
|
|
||||||
defaultSubkey: defaultSubkey,
|
defaultSubkey: defaultSubkey,
|
||||||
|
sharedDHTRecordData: openedRecordInfo.shared,
|
||||||
|
writer: null,
|
||||||
crypto: crypto ?? const DHTRecordCryptoPublic());
|
crypto: crypto ?? const DHTRecordCryptoPublic());
|
||||||
|
|
||||||
// Register the dependency
|
openedRecordInfo.records.add(rec);
|
||||||
await _addDependency(parent, rec.key);
|
|
||||||
_recordOpened(rec);
|
|
||||||
|
|
||||||
return rec;
|
return rec;
|
||||||
}
|
});
|
||||||
|
|
||||||
/// Open a DHTRecord writable
|
/// Open a DHTRecord writable
|
||||||
Future<DHTRecord> openWrite(
|
Future<DHTRecord> openWrite(
|
||||||
@ -286,33 +403,29 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
TypedKey? parent,
|
TypedKey? parent,
|
||||||
int defaultSubkey = 0,
|
int defaultSubkey = 0,
|
||||||
DHTRecordCrypto? crypto,
|
DHTRecordCrypto? crypto,
|
||||||
}) async {
|
}) async =>
|
||||||
await _locks.lockTag(recordKey);
|
_mutex.protect(() async {
|
||||||
|
|
||||||
final dhtctx = routingContext ?? _routingContext;
|
final dhtctx = routingContext ?? _routingContext;
|
||||||
|
|
||||||
late final DHTRecord rec;
|
final openedRecordInfo = await _recordOpenInner(
|
||||||
// If we are opening a key that already exists
|
dhtctx: dhtctx,
|
||||||
// make sure we are using the same parent if one was specified
|
recordKey: recordKey,
|
||||||
_validateParent(parent, recordKey);
|
parent: parent,
|
||||||
|
writer: writer);
|
||||||
|
|
||||||
// Open from the veilid api
|
final rec = DHTRecord(
|
||||||
final recordDescriptor = await dhtctx.openDHTRecord(recordKey, writer);
|
|
||||||
rec = DHTRecord(
|
|
||||||
routingContext: dhtctx,
|
routingContext: dhtctx,
|
||||||
recordDescriptor: recordDescriptor,
|
|
||||||
defaultSubkey: defaultSubkey,
|
defaultSubkey: defaultSubkey,
|
||||||
writer: writer,
|
writer: writer,
|
||||||
|
sharedDHTRecordData: openedRecordInfo.shared,
|
||||||
crypto: crypto ??
|
crypto: crypto ??
|
||||||
await DHTRecordCryptoPrivate.fromTypedKeyPair(
|
await DHTRecordCryptoPrivate.fromTypedKeyPair(
|
||||||
TypedKeyPair.fromKeyPair(recordKey.kind, writer)));
|
TypedKeyPair.fromKeyPair(recordKey.kind, writer)));
|
||||||
|
|
||||||
// Register the dependency if specified
|
openedRecordInfo.records.add(rec);
|
||||||
await _addDependency(parent, rec.key);
|
|
||||||
_recordOpened(rec);
|
|
||||||
|
|
||||||
return rec;
|
return rec;
|
||||||
}
|
});
|
||||||
|
|
||||||
/// Open a DHTRecord owned
|
/// Open a DHTRecord owned
|
||||||
/// This is the same as writable but uses an OwnedDHTRecordPointer
|
/// This is the same as writable but uses an OwnedDHTRecordPointer
|
||||||
@ -336,9 +449,6 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
crypto: crypto,
|
crypto: crypto,
|
||||||
);
|
);
|
||||||
|
|
||||||
/// Look up an opened DHTRecord
|
|
||||||
DHTRecord? getOpenedRecord(TypedKey recordKey) => _opened[recordKey];
|
|
||||||
|
|
||||||
/// Get the parent of a DHTRecord key if it exists
|
/// Get the parent of a DHTRecord key if it exists
|
||||||
TypedKey? getParentRecordKey(TypedKey child) {
|
TypedKey? getParentRecordKey(TypedKey child) {
|
||||||
final childJson = child.toJson();
|
final childJson = child.toJson();
|
||||||
@ -351,33 +461,107 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
// Change
|
// Change
|
||||||
for (final kv in _opened.entries) {
|
for (final kv in _opened.entries) {
|
||||||
if (kv.key == updateValueChange.key) {
|
if (kv.key == updateValueChange.key) {
|
||||||
kv.value.addRemoteValueChange(updateValueChange);
|
for (final rec in kv.value.records) {
|
||||||
|
rec.addRemoteValueChange(updateValueChange);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
final now = Veilid.instance.now().value;
|
||||||
// Expired, process renewal if desired
|
// Expired, process renewal if desired
|
||||||
for (final kv in _opened.entries) {
|
for (final entry in _opened.entries) {
|
||||||
if (kv.key == updateValueChange.key) {
|
final openedKey = entry.key;
|
||||||
// Renew watch state
|
final openedRecordInfo = entry.value;
|
||||||
kv.value.needsWatchStateUpdate = true;
|
|
||||||
|
|
||||||
|
if (openedKey == updateValueChange.key) {
|
||||||
|
// Renew watch state for each opened recrod
|
||||||
|
for (final rec in openedRecordInfo.records) {
|
||||||
// See if the watch had an expiration and if it has expired
|
// See if the watch had an expiration and if it has expired
|
||||||
// otherwise the renewal will keep the same parameters
|
// otherwise the renewal will keep the same parameters
|
||||||
final watchState = kv.value.watchState;
|
final watchState = rec.watchState;
|
||||||
if (watchState != null) {
|
if (watchState != null) {
|
||||||
final exp = watchState.expiration;
|
final exp = watchState.expiration;
|
||||||
if (exp != null && exp.value < Veilid.instance.now().value) {
|
if (exp != null && exp.value < now) {
|
||||||
// Has expiration, and it has expired, clear watch state
|
// Has expiration, and it has expired, clear watch state
|
||||||
kv.value.watchState = null;
|
rec.watchState = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
openedRecordInfo.shared.needsWatchStateUpdate = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
WatchState? _collectUnionWatchState(Iterable<DHTRecord> records) {
|
||||||
|
// Collect union of opened record watch states
|
||||||
|
int? totalCount;
|
||||||
|
Timestamp? maxExpiration;
|
||||||
|
List<ValueSubkeyRange>? allSubkeys;
|
||||||
|
|
||||||
|
var noExpiration = false;
|
||||||
|
var everySubkey = false;
|
||||||
|
var cancelWatch = true;
|
||||||
|
|
||||||
|
for (final rec in records) {
|
||||||
|
final ws = rec.watchState;
|
||||||
|
if (ws != null) {
|
||||||
|
cancelWatch = false;
|
||||||
|
final wsCount = ws.count;
|
||||||
|
if (wsCount != null) {
|
||||||
|
totalCount = totalCount ?? 0 + min(wsCount, 0x7FFFFFFF);
|
||||||
|
totalCount = min(totalCount, 0x7FFFFFFF);
|
||||||
|
}
|
||||||
|
final wsExp = ws.expiration;
|
||||||
|
if (wsExp != null && !noExpiration) {
|
||||||
|
maxExpiration = maxExpiration == null
|
||||||
|
? wsExp
|
||||||
|
: wsExp.value > maxExpiration.value
|
||||||
|
? wsExp
|
||||||
|
: maxExpiration;
|
||||||
|
} else {
|
||||||
|
noExpiration = true;
|
||||||
|
}
|
||||||
|
final wsSubkeys = ws.subkeys;
|
||||||
|
if (wsSubkeys != null && !everySubkey) {
|
||||||
|
allSubkeys = allSubkeys == null
|
||||||
|
? wsSubkeys
|
||||||
|
: allSubkeys.unionSubkeys(wsSubkeys);
|
||||||
|
} else {
|
||||||
|
everySubkey = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (noExpiration) {
|
||||||
|
maxExpiration = null;
|
||||||
|
}
|
||||||
|
if (everySubkey) {
|
||||||
|
allSubkeys = null;
|
||||||
|
}
|
||||||
|
if (cancelWatch) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return WatchState(
|
||||||
|
subkeys: allSubkeys, expiration: maxExpiration, count: totalCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _updateWatchExpirations(
|
||||||
|
Iterable<DHTRecord> records, Timestamp realExpiration) {
|
||||||
|
for (final rec in records) {
|
||||||
|
final ws = rec.watchState;
|
||||||
|
if (ws != null) {
|
||||||
|
rec.watchState = WatchState(
|
||||||
|
subkeys: ws.subkeys,
|
||||||
|
expiration: ws.expiration,
|
||||||
|
count: ws.count,
|
||||||
|
realExpiration: realExpiration);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Ticker to check watch state change requests
|
/// Ticker to check watch state change requests
|
||||||
Future<void> tick() async {
|
Future<void> tick() async {
|
||||||
if (inTick) {
|
if (inTick) {
|
||||||
@ -386,53 +570,55 @@ class DHTRecordPool with TableDBBacked<DHTRecordPoolAllocations> {
|
|||||||
inTick = true;
|
inTick = true;
|
||||||
try {
|
try {
|
||||||
// See if any opened records need watch state changes
|
// See if any opened records need watch state changes
|
||||||
final unord = List<Future<void>>.empty(growable: true);
|
final unord = <Future<void>>[];
|
||||||
|
|
||||||
for (final kv in _opened.entries) {
|
for (final kv in _opened.entries) {
|
||||||
|
final openedRecordKey = kv.key;
|
||||||
|
final openedRecordInfo = kv.value;
|
||||||
|
final dhtctx = openedRecordInfo.shared.defaultRoutingContext;
|
||||||
|
|
||||||
// Check if already updating
|
// Check if already updating
|
||||||
if (kv.value.inWatchStateUpdate) {
|
if (openedRecordInfo.shared.inWatchStateUpdate) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (kv.value.needsWatchStateUpdate) {
|
if (openedRecordInfo.shared.needsWatchStateUpdate) {
|
||||||
kv.value.inWatchStateUpdate = true;
|
openedRecordInfo.shared.inWatchStateUpdate = true;
|
||||||
|
|
||||||
final ws = kv.value.watchState;
|
final watchState = _collectUnionWatchState(openedRecordInfo.records);
|
||||||
if (ws == null) {
|
|
||||||
|
// Apply watch changes for record
|
||||||
|
if (watchState == null) {
|
||||||
unord.add(() async {
|
unord.add(() async {
|
||||||
// Record needs watch cancel
|
// Record needs watch cancel
|
||||||
try {
|
try {
|
||||||
final done =
|
final done = await dhtctx.cancelDHTWatch(openedRecordKey);
|
||||||
await kv.value.routingContext.cancelDHTWatch(kv.key);
|
|
||||||
assert(done,
|
assert(done,
|
||||||
'should always be done when cancelling whole subkey range');
|
'should always be done when cancelling whole subkey range');
|
||||||
kv.value.needsWatchStateUpdate = false;
|
openedRecordInfo.shared.needsWatchStateUpdate = false;
|
||||||
} on VeilidAPIException {
|
} on VeilidAPIException {
|
||||||
// Failed to cancel DHT watch, try again next tick
|
// Failed to cancel DHT watch, try again next tick
|
||||||
}
|
}
|
||||||
kv.value.inWatchStateUpdate = false;
|
openedRecordInfo.shared.inWatchStateUpdate = false;
|
||||||
}());
|
}());
|
||||||
} else {
|
} else {
|
||||||
unord.add(() async {
|
unord.add(() async {
|
||||||
// Record needs new watch
|
// Record needs new watch
|
||||||
try {
|
try {
|
||||||
final realExpiration = await kv.value.routingContext
|
final realExpiration = await dhtctx.watchDHTValues(
|
||||||
.watchDHTValues(kv.key,
|
openedRecordKey,
|
||||||
subkeys: ws.subkeys?.toList(),
|
subkeys: watchState.subkeys?.toList(),
|
||||||
count: ws.count,
|
count: watchState.count,
|
||||||
expiration: ws.expiration);
|
expiration: watchState.expiration);
|
||||||
kv.value.needsWatchStateUpdate = false;
|
openedRecordInfo.shared.needsWatchStateUpdate = false;
|
||||||
|
|
||||||
// Update watch state with real expiration
|
// Update watch states with real expiration
|
||||||
kv.value.watchState = WatchState(
|
_updateWatchExpirations(
|
||||||
subkeys: ws.subkeys,
|
openedRecordInfo.records, realExpiration);
|
||||||
expiration: ws.expiration,
|
|
||||||
count: ws.count,
|
|
||||||
realExpiration: realExpiration);
|
|
||||||
} on VeilidAPIException {
|
} on VeilidAPIException {
|
||||||
// Failed to cancel DHT watch, try again next tick
|
// Failed to cancel DHT watch, try again next tick
|
||||||
}
|
}
|
||||||
kv.value.inWatchStateUpdate = false;
|
openedRecordInfo.shared.inWatchStateUpdate = false;
|
||||||
}());
|
}());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -86,16 +86,12 @@ class DHTShortArray {
|
|||||||
final schema = DHTSchema.smpl(
|
final schema = DHTSchema.smpl(
|
||||||
oCnt: 0,
|
oCnt: 0,
|
||||||
members: [DHTSchemaMember(mKey: smplWriter.key, mCnt: stride + 1)]);
|
members: [DHTSchemaMember(mKey: smplWriter.key, mCnt: stride + 1)]);
|
||||||
final dhtCreateRecord = await pool.create(
|
dhtRecord = await pool.create(
|
||||||
parent: parent,
|
parent: parent,
|
||||||
routingContext: routingContext,
|
routingContext: routingContext,
|
||||||
schema: schema,
|
schema: schema,
|
||||||
crypto: crypto,
|
crypto: crypto,
|
||||||
writer: smplWriter);
|
writer: smplWriter);
|
||||||
// Reopen with SMPL writer
|
|
||||||
await dhtCreateRecord.close();
|
|
||||||
dhtRecord = await pool.openWrite(dhtCreateRecord.key, smplWriter,
|
|
||||||
parent: parent, routingContext: routingContext, crypto: crypto);
|
|
||||||
} else {
|
} else {
|
||||||
final schema = DHTSchema.dflt(oCnt: stride + 1);
|
final schema = DHTSchema.dflt(oCnt: stride + 1);
|
||||||
dhtRecord = await pool.create(
|
dhtRecord = await pool.create(
|
||||||
|
Loading…
Reference in New Issue
Block a user