import 'package:protobuf/protobuf.dart'; import 'package:veilid/veilid.dart'; import 'dart:typed_data'; import 'tools.dart'; class DHTRecord { final VeilidRoutingContext _dhtctx; final DHTRecordDescriptor _recordDescriptor; final int _defaultSubkey; final KeyPair? _writer; late final DHTRecordEncryption _encryption; static Future create(VeilidRoutingContext dhtctx, {DHTSchema schema = const DHTSchema.dflt(oCnt: 1), int defaultSubkey = 0, DHTRecordEncryptionFactory crypto = DHTRecordEncryption.private}) async { DHTRecordDescriptor recordDescriptor = await dhtctx.createDHTRecord(schema); final rec = DHTRecord( dhtctx: dhtctx, recordDescriptor: recordDescriptor, defaultSubkey: defaultSubkey, writer: recordDescriptor.ownerKeyPair()); rec._encryption = crypto(rec); return rec; } static Future openRead( VeilidRoutingContext dhtctx, TypedKey recordKey, {int defaultSubkey = 0, DHTRecordEncryptionFactory crypto = DHTRecordEncryption.private}) async { DHTRecordDescriptor recordDescriptor = await dhtctx.openDHTRecord(recordKey, null); final rec = DHTRecord( dhtctx: dhtctx, recordDescriptor: recordDescriptor, defaultSubkey: defaultSubkey, writer: null); rec._encryption = crypto(rec); return rec; } static Future openWrite( VeilidRoutingContext dhtctx, TypedKey recordKey, KeyPair writer, {int defaultSubkey = 0, DHTRecordEncryptionFactory crypto = DHTRecordEncryption.private}) async { DHTRecordDescriptor recordDescriptor = await dhtctx.openDHTRecord(recordKey, writer); final rec = DHTRecord( dhtctx: dhtctx, recordDescriptor: recordDescriptor, defaultSubkey: defaultSubkey, writer: writer); rec._encryption = crypto(rec); return rec; } DHTRecord( {required VeilidRoutingContext dhtctx, required DHTRecordDescriptor recordDescriptor, int defaultSubkey = 0, KeyPair? writer}) : _dhtctx = dhtctx, _recordDescriptor = recordDescriptor, _defaultSubkey = defaultSubkey, _writer = writer; int subkeyOrDefault(int subkey) => (subkey == -1) ? _defaultSubkey : subkey; TypedKey key() { return _recordDescriptor.key; } PublicKey owner() { return _recordDescriptor.owner; } KeyPair? ownerKeyPair() { return _recordDescriptor.ownerKeyPair(); } KeyPair? writer() { return _writer; } Future close() async { await _dhtctx.closeDHTRecord(_recordDescriptor.key); } Future delete() async { await _dhtctx.deleteDHTRecord(_recordDescriptor.key); } Future scope(Future Function(DHTRecord) scopeFunction) async { try { return await scopeFunction(this); } finally { close(); } } Future deleteScope(Future Function(DHTRecord) scopeFunction) async { try { return await scopeFunction(this); } catch (_) { delete(); rethrow; } finally { close(); } } Future get({int subkey = -1, bool forceRefresh = false}) async { subkey = subkeyOrDefault(subkey); ValueData? valueData = await _dhtctx.getDHTValue(_recordDescriptor.key, subkey, false); if (valueData == null) { return null; } return _encryption.decrypt(valueData.data, subkey); } Future getJson(T Function(Map) fromJson, {int subkey = -1, bool forceRefresh = false}) async { final data = await get(subkey: subkey, forceRefresh: forceRefresh); if (data == null) { return null; } return jsonDecodeBytes(fromJson, data); } Future eventualWriteBytes(Uint8List newValue, {int subkey = -1}) async { subkey = subkeyOrDefault(subkey); newValue = await _encryption.encrypt(newValue, subkey); // Get existing identity key ValueData? valueData; do { // Ensure it exists already if (valueData == null) { throw const FormatException("value does not exist"); } // Set the new data valueData = await _dhtctx.setDHTValue(_recordDescriptor.key, subkey, newValue); // Repeat if newer data on the network was found } while (valueData != null); } Future eventualUpdateBytes( Future Function(Uint8List oldValue) update, {int subkey = -1}) async { subkey = subkeyOrDefault(subkey); // Get existing identity key ValueData? valueData = await _dhtctx.getDHTValue(_recordDescriptor.key, subkey, false); do { // Ensure it exists already if (valueData == null) { throw const FormatException("value does not exist"); } // Update the data final oldData = await _encryption.decrypt(valueData.data, subkey); final updatedData = await update(oldData); final newData = await _encryption.encrypt(updatedData, subkey); // Set it back valueData = await _dhtctx.setDHTValue(_recordDescriptor.key, subkey, newData); // Repeat if newer data on the network was found } while (valueData != null); } Future eventualWriteJson(T newValue, {int subkey = -1}) { return eventualWriteBytes(jsonEncodeBytes(newValue), subkey: subkey); } Future eventualWriteProtobuf(T newValue, {int subkey = -1}) { return eventualWriteBytes(newValue.writeToBuffer(), subkey: subkey); } Future eventualUpdateJson( T Function(Map) fromJson, Future Function(T) update, {int subkey = -1}) { return eventualUpdateBytes(jsonUpdate(fromJson, update), subkey: subkey); } Future eventualUpdateProtobuf( T Function(List) fromBuffer, Future Function(T) update, {int subkey = -1}) { return eventualUpdateBytes(protobufUpdate(fromBuffer, update), subkey: subkey); } }