mirror of
https://gitlab.com/veilid/veilidchat.git
synced 2025-01-12 16:19:27 -05:00
dht log passes tests
This commit is contained in:
parent
8cd73b2844
commit
cf837e2176
@ -12,6 +12,8 @@ import 'test_dht_record_pool.dart';
|
|||||||
import 'test_dht_short_array.dart';
|
import 'test_dht_short_array.dart';
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
|
final startTime = DateTime.now();
|
||||||
|
|
||||||
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
|
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
|
||||||
final veilidFixture =
|
final veilidFixture =
|
||||||
DefaultVeilidFixture(programName: 'veilid_support integration test');
|
DefaultVeilidFixture(programName: 'veilid_support integration test');
|
||||||
@ -39,26 +41,26 @@ void main() {
|
|||||||
|
|
||||||
test('create pool', testDHTRecordPoolCreate);
|
test('create pool', testDHTRecordPoolCreate);
|
||||||
|
|
||||||
// group('DHTRecordPool Tests', () {
|
group('DHTRecordPool Tests', () {
|
||||||
// setUpAll(dhtRecordPoolFixture.setUp);
|
setUpAll(dhtRecordPoolFixture.setUp);
|
||||||
// tearDownAll(dhtRecordPoolFixture.tearDown);
|
tearDownAll(dhtRecordPoolFixture.tearDown);
|
||||||
|
|
||||||
// test('create/delete record', testDHTRecordCreateDelete);
|
test('create/delete record', testDHTRecordCreateDelete);
|
||||||
// test('record scopes', testDHTRecordScopes);
|
test('record scopes', testDHTRecordScopes);
|
||||||
// test('create/delete deep record', testDHTRecordDeepCreateDelete);
|
test('create/delete deep record', testDHTRecordDeepCreateDelete);
|
||||||
// });
|
});
|
||||||
|
|
||||||
// group('DHTShortArray Tests', () {
|
group('DHTShortArray Tests', () {
|
||||||
// setUpAll(dhtRecordPoolFixture.setUp);
|
setUpAll(dhtRecordPoolFixture.setUp);
|
||||||
// tearDownAll(dhtRecordPoolFixture.tearDown);
|
tearDownAll(dhtRecordPoolFixture.tearDown);
|
||||||
|
|
||||||
// for (final stride in [256, 16 /*64, 32, 16, 8, 4, 2, 1 */]) {
|
for (final stride in [256, 16 /*64, 32, 16, 8, 4, 2, 1 */]) {
|
||||||
// test('create shortarray stride=$stride',
|
test('create shortarray stride=$stride',
|
||||||
// makeTestDHTShortArrayCreateDelete(stride: stride));
|
makeTestDHTShortArrayCreateDelete(stride: stride));
|
||||||
// test('add shortarray stride=$stride',
|
test('add shortarray stride=$stride',
|
||||||
// makeTestDHTShortArrayAdd(stride: 256));
|
makeTestDHTShortArrayAdd(stride: 256));
|
||||||
// }
|
}
|
||||||
// });
|
});
|
||||||
|
|
||||||
group('DHTLog Tests', () {
|
group('DHTLog Tests', () {
|
||||||
setUpAll(dhtRecordPoolFixture.setUp);
|
setUpAll(dhtRecordPoolFixture.setUp);
|
||||||
@ -75,4 +77,7 @@ void main() {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
final endTime = DateTime.now();
|
||||||
|
print('Duration: ${endTime.difference(startTime)}');
|
||||||
}
|
}
|
||||||
|
@ -42,7 +42,7 @@ Future<void> Function() makeTestDHTLogCreateDelete({required int stride}) =>
|
|||||||
// Operate should still succeed because things aren't closed
|
// Operate should still succeed because things aren't closed
|
||||||
expect(await dlog.operate((r) async => r.length), isZero);
|
expect(await dlog.operate((r) async => r.length), isZero);
|
||||||
await dlog.close();
|
await dlog.close();
|
||||||
await dlog.close();
|
await expectLater(() async => dlog.close(), throwsA(isA<StateError>()));
|
||||||
// Operate should fail
|
// Operate should fail
|
||||||
await expectLater(() async => dlog.operate((r) async => r.length),
|
await expectLater(() async => dlog.operate((r) async => r.length),
|
||||||
throwsA(isA<StateError>()));
|
throwsA(isA<StateError>()));
|
||||||
@ -51,8 +51,6 @@ Future<void> Function() makeTestDHTLogCreateDelete({required int stride}) =>
|
|||||||
|
|
||||||
Future<void> Function() makeTestDHTLogAddTruncate({required int stride}) =>
|
Future<void> Function() makeTestDHTLogAddTruncate({required int stride}) =>
|
||||||
() async {
|
() async {
|
||||||
final startTime = DateTime.now();
|
|
||||||
|
|
||||||
final dlog = await DHTLog.create(
|
final dlog = await DHTLog.create(
|
||||||
debugName: 'log_add 1 stride $stride', stride: stride);
|
debugName: 'log_add 1 stride $stride', stride: stride);
|
||||||
|
|
||||||
@ -121,10 +119,8 @@ Future<void> Function() makeTestDHTLogAddTruncate({required int stride}) =>
|
|||||||
final dataset8 = await dlog.operate((r) async => r.getItemRange(0));
|
final dataset8 = await dlog.operate((r) async => r.getItemRange(0));
|
||||||
expect(dataset8, isEmpty);
|
expect(dataset8, isEmpty);
|
||||||
}
|
}
|
||||||
|
print('delete and close\n');
|
||||||
|
|
||||||
await dlog.delete();
|
await dlog.delete();
|
||||||
await dlog.close();
|
await dlog.close();
|
||||||
|
|
||||||
final endTime = DateTime.now();
|
|
||||||
print('Duration: ${endTime.difference(startTime)}');
|
|
||||||
};
|
};
|
||||||
|
@ -48,7 +48,7 @@ Future<void> testDHTRecordCreateDelete() async {
|
|||||||
// Set should succeed still
|
// Set should succeed still
|
||||||
await rec3.tryWriteBytes(utf8.encode('test'));
|
await rec3.tryWriteBytes(utf8.encode('test'));
|
||||||
await rec3.close();
|
await rec3.close();
|
||||||
await rec3.close();
|
await expectLater(() async => rec3.close(), throwsA(isA<StateError>()));
|
||||||
// Set should fail
|
// Set should fail
|
||||||
await expectLater(() async => rec3.tryWriteBytes(utf8.encode('test')),
|
await expectLater(() async => rec3.tryWriteBytes(utf8.encode('test')),
|
||||||
throwsA(isA<VeilidAPIException>()));
|
throwsA(isA<VeilidAPIException>()));
|
||||||
@ -84,7 +84,7 @@ Future<void> testDHTRecordScopes() async {
|
|||||||
} on Exception {
|
} on Exception {
|
||||||
assert(false, 'should not throw');
|
assert(false, 'should not throw');
|
||||||
}
|
}
|
||||||
await rec2.close();
|
await expectLater(() async => rec2.close(), throwsA(isA<StateError>()));
|
||||||
await pool.deleteRecord(rec2.key);
|
await pool.deleteRecord(rec2.key);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -115,6 +115,7 @@ Future<void> testDHTRecordGetSet() async {
|
|||||||
final val = await rec.get();
|
final val = await rec.get();
|
||||||
await pool.deleteRecord(rec.key);
|
await pool.deleteRecord(rec.key);
|
||||||
expect(val, isNull);
|
expect(val, isNull);
|
||||||
|
await rec.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Test set then get
|
// Test set then get
|
||||||
@ -125,6 +126,7 @@ Future<void> testDHTRecordGetSet() async {
|
|||||||
// Invalid subkey should throw
|
// Invalid subkey should throw
|
||||||
await expectLater(
|
await expectLater(
|
||||||
() async => rec2.get(subkey: 1), throwsA(isA<VeilidAPIException>()));
|
() async => rec2.get(subkey: 1), throwsA(isA<VeilidAPIException>()));
|
||||||
|
await rec2.close();
|
||||||
await pool.deleteRecord(rec2.key);
|
await pool.deleteRecord(rec2.key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -43,7 +43,7 @@ Future<void> Function() makeTestDHTShortArrayCreateDelete(
|
|||||||
// Operate should still succeed because things aren't closed
|
// Operate should still succeed because things aren't closed
|
||||||
expect(await arr.operate((r) async => r.length), isZero);
|
expect(await arr.operate((r) async => r.length), isZero);
|
||||||
await arr.close();
|
await arr.close();
|
||||||
await arr.close();
|
await expectLater(() async => arr.close(), throwsA(isA<StateError>()));
|
||||||
// Operate should fail
|
// Operate should fail
|
||||||
await expectLater(() async => arr.operate((r) async => r.length),
|
await expectLater(() async => arr.operate((r) async => r.length),
|
||||||
throwsA(isA<StateError>()));
|
throwsA(isA<StateError>()));
|
||||||
@ -52,8 +52,6 @@ Future<void> Function() makeTestDHTShortArrayCreateDelete(
|
|||||||
|
|
||||||
Future<void> Function() makeTestDHTShortArrayAdd({required int stride}) =>
|
Future<void> Function() makeTestDHTShortArrayAdd({required int stride}) =>
|
||||||
() async {
|
() async {
|
||||||
final startTime = DateTime.now();
|
|
||||||
|
|
||||||
final arr = await DHTShortArray.create(
|
final arr = await DHTShortArray.create(
|
||||||
debugName: 'sa_add 1 stride $stride', stride: stride);
|
debugName: 'sa_add 1 stride $stride', stride: stride);
|
||||||
|
|
||||||
@ -131,7 +129,4 @@ Future<void> Function() makeTestDHTShortArrayAdd({required int stride}) =>
|
|||||||
|
|
||||||
await arr.delete();
|
await arr.delete();
|
||||||
await arr.close();
|
await arr.close();
|
||||||
|
|
||||||
final endTime = DateTime.now();
|
|
||||||
print('Duration: ${endTime.difference(startTime)}');
|
|
||||||
};
|
};
|
||||||
|
@ -42,11 +42,13 @@ class DHTLogUpdate extends Equatable {
|
|||||||
/// * The head and tail position of the log
|
/// * The head and tail position of the log
|
||||||
/// - subkeyIdx = pos / recordsPerSubkey
|
/// - subkeyIdx = pos / recordsPerSubkey
|
||||||
/// - recordIdx = pos % recordsPerSubkey
|
/// - recordIdx = pos % recordsPerSubkey
|
||||||
class DHTLog implements DHTOpenable {
|
class DHTLog implements DHTOpenable<DHTLog> {
|
||||||
////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////
|
||||||
// Constructors
|
// Constructors
|
||||||
|
|
||||||
DHTLog._({required _DHTLogSpine spine}) : _spine = spine {
|
DHTLog._({required _DHTLogSpine spine})
|
||||||
|
: _spine = spine,
|
||||||
|
_openCount = 1 {
|
||||||
_spine.onUpdatedSpine = (update) {
|
_spine.onUpdatedSpine = (update) {
|
||||||
_watchController?.sink.add(update);
|
_watchController?.sink.add(update);
|
||||||
};
|
};
|
||||||
@ -162,18 +164,29 @@ class DHTLog implements DHTOpenable {
|
|||||||
|
|
||||||
/// Check if the DHTLog is open
|
/// Check if the DHTLog is open
|
||||||
@override
|
@override
|
||||||
bool get isOpen => _spine.isOpen;
|
bool get isOpen => _openCount > 0;
|
||||||
|
|
||||||
|
/// Add a reference to this log
|
||||||
|
@override
|
||||||
|
Future<DHTLog> ref() async => _mutex.protect(() async {
|
||||||
|
_openCount++;
|
||||||
|
return this;
|
||||||
|
});
|
||||||
|
|
||||||
/// Free all resources for the DHTLog
|
/// Free all resources for the DHTLog
|
||||||
@override
|
@override
|
||||||
Future<void> close() async {
|
Future<void> close() async => _mutex.protect(() async {
|
||||||
if (!isOpen) {
|
if (_openCount == 0) {
|
||||||
return;
|
throw StateError('already closed');
|
||||||
}
|
}
|
||||||
await _watchController?.close();
|
_openCount--;
|
||||||
_watchController = null;
|
if (_openCount != 0) {
|
||||||
await _spine.close();
|
return;
|
||||||
}
|
}
|
||||||
|
await _watchController?.close();
|
||||||
|
_watchController = null;
|
||||||
|
await _spine.close();
|
||||||
|
});
|
||||||
|
|
||||||
/// Free all resources for the DHTLog and delete it from the DHT
|
/// Free all resources for the DHTLog and delete it from the DHT
|
||||||
/// Will wait until the short array is closed to delete it
|
/// Will wait until the short array is closed to delete it
|
||||||
@ -284,6 +297,10 @@ class DHTLog implements DHTOpenable {
|
|||||||
// Internal representation refreshed from spine record
|
// Internal representation refreshed from spine record
|
||||||
final _DHTLogSpine _spine;
|
final _DHTLogSpine _spine;
|
||||||
|
|
||||||
|
// Openable
|
||||||
|
int _openCount;
|
||||||
|
final _mutex = Mutex();
|
||||||
|
|
||||||
// Watch mutex to ensure we keep the representation valid
|
// Watch mutex to ensure we keep the representation valid
|
||||||
final Mutex _listenMutex = Mutex();
|
final Mutex _listenMutex = Mutex();
|
||||||
// Stream of external changes
|
// Stream of external changes
|
||||||
|
@ -17,16 +17,16 @@ class _DHTLogAppend extends _DHTLogRead implements DHTAppendTruncateRandomRead {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Write item to the segment
|
// Write item to the segment
|
||||||
return lookup.shortArray.operateWrite((write) async {
|
return lookup.shortArray.scope((sa) => sa.operateWrite((write) async {
|
||||||
// If this a new segment, then clear it in case we have wrapped around
|
// If this a new segment, then clear it in case we have wrapped around
|
||||||
if (lookup.pos == 0) {
|
if (lookup.pos == 0) {
|
||||||
await write.clear();
|
await write.clear();
|
||||||
} else if (lookup.pos != write.length) {
|
} else if (lookup.pos != write.length) {
|
||||||
// We should always be appending at the length
|
// We should always be appending at the length
|
||||||
throw StateError('appending should be at the end');
|
throw StateError('appending should be at the end');
|
||||||
}
|
}
|
||||||
return write.tryAddItem(value);
|
return write.tryAddItem(value);
|
||||||
});
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -45,16 +45,19 @@ class _DHTLogAppend extends _DHTLogRead implements DHTAppendTruncateRandomRead {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final sacount = min(remaining, DHTShortArray.maxElements - lookup.pos);
|
final sacount = min(remaining, DHTShortArray.maxElements - lookup.pos);
|
||||||
final success = await lookup.shortArray.operateWrite((write) async {
|
final success =
|
||||||
// If this a new segment, then clear it in case we have wrapped around
|
await lookup.shortArray.scope((sa) => sa.operateWrite((write) async {
|
||||||
if (lookup.pos == 0) {
|
// If this a new segment, then clear it in
|
||||||
await write.clear();
|
// case we have wrapped around
|
||||||
} else if (lookup.pos != write.length) {
|
if (lookup.pos == 0) {
|
||||||
// We should always be appending at the length
|
await write.clear();
|
||||||
throw StateError('appending should be at the end');
|
} else if (lookup.pos != write.length) {
|
||||||
}
|
// We should always be appending at the length
|
||||||
return write.tryAddItems(values.sublist(valueIdx, valueIdx + sacount));
|
throw StateError('appending should be at the end');
|
||||||
});
|
}
|
||||||
|
return write
|
||||||
|
.tryAddItems(values.sublist(valueIdx, valueIdx + sacount));
|
||||||
|
}));
|
||||||
if (!success) {
|
if (!success) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -19,8 +19,8 @@ class _DHTLogRead implements DHTRandomRead {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return lookup.shortArray.operate(
|
return lookup.shortArray.scope((sa) => sa.operate(
|
||||||
(read) => read.getItem(lookup.pos, forceRefresh: forceRefresh));
|
(read) => read.getItem(lookup.pos, forceRefresh: forceRefresh)));
|
||||||
}
|
}
|
||||||
|
|
||||||
(int, int) _clampStartLen(int start, int? len) {
|
(int, int) _clampStartLen(int start, int? len) {
|
||||||
@ -71,22 +71,22 @@ class _DHTLogRead implements DHTRandomRead {
|
|||||||
|
|
||||||
// Check each segment for offline positions
|
// Check each segment for offline positions
|
||||||
var foundOffline = false;
|
var foundOffline = false;
|
||||||
await lookup.shortArray.operate((read) async {
|
await lookup.shortArray.scope((sa) => sa.operate((read) async {
|
||||||
final segmentOffline = await read.getOfflinePositions();
|
final segmentOffline = await read.getOfflinePositions();
|
||||||
|
|
||||||
// For each shortarray segment go through their segment positions
|
// For each shortarray segment go through their segment positions
|
||||||
// in reverse order and see if they are offline
|
// in reverse order and see if they are offline
|
||||||
for (var segmentPos = lookup.pos;
|
for (var segmentPos = lookup.pos;
|
||||||
segmentPos >= 0 && pos >= 0;
|
segmentPos >= 0 && pos >= 0;
|
||||||
segmentPos--, pos--) {
|
segmentPos--, pos--) {
|
||||||
// If the position in the segment is offline, then
|
// If the position in the segment is offline, then
|
||||||
// mark the position in the log as offline
|
// mark the position in the log as offline
|
||||||
if (segmentOffline.contains(segmentPos)) {
|
if (segmentOffline.contains(segmentPos)) {
|
||||||
positionOffline.add(pos);
|
positionOffline.add(pos);
|
||||||
foundOffline = true;
|
foundOffline = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
}));
|
||||||
|
|
||||||
// If we found nothing offline in this segment then we can stop
|
// If we found nothing offline in this segment then we can stop
|
||||||
if (!foundOffline) {
|
if (!foundOffline) {
|
||||||
|
@ -15,6 +15,13 @@ class _DHTLogSegmentLookup extends Equatable {
|
|||||||
List<Object?> get props => [subkey, segment];
|
List<Object?> get props => [subkey, segment];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _SubkeyData {
|
||||||
|
_SubkeyData({required this.subkey, required this.data});
|
||||||
|
int subkey;
|
||||||
|
Uint8List data;
|
||||||
|
bool changed = false;
|
||||||
|
}
|
||||||
|
|
||||||
class _DHTLogSpine {
|
class _DHTLogSpine {
|
||||||
_DHTLogSpine._(
|
_DHTLogSpine._(
|
||||||
{required DHTRecord spineRecord,
|
{required DHTRecord spineRecord,
|
||||||
@ -47,7 +54,7 @@ class _DHTLogSpine {
|
|||||||
static Future<_DHTLogSpine> load({required DHTRecord spineRecord}) async {
|
static Future<_DHTLogSpine> load({required DHTRecord spineRecord}) async {
|
||||||
// Get an updated spine head record copy if one exists
|
// Get an updated spine head record copy if one exists
|
||||||
final spineHead = await spineRecord.getProtobuf(proto.DHTLog.fromBuffer,
|
final spineHead = await spineRecord.getProtobuf(proto.DHTLog.fromBuffer,
|
||||||
subkey: 0, refreshMode: DHTRecordRefreshMode.refresh);
|
subkey: 0, refreshMode: DHTRecordRefreshMode.network);
|
||||||
if (spineHead == null) {
|
if (spineHead == null) {
|
||||||
throw StateError('spine head missing during refresh');
|
throw StateError('spine head missing during refresh');
|
||||||
}
|
}
|
||||||
@ -234,7 +241,7 @@ class _DHTLogSpine {
|
|||||||
segmentKeyBytes);
|
segmentKeyBytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<DHTShortArray> _getOrCreateSegmentInner(int segmentNumber) async {
|
Future<DHTShortArray> _openOrCreateSegmentInner(int segmentNumber) async {
|
||||||
assert(_spineMutex.isLocked, 'should be in mutex here');
|
assert(_spineMutex.isLocked, 'should be in mutex here');
|
||||||
assert(_spineRecord.writer != null, 'should be writable');
|
assert(_spineRecord.writer != null, 'should be writable');
|
||||||
|
|
||||||
@ -292,7 +299,7 @@ class _DHTLogSpine {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<DHTShortArray?> _getSegmentInner(int segmentNumber) async {
|
Future<DHTShortArray?> _openSegmentInner(int segmentNumber) async {
|
||||||
assert(_spineMutex.isLocked, 'should be in mutex here');
|
assert(_spineMutex.isLocked, 'should be in mutex here');
|
||||||
|
|
||||||
// Lookup what subkey and segment subrange has this position's segment
|
// Lookup what subkey and segment subrange has this position's segment
|
||||||
@ -321,7 +328,7 @@ class _DHTLogSpine {
|
|||||||
return segmentRec;
|
return segmentRec;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<DHTShortArray> getOrCreateSegment(int segmentNumber) async {
|
Future<DHTShortArray> _openOrCreateSegment(int segmentNumber) async {
|
||||||
assert(_spineMutex.isLocked, 'should be in mutex here');
|
assert(_spineMutex.isLocked, 'should be in mutex here');
|
||||||
|
|
||||||
// See if we already have this in the cache
|
// See if we already have this in the cache
|
||||||
@ -331,21 +338,22 @@ class _DHTLogSpine {
|
|||||||
final x = _spineCache.removeAt(i);
|
final x = _spineCache.removeAt(i);
|
||||||
_spineCache.add(x);
|
_spineCache.add(x);
|
||||||
// Return the shortarray for this position
|
// Return the shortarray for this position
|
||||||
return x.$2;
|
return x.$2.ref();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we don't have it in the cache, get/create it and then cache it
|
// If we don't have it in the cache, get/create it and then cache a ref
|
||||||
final segment = await _getOrCreateSegmentInner(segmentNumber);
|
final segment = await _openOrCreateSegmentInner(segmentNumber);
|
||||||
_spineCache.add((segmentNumber, segment));
|
_spineCache.add((segmentNumber, await segment.ref()));
|
||||||
if (_spineCache.length > _spineCacheLength) {
|
if (_spineCache.length > _spineCacheLength) {
|
||||||
// Trim the LRU cache
|
// Trim the LRU cache
|
||||||
_spineCache.removeAt(0);
|
final (_, sa) = _spineCache.removeAt(0);
|
||||||
|
await sa.close();
|
||||||
}
|
}
|
||||||
return segment;
|
return segment;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<DHTShortArray?> getSegment(int segmentNumber) async {
|
Future<DHTShortArray?> _openSegment(int segmentNumber) async {
|
||||||
assert(_spineMutex.isLocked, 'should be in mutex here');
|
assert(_spineMutex.isLocked, 'should be in mutex here');
|
||||||
|
|
||||||
// See if we already have this in the cache
|
// See if we already have this in the cache
|
||||||
@ -355,19 +363,20 @@ class _DHTLogSpine {
|
|||||||
final x = _spineCache.removeAt(i);
|
final x = _spineCache.removeAt(i);
|
||||||
_spineCache.add(x);
|
_spineCache.add(x);
|
||||||
// Return the shortarray for this position
|
// Return the shortarray for this position
|
||||||
return x.$2;
|
return x.$2.ref();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we don't have it in the cache, get it and then cache it
|
// If we don't have it in the cache, get it and then cache it
|
||||||
final segment = await _getSegmentInner(segmentNumber);
|
final segment = await _openSegmentInner(segmentNumber);
|
||||||
if (segment == null) {
|
if (segment == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
_spineCache.add((segmentNumber, segment));
|
_spineCache.add((segmentNumber, await segment.ref()));
|
||||||
if (_spineCache.length > _spineCacheLength) {
|
if (_spineCache.length > _spineCacheLength) {
|
||||||
// Trim the LRU cache
|
// Trim the LRU cache
|
||||||
_spineCache.removeAt(0);
|
final (_, sa) = _spineCache.removeAt(0);
|
||||||
|
await sa.close();
|
||||||
}
|
}
|
||||||
return segment;
|
return segment;
|
||||||
}
|
}
|
||||||
@ -409,8 +418,8 @@ class _DHTLogSpine {
|
|||||||
|
|
||||||
// Get the segment shortArray
|
// Get the segment shortArray
|
||||||
final shortArray = (_spineRecord.writer == null)
|
final shortArray = (_spineRecord.writer == null)
|
||||||
? await getSegment(segmentNumber)
|
? await _openSegment(segmentNumber)
|
||||||
: await getOrCreateSegment(segmentNumber);
|
: await _openOrCreateSegment(segmentNumber);
|
||||||
if (shortArray == null) {
|
if (shortArray == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -442,12 +451,16 @@ class _DHTLogSpine {
|
|||||||
throw StateError('ring buffer underflow');
|
throw StateError('ring buffer underflow');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
final oldHead = _head;
|
||||||
_head = (_head + count) % _positionLimit;
|
_head = (_head + count) % _positionLimit;
|
||||||
await _purgeUnusedSegments();
|
final newHead = _head;
|
||||||
|
await _purgeSegments(oldHead, newHead);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _deleteSegmentsContiguous(int start, int end) async {
|
Future<void> _deleteSegmentsContiguous(int start, int end) async {
|
||||||
assert(_spineMutex.isLocked, 'should be in mutex here');
|
assert(_spineMutex.isLocked, 'should be in mutex here');
|
||||||
|
DHTRecordPool.instance
|
||||||
|
.log('_deleteSegmentsContiguous: start=$start, end=$end');
|
||||||
|
|
||||||
final startSegmentNumber = start ~/ DHTShortArray.maxElements;
|
final startSegmentNumber = start ~/ DHTShortArray.maxElements;
|
||||||
final startSegmentPos = start % DHTShortArray.maxElements;
|
final startSegmentPos = start % DHTShortArray.maxElements;
|
||||||
@ -460,8 +473,7 @@ class _DHTLogSpine {
|
|||||||
final lastDeleteSegment =
|
final lastDeleteSegment =
|
||||||
(endSegmentPos == 0) ? endSegmentNumber - 1 : endSegmentNumber - 2;
|
(endSegmentPos == 0) ? endSegmentNumber - 1 : endSegmentNumber - 2;
|
||||||
|
|
||||||
int? lastSubkey;
|
_SubkeyData? lastSubkeyData;
|
||||||
Uint8List? subkeyData;
|
|
||||||
for (var segmentNumber = firstDeleteSegment;
|
for (var segmentNumber = firstDeleteSegment;
|
||||||
segmentNumber <= lastDeleteSegment;
|
segmentNumber <= lastDeleteSegment;
|
||||||
segmentNumber++) {
|
segmentNumber++) {
|
||||||
@ -471,44 +483,48 @@ class _DHTLogSpine {
|
|||||||
final subkey = l.subkey;
|
final subkey = l.subkey;
|
||||||
final segment = l.segment;
|
final segment = l.segment;
|
||||||
|
|
||||||
if (lastSubkey != subkey) {
|
if (subkey != lastSubkeyData?.subkey) {
|
||||||
// Flush subkey writes
|
// Flush subkey writes
|
||||||
if (lastSubkey != null) {
|
if (lastSubkeyData != null && lastSubkeyData.changed) {
|
||||||
await _spineRecord.eventualWriteBytes(subkeyData!,
|
await _spineRecord.eventualWriteBytes(lastSubkeyData.data,
|
||||||
subkey: lastSubkey);
|
subkey: lastSubkeyData.subkey);
|
||||||
}
|
}
|
||||||
|
|
||||||
xxx debug this, it takes forever
|
// Get next subkey if available locally
|
||||||
|
final data = await _spineRecord.get(
|
||||||
// Get next subkey
|
subkey: subkey, refreshMode: DHTRecordRefreshMode.local);
|
||||||
subkeyData = await _spineRecord.get(subkey: subkey);
|
if (data != null) {
|
||||||
if (subkeyData != null) {
|
lastSubkeyData = _SubkeyData(subkey: subkey, data: data);
|
||||||
lastSubkey = subkey;
|
|
||||||
} else {
|
} else {
|
||||||
lastSubkey = null;
|
lastSubkeyData = null;
|
||||||
|
// If the subkey was not available locally we can go to the
|
||||||
|
// last segment number at the end of this subkey
|
||||||
|
segmentNumber = ((subkey + 1) * DHTLog.segmentsPerSubkey) - 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (subkeyData != null) {
|
if (lastSubkeyData != null) {
|
||||||
final segmentKey = _getSegmentKey(subkeyData, segment);
|
final segmentKey = _getSegmentKey(lastSubkeyData.data, segment);
|
||||||
if (segmentKey != null) {
|
if (segmentKey != null) {
|
||||||
await DHTRecordPool.instance.deleteRecord(segmentKey);
|
await DHTRecordPool.instance.deleteRecord(segmentKey);
|
||||||
_setSegmentKey(subkeyData, segment, null);
|
_setSegmentKey(lastSubkeyData.data, segment, null);
|
||||||
|
lastSubkeyData.changed = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Flush subkey writes
|
// Flush subkey writes
|
||||||
if (lastSubkey != null) {
|
if (lastSubkeyData != null) {
|
||||||
await _spineRecord.eventualWriteBytes(subkeyData!, subkey: lastSubkey);
|
await _spineRecord.eventualWriteBytes(lastSubkeyData.data,
|
||||||
|
subkey: lastSubkeyData.subkey);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _purgeUnusedSegments() async {
|
Future<void> _purgeSegments(int from, int to) async {
|
||||||
assert(_spineMutex.isLocked, 'should be in mutex here');
|
assert(_spineMutex.isLocked, 'should be in mutex here');
|
||||||
if (_head < _tail) {
|
if (from < to) {
|
||||||
await _deleteSegmentsContiguous(0, _head);
|
await _deleteSegmentsContiguous(from, to);
|
||||||
await _deleteSegmentsContiguous(_tail, _positionLimit);
|
} else if (from > to) {
|
||||||
} else if (_head > _tail) {
|
await _deleteSegmentsContiguous(from, _positionLimit);
|
||||||
await _deleteSegmentsContiguous(_tail, _head);
|
await _deleteSegmentsContiguous(0, to);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,7 +39,7 @@ class DefaultDHTRecordCubit<T> extends DHTRecordCubit<T> {
|
|||||||
final firstSubkey = subkeys.firstOrNull!.low;
|
final firstSubkey = subkeys.firstOrNull!.low;
|
||||||
if (firstSubkey != defaultSubkey || updatedata == null) {
|
if (firstSubkey != defaultSubkey || updatedata == null) {
|
||||||
final maybeData =
|
final maybeData =
|
||||||
await record.get(refreshMode: DHTRecordRefreshMode.refresh);
|
await record.get(refreshMode: DHTRecordRefreshMode.network);
|
||||||
if (maybeData == null) {
|
if (maybeData == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -16,19 +16,27 @@ class DHTRecordWatchChange extends Equatable {
|
|||||||
/// Refresh mode for DHT record 'get'
|
/// Refresh mode for DHT record 'get'
|
||||||
enum DHTRecordRefreshMode {
|
enum DHTRecordRefreshMode {
|
||||||
/// Return existing subkey values if they exist locally already
|
/// Return existing subkey values if they exist locally already
|
||||||
existing,
|
/// And then check the network for a newer value
|
||||||
|
/// This is the default refresh mode
|
||||||
|
cached,
|
||||||
|
|
||||||
|
/// Return existing subkey values only if they exist locally already
|
||||||
|
local,
|
||||||
|
|
||||||
/// Always check the network for a newer subkey value
|
/// Always check the network for a newer subkey value
|
||||||
refresh,
|
network,
|
||||||
|
|
||||||
/// Always check the network for a newer subkey value but only
|
/// Always check the network for a newer subkey value but only
|
||||||
/// return that value if its sequence number is newer than the local value
|
/// return that value if its sequence number is newer than the local value
|
||||||
refreshOnlyUpdates,
|
update;
|
||||||
|
|
||||||
|
bool get _forceRefresh => this == network || this == update;
|
||||||
|
bool get _inspectLocal => this == local || this == update;
|
||||||
}
|
}
|
||||||
|
|
||||||
/////////////////////////////////////////////////
|
/////////////////////////////////////////////////
|
||||||
|
|
||||||
class DHTRecord implements DHTOpenable {
|
class DHTRecord implements DHTOpenable<DHTRecord> {
|
||||||
DHTRecord._(
|
DHTRecord._(
|
||||||
{required VeilidRoutingContext routingContext,
|
{required VeilidRoutingContext routingContext,
|
||||||
required SharedDHTRecordData sharedDHTRecordData,
|
required SharedDHTRecordData sharedDHTRecordData,
|
||||||
@ -40,7 +48,7 @@ class DHTRecord implements DHTOpenable {
|
|||||||
_routingContext = routingContext,
|
_routingContext = routingContext,
|
||||||
_defaultSubkey = defaultSubkey,
|
_defaultSubkey = defaultSubkey,
|
||||||
_writer = writer,
|
_writer = writer,
|
||||||
_open = true,
|
_openCount = 1,
|
||||||
_sharedDHTRecordData = sharedDHTRecordData;
|
_sharedDHTRecordData = sharedDHTRecordData;
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////
|
||||||
@ -48,25 +56,37 @@ class DHTRecord implements DHTOpenable {
|
|||||||
|
|
||||||
/// Check if the DHTRecord is open
|
/// Check if the DHTRecord is open
|
||||||
@override
|
@override
|
||||||
bool get isOpen => _open;
|
bool get isOpen => _openCount > 0;
|
||||||
|
|
||||||
|
/// Add a reference to this DHTRecord
|
||||||
|
@override
|
||||||
|
Future<DHTRecord> ref() async => _mutex.protect(() async {
|
||||||
|
_openCount++;
|
||||||
|
return this;
|
||||||
|
});
|
||||||
|
|
||||||
/// Free all resources for the DHTRecord
|
/// Free all resources for the DHTRecord
|
||||||
@override
|
@override
|
||||||
Future<void> close() async {
|
Future<void> close() async => _mutex.protect(() async {
|
||||||
if (!_open) {
|
if (_openCount == 0) {
|
||||||
return;
|
throw StateError('already closed');
|
||||||
}
|
}
|
||||||
await watchController?.close();
|
_openCount--;
|
||||||
await DHTRecordPool.instance._recordClosed(this);
|
if (_openCount != 0) {
|
||||||
_open = false;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await _watchController?.close();
|
||||||
|
_watchController = null;
|
||||||
|
await DHTRecordPool.instance._recordClosed(this);
|
||||||
|
});
|
||||||
|
|
||||||
/// Free all resources for the DHTRecord and delete it from the DHT
|
/// Free all resources for the DHTRecord and delete it from the DHT
|
||||||
/// Will wait until the record is closed to delete it
|
/// Will wait until the record is closed to delete it
|
||||||
@override
|
@override
|
||||||
Future<void> delete() async {
|
Future<void> delete() async => _mutex.protect(() async {
|
||||||
await DHTRecordPool.instance.deleteRecord(key);
|
await DHTRecordPool.instance.deleteRecord(key);
|
||||||
}
|
});
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////
|
||||||
// Public API
|
// Public API
|
||||||
@ -95,25 +115,37 @@ class DHTRecord implements DHTOpenable {
|
|||||||
Future<Uint8List?> get(
|
Future<Uint8List?> get(
|
||||||
{int subkey = -1,
|
{int subkey = -1,
|
||||||
DHTRecordCrypto? crypto,
|
DHTRecordCrypto? crypto,
|
||||||
DHTRecordRefreshMode refreshMode = DHTRecordRefreshMode.existing,
|
DHTRecordRefreshMode refreshMode = DHTRecordRefreshMode.cached,
|
||||||
Output<int>? outSeqNum}) async {
|
Output<int>? outSeqNum}) async {
|
||||||
subkey = subkeyOrDefault(subkey);
|
subkey = subkeyOrDefault(subkey);
|
||||||
|
|
||||||
|
// Get the last sequence number if we need it
|
||||||
|
final lastSeq =
|
||||||
|
refreshMode._inspectLocal ? await _localSubkeySeq(subkey) : null;
|
||||||
|
|
||||||
|
// See if we only ever want the locally stored value
|
||||||
|
if (refreshMode == DHTRecordRefreshMode.local && lastSeq == null) {
|
||||||
|
// If it's not available locally already just return null now
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
final valueData = await _routingContext.getDHTValue(key, subkey,
|
final valueData = await _routingContext.getDHTValue(key, subkey,
|
||||||
forceRefresh: refreshMode != DHTRecordRefreshMode.existing);
|
forceRefresh: refreshMode._forceRefresh);
|
||||||
if (valueData == null) {
|
if (valueData == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
final lastSeq = _sharedDHTRecordData.subkeySeqCache[subkey];
|
// See if this get resulted in a newer sequence number
|
||||||
if (refreshMode == DHTRecordRefreshMode.refreshOnlyUpdates &&
|
if (refreshMode == DHTRecordRefreshMode.update &&
|
||||||
lastSeq != null &&
|
lastSeq != null &&
|
||||||
valueData.seq <= lastSeq) {
|
valueData.seq <= lastSeq) {
|
||||||
|
// If we're only returning updates then punt now
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
// If we're returning a value, decrypt it
|
||||||
final out = (crypto ?? _crypto).decrypt(valueData.data, subkey);
|
final out = (crypto ?? _crypto).decrypt(valueData.data, subkey);
|
||||||
if (outSeqNum != null) {
|
if (outSeqNum != null) {
|
||||||
outSeqNum.save(valueData.seq);
|
outSeqNum.save(valueData.seq);
|
||||||
}
|
}
|
||||||
_sharedDHTRecordData.subkeySeqCache[subkey] = valueData.seq;
|
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -128,7 +160,7 @@ class DHTRecord implements DHTOpenable {
|
|||||||
Future<T?> getJson<T>(T Function(dynamic) fromJson,
|
Future<T?> getJson<T>(T Function(dynamic) fromJson,
|
||||||
{int subkey = -1,
|
{int subkey = -1,
|
||||||
DHTRecordCrypto? crypto,
|
DHTRecordCrypto? crypto,
|
||||||
DHTRecordRefreshMode refreshMode = DHTRecordRefreshMode.existing,
|
DHTRecordRefreshMode refreshMode = DHTRecordRefreshMode.cached,
|
||||||
Output<int>? outSeqNum}) async {
|
Output<int>? outSeqNum}) async {
|
||||||
final data = await get(
|
final data = await get(
|
||||||
subkey: subkey,
|
subkey: subkey,
|
||||||
@ -154,7 +186,7 @@ class DHTRecord implements DHTOpenable {
|
|||||||
T Function(List<int> i) fromBuffer,
|
T Function(List<int> i) fromBuffer,
|
||||||
{int subkey = -1,
|
{int subkey = -1,
|
||||||
DHTRecordCrypto? crypto,
|
DHTRecordCrypto? crypto,
|
||||||
DHTRecordRefreshMode refreshMode = DHTRecordRefreshMode.existing,
|
DHTRecordRefreshMode refreshMode = DHTRecordRefreshMode.cached,
|
||||||
Output<int>? outSeqNum}) async {
|
Output<int>? outSeqNum}) async {
|
||||||
final data = await get(
|
final data = await get(
|
||||||
subkey: subkey,
|
subkey: subkey,
|
||||||
@ -176,7 +208,7 @@ class DHTRecord implements DHTOpenable {
|
|||||||
KeyPair? writer,
|
KeyPair? writer,
|
||||||
Output<int>? outSeqNum}) async {
|
Output<int>? outSeqNum}) async {
|
||||||
subkey = subkeyOrDefault(subkey);
|
subkey = subkeyOrDefault(subkey);
|
||||||
final lastSeq = _sharedDHTRecordData.subkeySeqCache[subkey];
|
final lastSeq = await _localSubkeySeq(subkey);
|
||||||
final encryptedNewValue =
|
final encryptedNewValue =
|
||||||
await (crypto ?? _crypto).encrypt(newValue, subkey);
|
await (crypto ?? _crypto).encrypt(newValue, subkey);
|
||||||
|
|
||||||
@ -198,7 +230,6 @@ class DHTRecord implements DHTOpenable {
|
|||||||
if (isUpdated && outSeqNum != null) {
|
if (isUpdated && outSeqNum != null) {
|
||||||
outSeqNum.save(newValueData.seq);
|
outSeqNum.save(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
|
||||||
@ -228,7 +259,7 @@ class DHTRecord implements DHTOpenable {
|
|||||||
KeyPair? writer,
|
KeyPair? writer,
|
||||||
Output<int>? outSeqNum}) async {
|
Output<int>? outSeqNum}) async {
|
||||||
subkey = subkeyOrDefault(subkey);
|
subkey = subkeyOrDefault(subkey);
|
||||||
final lastSeq = _sharedDHTRecordData.subkeySeqCache[subkey];
|
final lastSeq = await _localSubkeySeq(subkey);
|
||||||
final encryptedNewValue =
|
final encryptedNewValue =
|
||||||
await (crypto ?? _crypto).encrypt(newValue, subkey);
|
await (crypto ?? _crypto).encrypt(newValue, subkey);
|
||||||
|
|
||||||
@ -254,7 +285,6 @@ class DHTRecord implements DHTOpenable {
|
|||||||
if (outSeqNum != null) {
|
if (outSeqNum != null) {
|
||||||
outSeqNum.save(newValueData.seq);
|
outSeqNum.save(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,
|
||||||
@ -402,13 +432,13 @@ class DHTRecord implements DHTOpenable {
|
|||||||
DHTRecordCrypto? crypto,
|
DHTRecordCrypto? crypto,
|
||||||
}) async {
|
}) async {
|
||||||
// Set up watch requirements
|
// Set up watch requirements
|
||||||
watchController ??=
|
_watchController ??=
|
||||||
StreamController<DHTRecordWatchChange>.broadcast(onCancel: () {
|
StreamController<DHTRecordWatchChange>.broadcast(onCancel: () {
|
||||||
// If there are no more listeners then we can get rid of the controller
|
// If there are no more listeners then we can get rid of the controller
|
||||||
watchController = null;
|
_watchController = null;
|
||||||
});
|
});
|
||||||
|
|
||||||
return watchController!.stream.listen(
|
return _watchController!.stream.listen(
|
||||||
(change) {
|
(change) {
|
||||||
if (change.local && !localChanges) {
|
if (change.local && !localChanges) {
|
||||||
return;
|
return;
|
||||||
@ -431,8 +461,8 @@ class DHTRecord implements DHTOpenable {
|
|||||||
},
|
},
|
||||||
cancelOnError: true,
|
cancelOnError: true,
|
||||||
onError: (e) async {
|
onError: (e) async {
|
||||||
await watchController!.close();
|
await _watchController!.close();
|
||||||
watchController = null;
|
_watchController = null;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -455,6 +485,14 @@ class DHTRecord implements DHTOpenable {
|
|||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
Future<int?> _localSubkeySeq(int subkey) async {
|
||||||
|
final rr = await _routingContext.inspectDHTRecord(
|
||||||
|
key,
|
||||||
|
subkeys: [ValueSubkeyRange.single(subkey)],
|
||||||
|
);
|
||||||
|
return rr.localSeqs.firstOrNull ?? 0xFFFFFFFF;
|
||||||
|
}
|
||||||
|
|
||||||
void _addValueChange(
|
void _addValueChange(
|
||||||
{required bool local,
|
{required bool local,
|
||||||
required Uint8List? data,
|
required Uint8List? data,
|
||||||
@ -464,7 +502,7 @@ class DHTRecord implements DHTOpenable {
|
|||||||
final watchedSubkeys = ws.subkeys;
|
final watchedSubkeys = ws.subkeys;
|
||||||
if (watchedSubkeys == null) {
|
if (watchedSubkeys == null) {
|
||||||
// Report all subkeys
|
// Report all subkeys
|
||||||
watchController?.add(
|
_watchController?.add(
|
||||||
DHTRecordWatchChange(local: local, data: data, subkeys: subkeys));
|
DHTRecordWatchChange(local: local, data: data, subkeys: subkeys));
|
||||||
} else {
|
} else {
|
||||||
// Only some subkeys are being watched, see if the reported update
|
// Only some subkeys are being watched, see if the reported update
|
||||||
@ -479,7 +517,7 @@ class DHTRecord implements DHTOpenable {
|
|||||||
overlappedFirstSubkey == updateFirstSubkey ? data : null;
|
overlappedFirstSubkey == updateFirstSubkey ? data : null;
|
||||||
|
|
||||||
// Report only watched subkeys
|
// Report only watched subkeys
|
||||||
watchController?.add(DHTRecordWatchChange(
|
_watchController?.add(DHTRecordWatchChange(
|
||||||
local: local, data: updatedData, subkeys: overlappedSubkeys));
|
local: local, data: updatedData, subkeys: overlappedSubkeys));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -504,10 +542,9 @@ class DHTRecord implements DHTOpenable {
|
|||||||
final KeyPair? _writer;
|
final KeyPair? _writer;
|
||||||
final DHTRecordCrypto _crypto;
|
final DHTRecordCrypto _crypto;
|
||||||
final String debugName;
|
final String debugName;
|
||||||
|
final _mutex = Mutex();
|
||||||
bool _open;
|
int _openCount;
|
||||||
@internal
|
StreamController<DHTRecordWatchChange>? _watchController;
|
||||||
StreamController<DHTRecordWatchChange>? watchController;
|
|
||||||
@internal
|
@internal
|
||||||
WatchState? watchState;
|
WatchState? watchState;
|
||||||
}
|
}
|
||||||
|
@ -93,7 +93,7 @@ class DHTRecordCubit<T> extends Cubit<AsyncValue<T>> {
|
|||||||
for (final skr in subkeys) {
|
for (final skr in subkeys) {
|
||||||
for (var sk = skr.low; sk <= skr.high; sk++) {
|
for (var sk = skr.low; sk <= skr.high; sk++) {
|
||||||
final data = await _record.get(
|
final data = await _record.get(
|
||||||
subkey: sk, refreshMode: DHTRecordRefreshMode.refreshOnlyUpdates);
|
subkey: sk, refreshMode: DHTRecordRefreshMode.update);
|
||||||
if (data != null) {
|
if (data != null) {
|
||||||
final newState = await _stateFunction(_record, updateSubkeys, data);
|
final newState = await _stateFunction(_record, updateSubkeys, data);
|
||||||
if (newState != null) {
|
if (newState != null) {
|
||||||
|
@ -88,7 +88,6 @@ class SharedDHTRecordData {
|
|||||||
DHTRecordDescriptor recordDescriptor;
|
DHTRecordDescriptor recordDescriptor;
|
||||||
KeyPair? defaultWriter;
|
KeyPair? defaultWriter;
|
||||||
VeilidRoutingContext defaultRoutingContext;
|
VeilidRoutingContext defaultRoutingContext;
|
||||||
Map<int, int> subkeySeqCache = {};
|
|
||||||
bool needsWatchStateUpdate = false;
|
bool needsWatchStateUpdate = false;
|
||||||
WatchState? unionWatchState;
|
WatchState? unionWatchState;
|
||||||
}
|
}
|
||||||
|
@ -13,12 +13,13 @@ part 'dht_short_array_write.dart';
|
|||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
class DHTShortArray implements DHTOpenable {
|
class DHTShortArray implements DHTOpenable<DHTShortArray> {
|
||||||
////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////
|
||||||
// Constructors
|
// Constructors
|
||||||
|
|
||||||
DHTShortArray._({required DHTRecord headRecord})
|
DHTShortArray._({required DHTRecord headRecord})
|
||||||
: _head = _DHTShortArrayHead(headRecord: headRecord) {
|
: _head = _DHTShortArrayHead(headRecord: headRecord),
|
||||||
|
_openCount = 1 {
|
||||||
_head.onUpdatedHead = () {
|
_head.onUpdatedHead = () {
|
||||||
_watchController?.sink.add(null);
|
_watchController?.sink.add(null);
|
||||||
};
|
};
|
||||||
@ -139,18 +140,30 @@ class DHTShortArray implements DHTOpenable {
|
|||||||
|
|
||||||
/// Check if the shortarray is open
|
/// Check if the shortarray is open
|
||||||
@override
|
@override
|
||||||
bool get isOpen => _head.isOpen;
|
bool get isOpen => _openCount > 0;
|
||||||
|
|
||||||
|
/// Add a reference to this shortarray
|
||||||
|
@override
|
||||||
|
Future<DHTShortArray> ref() async => _mutex.protect(() async {
|
||||||
|
_openCount++;
|
||||||
|
return this;
|
||||||
|
});
|
||||||
|
|
||||||
/// Free all resources for the DHTShortArray
|
/// Free all resources for the DHTShortArray
|
||||||
@override
|
@override
|
||||||
Future<void> close() async {
|
Future<void> close() async => _mutex.protect(() async {
|
||||||
if (!isOpen) {
|
if (_openCount == 0) {
|
||||||
return;
|
throw StateError('already closed');
|
||||||
}
|
}
|
||||||
await _watchController?.close();
|
_openCount--;
|
||||||
_watchController = null;
|
if (_openCount != 0) {
|
||||||
await _head.close();
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await _watchController?.close();
|
||||||
|
_watchController = null;
|
||||||
|
await _head.close();
|
||||||
|
});
|
||||||
|
|
||||||
/// Free all resources for the DHTShortArray and delete it from the DHT
|
/// Free all resources for the DHTShortArray and delete it from the DHT
|
||||||
/// Will wait until the short array is closed to delete it
|
/// Will wait until the short array is closed to delete it
|
||||||
@ -255,6 +268,10 @@ class DHTShortArray implements DHTOpenable {
|
|||||||
// Internal representation refreshed from head record
|
// Internal representation refreshed from head record
|
||||||
final _DHTShortArrayHead _head;
|
final _DHTShortArrayHead _head;
|
||||||
|
|
||||||
|
// Openable
|
||||||
|
int _openCount;
|
||||||
|
final _mutex = Mutex();
|
||||||
|
|
||||||
// Watch mutex to ensure we keep the representation valid
|
// Watch mutex to ensure we keep the representation valid
|
||||||
final Mutex _listenMutex = Mutex();
|
final Mutex _listenMutex = Mutex();
|
||||||
// Stream of external changes
|
// Stream of external changes
|
||||||
|
@ -248,7 +248,7 @@ class _DHTShortArrayHead {
|
|||||||
Future<void> _loadHead() async {
|
Future<void> _loadHead() async {
|
||||||
// Get an updated head record copy if one exists
|
// Get an updated head record copy if one exists
|
||||||
final head = await _headRecord.getProtobuf(proto.DHTShortArray.fromBuffer,
|
final head = await _headRecord.getProtobuf(proto.DHTShortArray.fromBuffer,
|
||||||
subkey: 0, refreshMode: DHTRecordRefreshMode.refresh);
|
subkey: 0, refreshMode: DHTRecordRefreshMode.network);
|
||||||
if (head == null) {
|
if (head == null) {
|
||||||
throw StateError('shortarray head missing during refresh');
|
throw StateError('shortarray head missing during refresh');
|
||||||
}
|
}
|
||||||
|
@ -22,8 +22,8 @@ class _DHTShortArrayRead implements DHTRandomRead {
|
|||||||
final out = lookup.record.get(
|
final out = lookup.record.get(
|
||||||
subkey: lookup.recordSubkey,
|
subkey: lookup.recordSubkey,
|
||||||
refreshMode: refresh
|
refreshMode: refresh
|
||||||
? DHTRecordRefreshMode.refresh
|
? DHTRecordRefreshMode.network
|
||||||
: DHTRecordRefreshMode.existing,
|
: DHTRecordRefreshMode.cached,
|
||||||
outSeqNum: outSeqNum);
|
outSeqNum: outSeqNum);
|
||||||
if (outSeqNum.value != null) {
|
if (outSeqNum.value != null) {
|
||||||
_head.updatePositionSeq(pos, false, outSeqNum.value!);
|
_head.updatePositionSeq(pos, false, outSeqNum.value!);
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
|
||||||
abstract class DHTOpenable {
|
abstract class DHTOpenable<C> {
|
||||||
bool get isOpen;
|
bool get isOpen;
|
||||||
|
Future<C> ref();
|
||||||
Future<void> close();
|
Future<void> close();
|
||||||
Future<void> delete();
|
Future<void> delete();
|
||||||
}
|
}
|
||||||
|
|
||||||
extension DHTOpenableExt<D extends DHTOpenable> on D {
|
extension DHTOpenableExt<D extends DHTOpenable<D>> on D {
|
||||||
/// Runs a closure that guarantees the DHTOpenable
|
/// Runs a closure that guarantees the DHTOpenable
|
||||||
/// will be closed upon exit, even if an uncaught exception is thrown
|
/// will be closed upon exit, even if an uncaught exception is thrown
|
||||||
Future<T> scope<T>(Future<T> Function(D) scopeFunction) async {
|
Future<T> scope<T>(Future<T> Function(D) scopeFunction) async {
|
||||||
|
@ -301,7 +301,7 @@ Future<IdentityMaster> openIdentityMaster(
|
|||||||
'IdentityMaster::openIdentityMaster::IdentityMasterRecord'))
|
'IdentityMaster::openIdentityMaster::IdentityMasterRecord'))
|
||||||
.deleteScope((masterRec) async {
|
.deleteScope((masterRec) async {
|
||||||
final identityMaster = (await masterRec.getJson(IdentityMaster.fromJson,
|
final identityMaster = (await masterRec.getJson(IdentityMaster.fromJson,
|
||||||
refreshMode: DHTRecordRefreshMode.refresh))!;
|
refreshMode: DHTRecordRefreshMode.network))!;
|
||||||
|
|
||||||
// Validate IdentityMaster
|
// Validate IdentityMaster
|
||||||
final masterRecordKey = masterRec.key;
|
final masterRecordKey = masterRec.key;
|
||||||
|
Loading…
Reference in New Issue
Block a user