Line data Source code
1 : /*
2 : * Famedly Matrix SDK
3 : * Copyright (C) 2019, 2020, 2021 Famedly GmbH
4 : *
5 : * This program is free software: you can redistribute it and/or modify
6 : * it under the terms of the GNU Affero General Public License as
7 : * published by the Free Software Foundation, either version 3 of the
8 : * License, or (at your option) any later version.
9 : *
10 : * This program is distributed in the hope that it will be useful,
11 : * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 : * GNU Affero General Public License for more details.
14 : *
15 : * You should have received a copy of the GNU Affero General Public License
16 : * along with this program. If not, see <https://www.gnu.org/licenses/>.
17 : */
18 :
19 : import 'dart:async';
20 : import 'dart:convert';
21 : import 'dart:math';
22 : import 'dart:typed_data';
23 :
24 : import 'package:hive/hive.dart';
25 :
26 : import 'package:matrix/encryption/utils/olm_session.dart';
27 : import 'package:matrix/encryption/utils/outbound_group_session.dart';
28 : import 'package:matrix/encryption/utils/ssss_cache.dart';
29 : import 'package:matrix/encryption/utils/stored_inbound_group_session.dart';
30 : import 'package:matrix/matrix.dart';
31 : import 'package:matrix/src/database/zone_transaction_mixin.dart';
32 : import 'package:matrix/src/utils/copy_map.dart';
33 : import 'package:matrix/src/utils/queued_to_device_event.dart';
34 : import 'package:matrix/src/utils/run_benchmarked.dart';
35 :
36 : /// This is a basic database for the Matrix SDK using the hive store. You need
37 : /// to make sure that you perform `Hive.init()` or `Hive.flutterInit()` before
38 : /// you use this.
39 : ///
40 : /// This database does not support file caching!
41 : @Deprecated(
42 : 'Use [MatrixSdkDatabase] instead. Don\'t forget to properly migrate!',
43 : )
44 : class FamedlySdkHiveDatabase extends DatabaseApi with ZoneTransactionMixin {
45 : static const int version = 6;
46 : final String name;
47 : late Box _clientBox;
48 : late Box _accountDataBox;
49 : late Box _roomsBox;
50 : late Box _toDeviceQueueBox;
51 :
52 : /// Key is a tuple as MultiKey(roomId, type) where stateKey can be
53 : /// an empty string.
54 : late LazyBox _roomStateBox;
55 :
56 : /// Key is a tuple as MultiKey(roomId, userId)
57 : late LazyBox _roomMembersBox;
58 :
59 : /// Key is a tuple as MultiKey(roomId, type)
60 : late LazyBox _roomAccountDataBox;
61 : late LazyBox _inboundGroupSessionsBox;
62 : late LazyBox _outboundGroupSessionsBox;
63 : late LazyBox _olmSessionsBox;
64 :
65 : /// Key is a tuple as MultiKey(userId, deviceId)
66 : late LazyBox _userDeviceKeysBox;
67 :
68 : /// Key is the user ID as a String
69 : late LazyBox _userDeviceKeysOutdatedBox;
70 :
71 : /// Key is a tuple as MultiKey(userId, publicKey)
72 : late LazyBox _userCrossSigningKeysBox;
73 : late LazyBox _ssssCacheBox;
74 : late LazyBox _presencesBox;
75 :
76 : /// Key is a tuple as Multikey(roomId, fragmentId) while the default
77 : /// fragmentId is an empty String
78 : late LazyBox _timelineFragmentsBox;
79 :
80 : /// Key is a tuple as MultiKey(roomId, eventId)
81 : late LazyBox _eventsBox;
82 :
83 : /// Key is a tuple as MultiKey(userId, deviceId)
84 : late LazyBox _seenDeviceIdsBox;
85 :
86 : late LazyBox _seenDeviceKeysBox;
87 :
88 3 : String get _clientBoxName => '$name.box.client';
89 :
90 3 : String get _accountDataBoxName => '$name.box.account_data';
91 :
92 3 : String get _roomsBoxName => '$name.box.rooms';
93 :
94 3 : String get _toDeviceQueueBoxName => '$name.box.to_device_queue';
95 :
96 3 : String get _roomStateBoxName => '$name.box.room_states';
97 :
98 3 : String get _roomMembersBoxName => '$name.box.room_members';
99 :
100 3 : String get _roomAccountDataBoxName => '$name.box.room_account_data';
101 :
102 3 : String get _inboundGroupSessionsBoxName => '$name.box.inbound_group_session';
103 :
104 1 : String get _outboundGroupSessionsBoxName =>
105 2 : '$name.box.outbound_group_session';
106 :
107 3 : String get _olmSessionsBoxName => '$name.box.olm_session';
108 :
109 3 : String get _userDeviceKeysBoxName => '$name.box.user_device_keys';
110 :
111 1 : String get _userDeviceKeysOutdatedBoxName =>
112 2 : '$name.box.user_device_keys_outdated';
113 :
114 3 : String get _userCrossSigningKeysBoxName => '$name.box.cross_signing_keys';
115 :
116 3 : String get _ssssCacheBoxName => '$name.box.ssss_cache';
117 :
118 3 : String get _presencesBoxName => '$name.box.presences';
119 :
120 3 : String get _timelineFragmentsBoxName => '$name.box.timeline_fragments';
121 :
122 3 : String get _eventsBoxName => '$name.box.events';
123 :
124 3 : String get _seenDeviceIdsBoxName => '$name.box.seen_device_ids';
125 :
126 3 : String get _seenDeviceKeysBoxName => '$name.box.seen_device_keys';
127 :
128 : final HiveCipher? encryptionCipher;
129 :
130 1 : FamedlySdkHiveDatabase(this.name, {this.encryptionCipher});
131 :
132 0 : @override
133 : int get maxFileSize => 0;
134 :
135 1 : Future<void> _actionOnAllBoxes(Future<void> Function(BoxBase box) action) =>
136 2 : Future.wait([
137 2 : action(_clientBox),
138 2 : action(_accountDataBox),
139 2 : action(_roomsBox),
140 2 : action(_roomStateBox),
141 2 : action(_roomMembersBox),
142 2 : action(_toDeviceQueueBox),
143 2 : action(_roomAccountDataBox),
144 2 : action(_inboundGroupSessionsBox),
145 2 : action(_outboundGroupSessionsBox),
146 2 : action(_olmSessionsBox),
147 2 : action(_userDeviceKeysBox),
148 2 : action(_userDeviceKeysOutdatedBox),
149 2 : action(_userCrossSigningKeysBox),
150 2 : action(_ssssCacheBox),
151 2 : action(_presencesBox),
152 2 : action(_timelineFragmentsBox),
153 2 : action(_eventsBox),
154 2 : action(_seenDeviceIdsBox),
155 2 : action(_seenDeviceKeysBox),
156 : ]);
157 :
158 1 : Future<void> open() async {
159 3 : _clientBox = await Hive.openBox(
160 1 : _clientBoxName,
161 1 : encryptionCipher: encryptionCipher,
162 : );
163 3 : _accountDataBox = await Hive.openBox(
164 1 : _accountDataBoxName,
165 1 : encryptionCipher: encryptionCipher,
166 : );
167 3 : _roomsBox = await Hive.openBox(
168 1 : _roomsBoxName,
169 1 : encryptionCipher: encryptionCipher,
170 : );
171 3 : _roomStateBox = await Hive.openLazyBox(
172 1 : _roomStateBoxName,
173 1 : encryptionCipher: encryptionCipher,
174 : );
175 3 : _roomMembersBox = await Hive.openLazyBox(
176 1 : _roomMembersBoxName,
177 1 : encryptionCipher: encryptionCipher,
178 : );
179 3 : _toDeviceQueueBox = await Hive.openBox(
180 1 : _toDeviceQueueBoxName,
181 1 : encryptionCipher: encryptionCipher,
182 : );
183 3 : _roomAccountDataBox = await Hive.openLazyBox(
184 1 : _roomAccountDataBoxName,
185 1 : encryptionCipher: encryptionCipher,
186 : );
187 3 : _inboundGroupSessionsBox = await Hive.openLazyBox(
188 1 : _inboundGroupSessionsBoxName,
189 1 : encryptionCipher: encryptionCipher,
190 : );
191 3 : _outboundGroupSessionsBox = await Hive.openLazyBox(
192 1 : _outboundGroupSessionsBoxName,
193 1 : encryptionCipher: encryptionCipher,
194 : );
195 3 : _olmSessionsBox = await Hive.openLazyBox(
196 1 : _olmSessionsBoxName,
197 1 : encryptionCipher: encryptionCipher,
198 : );
199 3 : _userDeviceKeysBox = await Hive.openLazyBox(
200 1 : _userDeviceKeysBoxName,
201 1 : encryptionCipher: encryptionCipher,
202 : );
203 3 : _userDeviceKeysOutdatedBox = await Hive.openLazyBox(
204 1 : _userDeviceKeysOutdatedBoxName,
205 1 : encryptionCipher: encryptionCipher,
206 : );
207 3 : _userCrossSigningKeysBox = await Hive.openLazyBox(
208 1 : _userCrossSigningKeysBoxName,
209 1 : encryptionCipher: encryptionCipher,
210 : );
211 3 : _ssssCacheBox = await Hive.openLazyBox(
212 1 : _ssssCacheBoxName,
213 1 : encryptionCipher: encryptionCipher,
214 : );
215 3 : _presencesBox = await Hive.openLazyBox(
216 1 : _presencesBoxName,
217 1 : encryptionCipher: encryptionCipher,
218 : );
219 3 : _timelineFragmentsBox = await Hive.openLazyBox(
220 1 : _timelineFragmentsBoxName,
221 1 : encryptionCipher: encryptionCipher,
222 : );
223 3 : _eventsBox = await Hive.openLazyBox(
224 1 : _eventsBoxName,
225 1 : encryptionCipher: encryptionCipher,
226 : );
227 3 : _seenDeviceIdsBox = await Hive.openLazyBox(
228 1 : _seenDeviceIdsBoxName,
229 1 : encryptionCipher: encryptionCipher,
230 : );
231 3 : _seenDeviceKeysBox = await Hive.openLazyBox(
232 1 : _seenDeviceKeysBoxName,
233 1 : encryptionCipher: encryptionCipher,
234 : );
235 :
236 : // Check version and check if we need a migration
237 2 : final currentVersion = (await _clientBox.get('version') as int?);
238 : if (currentVersion == null) {
239 2 : await _clientBox.put('version', version);
240 0 : } else if (currentVersion != version) {
241 0 : await _migrateFromVersion(currentVersion);
242 : }
243 :
244 : return;
245 : }
246 :
247 0 : Future<void> _migrateFromVersion(int currentVersion) async {
248 0 : Logs().i('Migrate Hive database from version $currentVersion to $version');
249 0 : if (version == 5) {
250 0 : for (final key in _userDeviceKeysBox.keys) {
251 : try {
252 0 : final raw = await _userDeviceKeysBox.get(key) as Map;
253 0 : if (!raw.containsKey('keys')) continue;
254 0 : final deviceKeys = DeviceKeys.fromJson(
255 0 : convertToJson(raw),
256 0 : Client(''),
257 : );
258 0 : await addSeenDeviceId(
259 0 : deviceKeys.userId,
260 0 : deviceKeys.deviceId!,
261 0 : deviceKeys.curve25519Key! + deviceKeys.ed25519Key!,
262 : );
263 0 : await addSeenPublicKey(deviceKeys.ed25519Key!, deviceKeys.deviceId!);
264 0 : await addSeenPublicKey(
265 0 : deviceKeys.curve25519Key!,
266 0 : deviceKeys.deviceId!,
267 : );
268 : } catch (e) {
269 0 : Logs().w('Can not migrate device $key', e);
270 : }
271 : }
272 : }
273 0 : await clearCache();
274 0 : await _clientBox.put('version', version);
275 : }
276 :
277 1 : @override
278 : Future<void> clear() async {
279 2 : Logs().i('Clear and close hive database...');
280 2 : await _actionOnAllBoxes((box) async {
281 : try {
282 2 : await box.deleteAll(box.keys);
283 1 : await box.close();
284 : } catch (e) {
285 0 : Logs().v('Unable to clear box ${box.name}', e);
286 0 : await box.deleteFromDisk();
287 : }
288 : });
289 : return;
290 : }
291 :
292 1 : @override
293 : Future<void> clearCache() async {
294 4 : await _roomsBox.deleteAll(_roomsBox.keys);
295 4 : await _accountDataBox.deleteAll(_accountDataBox.keys);
296 4 : await _roomAccountDataBox.deleteAll(_roomAccountDataBox.keys);
297 4 : await _roomStateBox.deleteAll(_roomStateBox.keys);
298 4 : await _roomMembersBox.deleteAll(_roomMembersBox.keys);
299 4 : await _eventsBox.deleteAll(_eventsBox.keys);
300 4 : await _timelineFragmentsBox.deleteAll(_timelineFragmentsBox.keys);
301 4 : await _outboundGroupSessionsBox.deleteAll(_outboundGroupSessionsBox.keys);
302 4 : await _presencesBox.deleteAll(_presencesBox.keys);
303 2 : await _clientBox.delete('prev_batch');
304 : }
305 :
306 1 : @override
307 : Future<void> clearSSSSCache() async {
308 4 : await _ssssCacheBox.deleteAll(_ssssCacheBox.keys);
309 : }
310 :
311 1 : @override
312 3 : Future<void> close() => _actionOnAllBoxes((box) => box.close());
313 :
314 1 : @override
315 : Future<void> deleteFromToDeviceQueue(int id) async {
316 2 : await _toDeviceQueueBox.delete(id);
317 : return;
318 : }
319 :
320 1 : @override
321 : Future<void> deleteOldFiles(int savedAt) async {
322 : return;
323 : }
324 :
325 1 : @override
326 : Future<void> forgetRoom(String roomId) async {
327 4 : await _timelineFragmentsBox.delete(MultiKey(roomId, '').toString());
328 2 : for (final key in _eventsBox.keys) {
329 0 : final multiKey = MultiKey.fromString(key);
330 0 : if (multiKey.parts.first != roomId) continue;
331 0 : await _eventsBox.delete(key);
332 : }
333 2 : for (final key in _roomStateBox.keys) {
334 0 : final multiKey = MultiKey.fromString(key);
335 0 : if (multiKey.parts.first != roomId) continue;
336 0 : await _roomStateBox.delete(key);
337 : }
338 2 : for (final key in _roomMembersBox.keys) {
339 0 : final multiKey = MultiKey.fromString(key);
340 0 : if (multiKey.parts.first != roomId) continue;
341 0 : await _roomMembersBox.delete(key);
342 : }
343 2 : for (final key in _roomAccountDataBox.keys) {
344 0 : final multiKey = MultiKey.fromString(key);
345 0 : if (multiKey.parts.first != roomId) continue;
346 0 : await _roomAccountDataBox.delete(key);
347 : }
348 3 : await _roomsBox.delete(roomId.toHiveKey);
349 : }
350 :
351 1 : @override
352 : Future<Map<String, BasicEvent>> getAccountData() =>
353 1 : runBenchmarked<Map<String, BasicEvent>>(
354 : 'Get all account data from Hive',
355 1 : () async {
356 1 : final accountData = <String, BasicEvent>{};
357 3 : for (final key in _accountDataBox.keys) {
358 2 : final raw = await _accountDataBox.get(key);
359 4 : accountData[key.toString().fromHiveKey] = BasicEvent(
360 2 : type: key.toString().fromHiveKey,
361 1 : content: convertToJson(raw),
362 : );
363 : }
364 : return accountData;
365 : },
366 3 : _accountDataBox.keys.length,
367 : );
368 :
369 1 : @override
370 : Future<Map<String, dynamic>?> getClient(String name) =>
371 2 : runBenchmarked('Get Client from Hive', () async {
372 1 : final map = <String, dynamic>{};
373 3 : for (final key in _clientBox.keys) {
374 1 : if (key == 'version') continue;
375 3 : map[key] = await _clientBox.get(key);
376 : }
377 1 : if (map.isEmpty) return null;
378 : return map;
379 : });
380 :
381 1 : @override
382 : Future<Event?> getEventById(String eventId, Room room) async {
383 5 : final raw = await _eventsBox.get(MultiKey(room.id, eventId).toString());
384 : if (raw == null) return null;
385 2 : return Event.fromJson(convertToJson(raw), room);
386 : }
387 :
388 : /// Loads a whole list of events at once from the store for a specific room
389 1 : Future<List<Event>> _getEventsByIds(List<String> eventIds, Room room) async {
390 1 : final events = await Future.wait(
391 2 : eventIds.map((String eventId) async {
392 : final entry =
393 5 : await _eventsBox.get(MultiKey(room.id, eventId).toString());
394 3 : return entry is Map ? Event.fromJson(convertToJson(entry), room) : null;
395 : }),
396 : );
397 :
398 2 : return events.whereType<Event>().toList();
399 : }
400 :
401 1 : @override
402 : Future<List<Event>> getEventList(
403 : Room room, {
404 : int start = 0,
405 : bool onlySending = false,
406 : int? limit,
407 : }) =>
408 2 : runBenchmarked<List<Event>>('Get event list', () async {
409 : // Get the synced event IDs from the store
410 3 : final timelineKey = MultiKey(room.id, '').toString();
411 1 : final timelineEventIds = List<String>.from(
412 2 : (await _timelineFragmentsBox.get(timelineKey)) ?? [],
413 : );
414 : // Get the local stored SENDING events from the store
415 : late final List sendingEventIds;
416 1 : if (start != 0) {
417 0 : sendingEventIds = [];
418 : } else {
419 3 : final sendingTimelineKey = MultiKey(room.id, 'SENDING').toString();
420 1 : sendingEventIds = List<String>.from(
421 3 : (await _timelineFragmentsBox.get(sendingTimelineKey)) ?? [],
422 : );
423 : }
424 :
425 : // Combine those two lists while respecting the start and limit parameters.
426 1 : final end = min(
427 1 : timelineEventIds.length,
428 2 : start + (limit ?? timelineEventIds.length),
429 : );
430 1 : final eventIds = List<String>.from(
431 1 : [
432 : ...sendingEventIds,
433 2 : ...(start < timelineEventIds.length && !onlySending
434 3 : ? timelineEventIds.getRange(start, end).toList()
435 0 : : []),
436 : ],
437 : );
438 :
439 1 : return await _getEventsByIds(eventIds, room);
440 : });
441 :
442 0 : @override
443 : Future<List<String>> getEventIdList(
444 : Room room, {
445 : int start = 0,
446 : bool includeSending = false,
447 : int? limit,
448 : }) =>
449 0 : runBenchmarked<List<String>>('Get event id list', () async {
450 : // Get the synced event IDs from the store
451 0 : final timelineKey = MultiKey(room.id, '').toString();
452 :
453 0 : final timelineEventIds = List<String>.from(
454 0 : (await _timelineFragmentsBox.get(timelineKey)) ?? [],
455 : );
456 :
457 : // Get the local stored SENDING events from the store
458 : late final List<String> sendingEventIds;
459 : if (!includeSending) {
460 0 : sendingEventIds = [];
461 : } else {
462 0 : final sendingTimelineKey = MultiKey(room.id, 'SENDING').toString();
463 0 : sendingEventIds = List<String>.from(
464 0 : (await _timelineFragmentsBox.get(sendingTimelineKey)) ?? [],
465 : );
466 : }
467 :
468 : // Combine those two lists while respecting the start and limit parameters.
469 0 : final eventIds = sendingEventIds + timelineEventIds;
470 0 : if (limit != null && eventIds.length > limit) {
471 0 : eventIds.removeRange(limit, eventIds.length);
472 : }
473 :
474 : return eventIds;
475 : });
476 :
477 1 : @override
478 : Future<Uint8List?> getFile(Uri mxcUri) async {
479 : return null;
480 : }
481 :
482 1 : @override
483 : Future<bool> deleteFile(Uri mxcUri) async {
484 : return false;
485 : }
486 :
487 1 : @override
488 : Future<StoredInboundGroupSession?> getInboundGroupSession(
489 : String roomId,
490 : String sessionId,
491 : ) async {
492 3 : final raw = await _inboundGroupSessionsBox.get(sessionId.toHiveKey);
493 : if (raw == null) return null;
494 2 : return StoredInboundGroupSession.fromJson(convertToJson(raw));
495 : }
496 :
497 1 : @override
498 : Future<List<StoredInboundGroupSession>>
499 : getInboundGroupSessionsToUpload() async {
500 1 : final sessions = (await Future.wait(
501 3 : _inboundGroupSessionsBox.keys.map(
502 0 : (sessionId) async => await _inboundGroupSessionsBox.get(sessionId),
503 : ),
504 : ))
505 1 : .where((rawSession) => rawSession['uploaded'] == false)
506 1 : .take(500)
507 1 : .map(
508 0 : (json) => StoredInboundGroupSession.fromJson(
509 0 : convertToJson(json),
510 : ),
511 : )
512 1 : .toList();
513 : return sessions;
514 : }
515 :
516 1 : @override
517 : Future<List<String>> getLastSentMessageUserDeviceKey(
518 : String userId,
519 : String deviceId,
520 : ) async {
521 : final raw =
522 4 : await _userDeviceKeysBox.get(MultiKey(userId, deviceId).toString());
523 1 : if (raw == null) return <String>[];
524 0 : return <String>[raw['last_sent_message']];
525 : }
526 :
527 1 : @override
528 : Future<void> storeOlmSession(
529 : String identityKey,
530 : String sessionId,
531 : String pickle,
532 : int lastReceived,
533 : ) async {
534 : final rawSessions =
535 4 : (await _olmSessionsBox.get(identityKey.toHiveKey) as Map?) ?? {};
536 2 : rawSessions[sessionId] = <String, dynamic>{
537 : 'identity_key': identityKey,
538 : 'pickle': pickle,
539 : 'session_id': sessionId,
540 : 'last_received': lastReceived,
541 : };
542 3 : await _olmSessionsBox.put(identityKey.toHiveKey, rawSessions);
543 : return;
544 : }
545 :
546 1 : @override
547 : Future<List<OlmSession>> getOlmSessions(
548 : String identityKey,
549 : String userId,
550 : ) async {
551 : final rawSessions =
552 4 : await _olmSessionsBox.get(identityKey.toHiveKey) as Map? ?? {};
553 :
554 1 : return rawSessions.values
555 4 : .map((json) => OlmSession.fromJson(convertToJson(json), userId))
556 1 : .toList();
557 : }
558 :
559 1 : @override
560 : Future<Map<String, Map>> getAllOlmSessions() async {
561 1 : final backup = Map.fromEntries(
562 1 : await Future.wait(
563 3 : _olmSessionsBox.keys.map(
564 2 : (key) async => MapEntry(
565 : key,
566 2 : await _olmSessionsBox.get(key),
567 : ),
568 : ),
569 : ),
570 : );
571 1 : return backup.cast<String, Map>();
572 : }
573 :
574 1 : @override
575 : Future<List<OlmSession>> getOlmSessionsForDevices(
576 : List<String> identityKeys,
577 : String userId,
578 : ) async {
579 1 : final sessions = await Future.wait(
580 3 : identityKeys.map((identityKey) => getOlmSessions(identityKey, userId)),
581 : );
582 3 : return <OlmSession>[for (final sublist in sessions) ...sublist];
583 : }
584 :
585 1 : @override
586 : Future<OutboundGroupSession?> getOutboundGroupSession(
587 : String roomId,
588 : String userId,
589 : ) async {
590 3 : final raw = await _outboundGroupSessionsBox.get(roomId.toHiveKey);
591 : if (raw == null) return null;
592 2 : return OutboundGroupSession.fromJson(convertToJson(raw), userId);
593 : }
594 :
595 1 : @override
596 : Future<Room?> getSingleRoom(
597 : Client client,
598 : String roomId, {
599 : bool loadImportantStates = true,
600 : }) async {
601 : // Get raw room from database:
602 2 : final roomData = await _roomsBox.get(roomId);
603 : if (roomData == null) return null;
604 2 : final room = Room.fromJson(convertToJson(roomData), client);
605 :
606 : // Get important states:
607 : if (loadImportantStates) {
608 1 : final dbKeys = client.importantStateEvents
609 4 : .map((state) => TupleKey(roomId, state).toString())
610 1 : .toList();
611 1 : final rawStates = await Future.wait(
612 4 : dbKeys.map((key) => _roomStateBox.get(key)),
613 : );
614 2 : for (final rawState in rawStates) {
615 1 : if (rawState == null || rawState[''] == null) continue;
616 4 : room.setState(Event.fromJson(convertToJson(rawState['']), room));
617 : }
618 : }
619 :
620 : return room;
621 : }
622 :
623 1 : @override
624 1 : Future<List<Room>> getRoomList(Client client) => runBenchmarked<List<Room>>(
625 : 'Get room list from hive',
626 1 : () async {
627 1 : final rooms = <String, Room>{};
628 1 : final userID = client.userID;
629 1 : final importantRoomStates = client.importantStateEvents;
630 3 : for (final key in _roomsBox.keys) {
631 : // Get the room
632 2 : final raw = await _roomsBox.get(key);
633 2 : final room = Room.fromJson(convertToJson(raw), client);
634 :
635 : // let's see if we need any m.room.member events
636 : // We always need the member event for ourself
637 0 : final membersToPostload = <String>{if (userID != null) userID};
638 : // If the room is a direct chat, those IDs should be there too
639 1 : if (room.isDirectChat) {
640 0 : membersToPostload.add(room.directChatMatrixID!);
641 : }
642 : // the lastEvent message preview might have an author we need to fetch, if it is a group chat
643 1 : final lastEvent = room.getState(EventTypes.Message);
644 0 : if (lastEvent != null && !room.isDirectChat) {
645 0 : membersToPostload.add(lastEvent.senderId);
646 : }
647 : // if the room has no name and no canonical alias, its name is calculated
648 : // based on the heroes of the room
649 1 : if (room.getState(EventTypes.RoomName) == null &&
650 1 : room.getState(EventTypes.RoomCanonicalAlias) == null) {
651 : // we don't have a name and no canonical alias, so we'll need to
652 : // post-load the heroes
653 3 : membersToPostload.addAll(room.summary.mHeroes ?? []);
654 : }
655 : // Load members
656 1 : for (final userId in membersToPostload) {
657 0 : final state = await _roomMembersBox
658 0 : .get(MultiKey(room.id, userId).toString());
659 : if (state == null) {
660 0 : Logs().w('Unable to post load member $userId');
661 : continue;
662 : }
663 0 : room.setState(
664 0 : room.membership == Membership.invite
665 0 : ? StrippedStateEvent.fromJson(copyMap(raw))
666 0 : : Event.fromJson(convertToJson(state), room),
667 : );
668 : }
669 :
670 : // Get the "important" room states. All other states will be loaded once
671 : // `getUnimportantRoomStates()` is called.
672 2 : for (final type in importantRoomStates) {
673 1 : final states = await _roomStateBox
674 4 : .get(MultiKey(room.id, type).toString()) as Map?;
675 : if (states == null) continue;
676 0 : final stateEvents = states.values
677 0 : .map(
678 0 : (raw) => room.membership == Membership.invite
679 0 : ? StrippedStateEvent.fromJson(copyMap(raw))
680 0 : : Event.fromJson(convertToJson(raw), room),
681 : )
682 0 : .toList();
683 0 : for (final state in stateEvents) {
684 0 : room.setState(state);
685 : }
686 : }
687 :
688 : // Add to the list and continue.
689 2 : rooms[room.id] = room;
690 : }
691 :
692 : // Get the room account data
693 2 : for (final key in _roomAccountDataBox.keys) {
694 0 : final roomId = MultiKey.fromString(key).parts.first;
695 0 : if (rooms.containsKey(roomId)) {
696 0 : final raw = await _roomAccountDataBox.get(key);
697 0 : final basicRoomEvent = BasicRoomEvent.fromJson(
698 0 : convertToJson(raw),
699 : );
700 0 : rooms[roomId]!.roomAccountData[basicRoomEvent.type] =
701 : basicRoomEvent;
702 : } else {
703 0 : Logs().w(
704 0 : 'Found account data for unknown room $roomId. Delete now...',
705 : );
706 0 : await _roomAccountDataBox.delete(key);
707 : }
708 : }
709 :
710 2 : return rooms.values.toList();
711 : },
712 3 : _roomsBox.keys.length,
713 : );
714 :
715 1 : @override
716 : Future<SSSSCache?> getSSSSCache(String type) async {
717 2 : final raw = await _ssssCacheBox.get(type);
718 : if (raw == null) return null;
719 2 : return SSSSCache.fromJson(convertToJson(raw));
720 : }
721 :
722 1 : @override
723 : Future<List<QueuedToDeviceEvent>> getToDeviceEventQueue() async =>
724 1 : await Future.wait(
725 4 : _toDeviceQueueBox.keys.map((i) async {
726 2 : final raw = await _toDeviceQueueBox.get(i);
727 1 : raw['id'] = i;
728 2 : return QueuedToDeviceEvent.fromJson(convertToJson(raw));
729 1 : }).toList(),
730 : );
731 :
732 1 : @override
733 : Future<List<Event>> getUnimportantRoomEventStatesForRoom(
734 : List<String> events,
735 : Room room,
736 : ) async {
737 4 : final keys = _roomStateBox.keys.where((key) {
738 1 : final tuple = MultiKey.fromString(key);
739 4 : return tuple.parts.first == room.id && !events.contains(tuple.parts[1]);
740 : });
741 :
742 1 : final unimportantEvents = <Event>[];
743 1 : for (final key in keys) {
744 0 : final Map states = await _roomStateBox.get(key);
745 0 : unimportantEvents.addAll(
746 0 : states.values.map((raw) => Event.fromJson(convertToJson(raw), room)),
747 : );
748 : }
749 2 : return unimportantEvents.where((event) => event.stateKey != null).toList();
750 : }
751 :
752 1 : @override
753 : Future<User?> getUser(String userId, Room room) async {
754 : final state =
755 5 : await _roomMembersBox.get(MultiKey(room.id, userId).toString());
756 : if (state == null) return null;
757 0 : return Event.fromJson(convertToJson(state), room).asUser;
758 : }
759 :
760 1 : @override
761 : Future<Map<String, DeviceKeysList>> getUserDeviceKeys(Client client) =>
762 1 : runBenchmarked<Map<String, DeviceKeysList>>(
763 : 'Get all user device keys from Hive',
764 1 : () async {
765 2 : final deviceKeysOutdated = _userDeviceKeysOutdatedBox.keys;
766 1 : if (deviceKeysOutdated.isEmpty) {
767 1 : return {};
768 : }
769 0 : final res = <String, DeviceKeysList>{};
770 0 : for (final userId in deviceKeysOutdated) {
771 0 : final deviceKeysBoxKeys = _userDeviceKeysBox.keys.where((tuple) {
772 0 : final tupleKey = MultiKey.fromString(tuple);
773 0 : return tupleKey.parts.first == userId;
774 : });
775 : final crossSigningKeysBoxKeys =
776 0 : _userCrossSigningKeysBox.keys.where((tuple) {
777 0 : final tupleKey = MultiKey.fromString(tuple);
778 0 : return tupleKey.parts.first == userId;
779 : });
780 0 : res[userId] = DeviceKeysList.fromDbJson(
781 0 : {
782 0 : 'client_id': client.id,
783 : 'user_id': userId,
784 0 : 'outdated': await _userDeviceKeysOutdatedBox.get(userId),
785 : },
786 0 : await Future.wait(
787 0 : deviceKeysBoxKeys.map(
788 0 : (key) async =>
789 0 : convertToJson(await _userDeviceKeysBox.get(key)),
790 : ),
791 : ),
792 0 : await Future.wait(
793 0 : crossSigningKeysBoxKeys.map(
794 0 : (key) async =>
795 0 : convertToJson(await _userCrossSigningKeysBox.get(key)),
796 : ),
797 : ),
798 : client,
799 : );
800 : }
801 : return res;
802 : },
803 3 : _userDeviceKeysBox.keys.length,
804 : );
805 :
806 1 : @override
807 : Future<List<User>> getUsers(Room room) async {
808 1 : final users = <User>[];
809 2 : for (final key in _roomMembersBox.keys) {
810 0 : final statesKey = MultiKey.fromString(key);
811 0 : if (statesKey.parts[0] != room.id) continue;
812 0 : final state = await _roomMembersBox.get(key);
813 0 : users.add(Event.fromJson(convertToJson(state), room).asUser);
814 : }
815 : return users;
816 : }
817 :
818 1 : @override
819 : Future<void> insertClient(
820 : String name,
821 : String homeserverUrl,
822 : String token,
823 : DateTime? tokenExpiresAt,
824 : String? refreshToken,
825 : String userId,
826 : String? deviceId,
827 : String? deviceName,
828 : String? prevBatch,
829 : String? olmAccount,
830 : ) async {
831 2 : await _clientBox.put('homeserver_url', homeserverUrl);
832 2 : await _clientBox.put('token', token);
833 2 : await _clientBox.put(
834 : 'token_expires_at',
835 2 : tokenExpiresAt?.millisecondsSinceEpoch.toString(),
836 : );
837 2 : await _clientBox.put('refresh_token', refreshToken);
838 2 : await _clientBox.put('user_id', userId);
839 2 : await _clientBox.put('device_id', deviceId);
840 2 : await _clientBox.put('device_name', deviceName);
841 2 : await _clientBox.put('prev_batch', prevBatch);
842 2 : await _clientBox.put('olm_account', olmAccount);
843 2 : await _clientBox.put('sync_filter_id', null);
844 : return;
845 : }
846 :
847 1 : @override
848 : Future<int> insertIntoToDeviceQueue(
849 : String type,
850 : String txnId,
851 : String content,
852 : ) async {
853 3 : return await _toDeviceQueueBox.add(<String, dynamic>{
854 : 'type': type,
855 : 'txn_id': txnId,
856 : 'content': content,
857 : });
858 : }
859 :
860 1 : @override
861 : Future<void> markInboundGroupSessionAsUploaded(
862 : String roomId,
863 : String sessionId,
864 : ) async {
865 3 : final raw = await _inboundGroupSessionsBox.get(sessionId.toHiveKey);
866 : if (raw == null) {
867 0 : Logs().w(
868 : 'Tried to mark inbound group session as uploaded which was not found in the database!',
869 : );
870 : return;
871 : }
872 1 : raw['uploaded'] = true;
873 3 : await _inboundGroupSessionsBox.put(sessionId.toHiveKey, raw);
874 : return;
875 : }
876 :
877 1 : @override
878 : Future<void> markInboundGroupSessionsAsNeedingUpload() async {
879 3 : for (final sessionId in _inboundGroupSessionsBox.keys) {
880 2 : final raw = await _inboundGroupSessionsBox.get(sessionId);
881 1 : raw['uploaded'] = false;
882 2 : await _inboundGroupSessionsBox.put(sessionId, raw);
883 : }
884 : return;
885 : }
886 :
887 1 : @override
888 : Future<void> removeEvent(String eventId, String roomId) async {
889 4 : await _eventsBox.delete(MultiKey(roomId, eventId).toString());
890 3 : for (final key in _timelineFragmentsBox.keys) {
891 1 : final multiKey = MultiKey.fromString(key);
892 3 : if (multiKey.parts.first != roomId) continue;
893 2 : final List eventIds = await _timelineFragmentsBox.get(key) ?? [];
894 1 : final prevLength = eventIds.length;
895 3 : eventIds.removeWhere((id) => id == eventId);
896 2 : if (eventIds.length < prevLength) {
897 2 : await _timelineFragmentsBox.put(key, eventIds);
898 : }
899 : }
900 : return;
901 : }
902 :
903 0 : @override
904 : Future<void> removeOutboundGroupSession(String roomId) async {
905 0 : await _outboundGroupSessionsBox.delete(roomId.toHiveKey);
906 : return;
907 : }
908 :
909 1 : @override
910 : Future<void> removeUserCrossSigningKey(
911 : String userId,
912 : String publicKey,
913 : ) async {
914 1 : await _userCrossSigningKeysBox
915 3 : .delete(MultiKey(userId, publicKey).toString());
916 : return;
917 : }
918 :
919 0 : @override
920 : Future<void> removeUserDeviceKey(String userId, String deviceId) async {
921 0 : await _userDeviceKeysBox.delete(MultiKey(userId, deviceId).toString());
922 : return;
923 : }
924 :
925 1 : @override
926 : Future<void> setBlockedUserCrossSigningKey(
927 : bool blocked,
928 : String userId,
929 : String publicKey,
930 : ) async {
931 1 : final raw = await _userCrossSigningKeysBox
932 3 : .get(MultiKey(userId, publicKey).toString());
933 1 : raw['blocked'] = blocked;
934 2 : await _userCrossSigningKeysBox.put(
935 2 : MultiKey(userId, publicKey).toString(),
936 : raw,
937 : );
938 : return;
939 : }
940 :
941 1 : @override
942 : Future<void> setBlockedUserDeviceKey(
943 : bool blocked,
944 : String userId,
945 : String deviceId,
946 : ) async {
947 : final raw =
948 4 : await _userDeviceKeysBox.get(MultiKey(userId, deviceId).toString());
949 1 : raw['blocked'] = blocked;
950 2 : await _userDeviceKeysBox.put(
951 2 : MultiKey(userId, deviceId).toString(),
952 : raw,
953 : );
954 : return;
955 : }
956 :
957 0 : @override
958 : Future<void> setLastActiveUserDeviceKey(
959 : int lastActive,
960 : String userId,
961 : String deviceId,
962 : ) async {
963 : final raw =
964 0 : await _userDeviceKeysBox.get(MultiKey(userId, deviceId).toString());
965 0 : raw['last_active'] = lastActive;
966 0 : await _userDeviceKeysBox.put(
967 0 : MultiKey(userId, deviceId).toString(),
968 : raw,
969 : );
970 : }
971 :
972 0 : @override
973 : Future<void> setLastSentMessageUserDeviceKey(
974 : String lastSentMessage,
975 : String userId,
976 : String deviceId,
977 : ) async {
978 : final raw =
979 0 : await _userDeviceKeysBox.get(MultiKey(userId, deviceId).toString());
980 0 : raw['last_sent_message'] = lastSentMessage;
981 0 : await _userDeviceKeysBox.put(
982 0 : MultiKey(userId, deviceId).toString(),
983 : raw,
984 : );
985 : }
986 :
987 1 : @override
988 : Future<void> setRoomPrevBatch(
989 : String? prevBatch,
990 : String roomId,
991 : Client client,
992 : ) async {
993 3 : final raw = await _roomsBox.get(roomId.toHiveKey);
994 : if (raw == null) return;
995 2 : final room = Room.fromJson(convertToJson(raw), client);
996 1 : room.prev_batch = prevBatch;
997 4 : await _roomsBox.put(roomId.toHiveKey, room.toJson());
998 : return;
999 : }
1000 :
1001 1 : @override
1002 : Future<void> setVerifiedUserCrossSigningKey(
1003 : bool verified,
1004 : String userId,
1005 : String publicKey,
1006 : ) async {
1007 1 : final raw = (await _userCrossSigningKeysBox
1008 3 : .get(MultiKey(userId, publicKey).toString()) as Map?) ??
1009 0 : {};
1010 1 : raw['verified'] = verified;
1011 2 : await _userCrossSigningKeysBox.put(
1012 2 : MultiKey(userId, publicKey).toString(),
1013 : raw,
1014 : );
1015 : return;
1016 : }
1017 :
1018 1 : @override
1019 : Future<void> setVerifiedUserDeviceKey(
1020 : bool verified,
1021 : String userId,
1022 : String deviceId,
1023 : ) async {
1024 : final raw =
1025 4 : await _userDeviceKeysBox.get(MultiKey(userId, deviceId).toString());
1026 1 : raw['verified'] = verified;
1027 2 : await _userDeviceKeysBox.put(
1028 2 : MultiKey(userId, deviceId).toString(),
1029 : raw,
1030 : );
1031 : return;
1032 : }
1033 :
1034 1 : @override
1035 : Future<void> storeAccountData(String type, String content) async {
1036 2 : await _accountDataBox.put(
1037 1 : type.toHiveKey,
1038 2 : convertToJson(jsonDecode(content)),
1039 : );
1040 : return;
1041 : }
1042 :
1043 1 : @override
1044 : Future<void> storeEventUpdate(EventUpdate eventUpdate, Client client) async {
1045 : // Ephemerals should not be stored
1046 2 : if (eventUpdate.type == EventUpdateType.ephemeral) return;
1047 :
1048 : // In case of this is a redaction event
1049 3 : if (eventUpdate.content['type'] == EventTypes.Redaction) {
1050 0 : final tmpRoom = client.getRoomById(eventUpdate.roomID) ??
1051 0 : Room(id: eventUpdate.roomID, client: client);
1052 0 : final eventId = eventUpdate.content.tryGet<String>('redacts');
1053 : final event =
1054 0 : eventId != null ? await getEventById(eventId, tmpRoom) : null;
1055 : if (event != null) {
1056 0 : event.setRedactionEvent(Event.fromJson(eventUpdate.content, tmpRoom));
1057 0 : await _eventsBox.put(
1058 0 : MultiKey(eventUpdate.roomID, event.eventId).toString(),
1059 0 : event.toJson(),
1060 : );
1061 :
1062 0 : if (tmpRoom.lastEvent?.eventId == event.eventId) {
1063 0 : await _roomStateBox.put(
1064 0 : MultiKey(eventUpdate.roomID, event.type).toString(),
1065 0 : {'': event.toJson()},
1066 : );
1067 : }
1068 : }
1069 : }
1070 :
1071 : // Store a common message event
1072 : if ({
1073 1 : EventUpdateType.timeline,
1074 1 : EventUpdateType.history,
1075 1 : EventUpdateType.decryptedTimelineQueue,
1076 2 : }.contains(eventUpdate.type)) {
1077 2 : final eventId = eventUpdate.content['event_id'];
1078 : // Is this ID already in the store?
1079 1 : final Map? prevEvent = await _eventsBox
1080 4 : .get(MultiKey(eventUpdate.roomID, eventId).toString());
1081 : final prevStatus = prevEvent == null
1082 : ? null
1083 0 : : () {
1084 0 : final json = convertToJson(prevEvent);
1085 0 : final statusInt = json.tryGet<int>('status') ??
1086 : json
1087 0 : .tryGetMap<String, dynamic>('unsigned')
1088 0 : ?.tryGet<int>(messageSendingStatusKey);
1089 0 : return statusInt == null ? null : eventStatusFromInt(statusInt);
1090 0 : }();
1091 :
1092 : // calculate the status
1093 1 : final newStatus = eventStatusFromInt(
1094 2 : eventUpdate.content.tryGet<int>('status') ??
1095 1 : eventUpdate.content
1096 1 : .tryGetMap<String, dynamic>('unsigned')
1097 0 : ?.tryGet<int>(messageSendingStatusKey) ??
1098 1 : EventStatus.synced.intValue,
1099 : );
1100 :
1101 : // Is this the response to a sending event which is already synced? Then
1102 : // there is nothing to do here.
1103 1 : if (!newStatus.isSynced && prevStatus != null && prevStatus.isSynced) {
1104 : return;
1105 : }
1106 :
1107 1 : final status = newStatus.isError || prevStatus == null
1108 : ? newStatus
1109 0 : : latestEventStatus(
1110 : prevStatus,
1111 : newStatus,
1112 : );
1113 :
1114 : // Add the status and the sort order to the content so it get stored
1115 3 : eventUpdate.content['unsigned'] ??= <String, dynamic>{};
1116 3 : eventUpdate.content['unsigned'][messageSendingStatusKey] =
1117 3 : eventUpdate.content['status'] = status.intValue;
1118 :
1119 : // In case this event has sent from this account we have a transaction ID
1120 1 : final transactionId = eventUpdate.content
1121 1 : .tryGetMap<String, dynamic>('unsigned')
1122 1 : ?.tryGet<String>('transaction_id');
1123 :
1124 2 : await _eventsBox.put(
1125 3 : MultiKey(eventUpdate.roomID, eventId).toString(),
1126 1 : eventUpdate.content,
1127 : );
1128 :
1129 : // Update timeline fragments
1130 3 : final key = MultiKey(eventUpdate.roomID, status.isSent ? '' : 'SENDING')
1131 1 : .toString();
1132 :
1133 3 : final List eventIds = (await _timelineFragmentsBox.get(key) ?? []);
1134 :
1135 1 : if (!eventIds.contains(eventId)) {
1136 2 : if (eventUpdate.type == EventUpdateType.history) {
1137 1 : eventIds.add(eventId);
1138 : } else {
1139 1 : eventIds.insert(0, eventId);
1140 : }
1141 2 : await _timelineFragmentsBox.put(key, eventIds);
1142 0 : } else if (status.isSynced &&
1143 : prevStatus != null &&
1144 0 : prevStatus.isSent &&
1145 0 : eventUpdate.type != EventUpdateType.history) {
1146 : // Status changes from 1 -> 2? Make sure event is correctly sorted.
1147 0 : eventIds.remove(eventId);
1148 0 : eventIds.insert(0, eventId);
1149 : }
1150 :
1151 : // If event comes from server timeline, remove sending events with this ID
1152 1 : if (status.isSent) {
1153 3 : final key = MultiKey(eventUpdate.roomID, 'SENDING').toString();
1154 3 : final List eventIds = (await _timelineFragmentsBox.get(key) ?? []);
1155 1 : final i = eventIds.indexWhere((id) => id == eventId);
1156 2 : if (i != -1) {
1157 0 : await _timelineFragmentsBox.put(key, eventIds..removeAt(i));
1158 : }
1159 : }
1160 :
1161 : // Is there a transaction id? Then delete the event with this id.
1162 2 : if (!status.isError && !status.isSending && transactionId != null) {
1163 0 : await removeEvent(transactionId, eventUpdate.roomID);
1164 : }
1165 : }
1166 :
1167 2 : final stateKey = eventUpdate.content['state_key'];
1168 : // Store a common state event
1169 : if (stateKey != null &&
1170 : // Don't store events as state updates when paginating backwards.
1171 2 : (eventUpdate.type == EventUpdateType.timeline ||
1172 2 : eventUpdate.type == EventUpdateType.state ||
1173 2 : eventUpdate.type == EventUpdateType.inviteState)) {
1174 3 : if (eventUpdate.content['type'] == EventTypes.RoomMember) {
1175 0 : await _roomMembersBox.put(
1176 0 : MultiKey(
1177 0 : eventUpdate.roomID,
1178 0 : eventUpdate.content['state_key'],
1179 0 : ).toString(),
1180 0 : eventUpdate.content,
1181 : );
1182 : } else {
1183 1 : final key = MultiKey(
1184 1 : eventUpdate.roomID,
1185 2 : eventUpdate.content['type'],
1186 1 : ).toString();
1187 3 : final Map stateMap = await _roomStateBox.get(key) ?? {};
1188 :
1189 2 : stateMap[stateKey] = eventUpdate.content;
1190 2 : await _roomStateBox.put(key, stateMap);
1191 : }
1192 : }
1193 :
1194 : // Store a room account data event
1195 2 : if (eventUpdate.type == EventUpdateType.accountData) {
1196 0 : await _roomAccountDataBox.put(
1197 0 : MultiKey(
1198 0 : eventUpdate.roomID,
1199 0 : eventUpdate.content['type'],
1200 0 : ).toString(),
1201 0 : eventUpdate.content,
1202 : );
1203 : }
1204 : }
1205 :
1206 1 : @override
1207 : Future<void> storeFile(Uri mxcUri, Uint8List bytes, int time) async {
1208 : return;
1209 : }
1210 :
1211 1 : @override
1212 : Future<void> storeInboundGroupSession(
1213 : String roomId,
1214 : String sessionId,
1215 : String pickle,
1216 : String content,
1217 : String indexes,
1218 : String allowedAtIndex,
1219 : String senderKey,
1220 : String senderClaimedKey,
1221 : ) async {
1222 2 : await _inboundGroupSessionsBox.put(
1223 1 : sessionId.toHiveKey,
1224 1 : StoredInboundGroupSession(
1225 : roomId: roomId,
1226 : sessionId: sessionId,
1227 : pickle: pickle,
1228 : content: content,
1229 : indexes: indexes,
1230 : allowedAtIndex: allowedAtIndex,
1231 : senderKey: senderKey,
1232 : senderClaimedKeys: senderClaimedKey,
1233 : uploaded: false,
1234 1 : ).toJson(),
1235 : );
1236 : return;
1237 : }
1238 :
1239 1 : @override
1240 : Future<void> storeOutboundGroupSession(
1241 : String roomId,
1242 : String pickle,
1243 : String deviceIds,
1244 : int creationTime,
1245 : ) async {
1246 4 : await _outboundGroupSessionsBox.put(roomId.toHiveKey, <String, dynamic>{
1247 : 'room_id': roomId,
1248 : 'pickle': pickle,
1249 : 'device_ids': deviceIds,
1250 : 'creation_time': creationTime,
1251 : });
1252 : return;
1253 : }
1254 :
1255 0 : @override
1256 : Future<void> storePrevBatch(
1257 : String prevBatch,
1258 : ) async {
1259 0 : if (_clientBox.keys.isEmpty) return;
1260 0 : await _clientBox.put('prev_batch', prevBatch);
1261 : return;
1262 : }
1263 :
1264 1 : @override
1265 : Future<void> storeRoomUpdate(
1266 : String roomId,
1267 : SyncRoomUpdate roomUpdate,
1268 : Event? lastEvent,
1269 : Client client,
1270 : ) async {
1271 : // Leave room if membership is leave
1272 1 : if (roomUpdate is LeftRoomUpdate) {
1273 0 : await forgetRoom(roomId);
1274 : return;
1275 : }
1276 1 : final membership = roomUpdate is LeftRoomUpdate
1277 : ? Membership.leave
1278 1 : : roomUpdate is InvitedRoomUpdate
1279 : ? Membership.invite
1280 : : Membership.join;
1281 : // Make sure room exists
1282 3 : if (!_roomsBox.containsKey(roomId.toHiveKey)) {
1283 2 : await _roomsBox.put(
1284 1 : roomId.toHiveKey,
1285 1 : roomUpdate is JoinedRoomUpdate
1286 1 : ? Room(
1287 : client: client,
1288 : id: roomId,
1289 : membership: membership,
1290 : highlightCount:
1291 1 : roomUpdate.unreadNotifications?.highlightCount?.toInt() ??
1292 : 0,
1293 : notificationCount: roomUpdate
1294 1 : .unreadNotifications?.notificationCount
1295 0 : ?.toInt() ??
1296 : 0,
1297 1 : prev_batch: roomUpdate.timeline?.prevBatch,
1298 1 : summary: roomUpdate.summary,
1299 : lastEvent: lastEvent,
1300 1 : ).toJson()
1301 0 : : Room(
1302 : client: client,
1303 : id: roomId,
1304 : membership: membership,
1305 : lastEvent: lastEvent,
1306 0 : ).toJson(),
1307 : );
1308 0 : } else if (roomUpdate is JoinedRoomUpdate) {
1309 0 : final currentRawRoom = await _roomsBox.get(roomId.toHiveKey);
1310 0 : final currentRoom = Room.fromJson(convertToJson(currentRawRoom), client);
1311 0 : await _roomsBox.put(
1312 0 : roomId.toHiveKey,
1313 0 : Room(
1314 : client: client,
1315 : id: roomId,
1316 : membership: membership,
1317 : highlightCount:
1318 0 : roomUpdate.unreadNotifications?.highlightCount?.toInt() ??
1319 0 : currentRoom.highlightCount,
1320 : notificationCount:
1321 0 : roomUpdate.unreadNotifications?.notificationCount?.toInt() ??
1322 0 : currentRoom.notificationCount,
1323 0 : prev_batch: roomUpdate.timeline?.prevBatch ?? currentRoom.prev_batch,
1324 0 : summary: RoomSummary.fromJson(
1325 0 : currentRoom.summary.toJson()
1326 0 : ..addAll(roomUpdate.summary?.toJson() ?? {}),
1327 : ),
1328 0 : ).toJson(),
1329 : );
1330 : }
1331 : }
1332 :
1333 0 : @override
1334 : Future<void> deleteTimelineForRoom(String roomId) =>
1335 0 : _timelineFragmentsBox.delete(TupleKey(roomId, '').toString());
1336 :
1337 1 : @override
1338 : Future<void> storeSSSSCache(
1339 : String type,
1340 : String keyId,
1341 : String ciphertext,
1342 : String content,
1343 : ) async {
1344 2 : await _ssssCacheBox.put(
1345 : type,
1346 1 : SSSSCache(
1347 : type: type,
1348 : keyId: keyId,
1349 : ciphertext: ciphertext,
1350 : content: content,
1351 1 : ).toJson(),
1352 : );
1353 : }
1354 :
1355 1 : @override
1356 : Future<void> storeSyncFilterId(
1357 : String syncFilterId,
1358 : ) async {
1359 2 : await _clientBox.put('sync_filter_id', syncFilterId);
1360 : }
1361 :
1362 1 : @override
1363 : Future<void> storeUserCrossSigningKey(
1364 : String userId,
1365 : String publicKey,
1366 : String content,
1367 : bool verified,
1368 : bool blocked,
1369 : ) async {
1370 2 : await _userCrossSigningKeysBox.put(
1371 2 : MultiKey(userId, publicKey).toString(),
1372 1 : {
1373 : 'user_id': userId,
1374 : 'public_key': publicKey,
1375 : 'content': content,
1376 : 'verified': verified,
1377 : 'blocked': blocked,
1378 : },
1379 : );
1380 : }
1381 :
1382 1 : @override
1383 : Future<void> storeUserDeviceKey(
1384 : String userId,
1385 : String deviceId,
1386 : String content,
1387 : bool verified,
1388 : bool blocked,
1389 : int lastActive,
1390 : ) async {
1391 5 : await _userDeviceKeysBox.put(MultiKey(userId, deviceId).toString(), {
1392 : 'user_id': userId,
1393 : 'device_id': deviceId,
1394 : 'content': content,
1395 : 'verified': verified,
1396 : 'blocked': blocked,
1397 : 'last_active': lastActive,
1398 : 'last_sent_message': '',
1399 : });
1400 : return;
1401 : }
1402 :
1403 1 : @override
1404 : Future<void> storeUserDeviceKeysInfo(String userId, bool outdated) async {
1405 3 : await _userDeviceKeysOutdatedBox.put(userId.toHiveKey, outdated);
1406 : return;
1407 : }
1408 :
1409 1 : @override
1410 : Future<void> transaction(Future<void> Function() action) =>
1411 1 : zoneTransaction(action);
1412 :
1413 1 : @override
1414 : Future<void> updateClient(
1415 : String homeserverUrl,
1416 : String token,
1417 : DateTime? tokenExpiresAt,
1418 : String? refreshToken,
1419 : String userId,
1420 : String? deviceId,
1421 : String? deviceName,
1422 : String? prevBatch,
1423 : String? olmAccount,
1424 : ) async {
1425 2 : await _clientBox.put('homeserver_url', homeserverUrl);
1426 2 : await _clientBox.put('token', token);
1427 2 : await _clientBox.put(
1428 : 'token_expires_at',
1429 2 : tokenExpiresAt?.millisecondsSinceEpoch.toString(),
1430 : );
1431 2 : await _clientBox.put('refresh_token', refreshToken);
1432 2 : await _clientBox.put('user_id', userId);
1433 2 : await _clientBox.put('device_id', deviceId);
1434 2 : await _clientBox.put('device_name', deviceName);
1435 2 : await _clientBox.put('prev_batch', prevBatch);
1436 2 : await _clientBox.put('olm_account', olmAccount);
1437 : return;
1438 : }
1439 :
1440 1 : @override
1441 : Future<void> updateClientKeys(
1442 : String olmAccount,
1443 : ) async {
1444 2 : await _clientBox.put('olm_account', olmAccount);
1445 : return;
1446 : }
1447 :
1448 1 : @override
1449 : Future<void> updateInboundGroupSessionAllowedAtIndex(
1450 : String allowedAtIndex,
1451 : String roomId,
1452 : String sessionId,
1453 : ) async {
1454 3 : final raw = await _inboundGroupSessionsBox.get(sessionId.toHiveKey);
1455 : if (raw == null) {
1456 0 : Logs().w(
1457 : 'Tried to update inbound group session as uploaded which wasnt found in the database!',
1458 : );
1459 : return;
1460 : }
1461 1 : raw['allowed_at_index'] = allowedAtIndex;
1462 3 : await _inboundGroupSessionsBox.put(sessionId.toHiveKey, raw);
1463 : return;
1464 : }
1465 :
1466 1 : @override
1467 : Future<void> updateInboundGroupSessionIndexes(
1468 : String indexes,
1469 : String roomId,
1470 : String sessionId,
1471 : ) async {
1472 3 : final raw = await _inboundGroupSessionsBox.get(sessionId.toHiveKey);
1473 : if (raw == null) {
1474 0 : Logs().w(
1475 : 'Tried to update inbound group session indexes of a session which was not found in the database!',
1476 : );
1477 : return;
1478 : }
1479 1 : raw['indexes'] = indexes;
1480 3 : await _inboundGroupSessionsBox.put(sessionId.toHiveKey, raw);
1481 : return;
1482 : }
1483 :
1484 1 : @override
1485 : Future<List<StoredInboundGroupSession>> getAllInboundGroupSessions() async {
1486 1 : final rawSessions = await Future.wait(
1487 2 : _inboundGroupSessionsBox.keys
1488 1 : .map((key) => _inboundGroupSessionsBox.get(key)),
1489 : );
1490 : return rawSessions
1491 1 : .map((raw) => StoredInboundGroupSession.fromJson(convertToJson(raw)))
1492 1 : .toList();
1493 : }
1494 :
1495 0 : @override
1496 : Future<void> addSeenDeviceId(
1497 : String userId,
1498 : String deviceId,
1499 : String publicKeys,
1500 : ) =>
1501 0 : _seenDeviceIdsBox.put(MultiKey(userId, deviceId).toString(), publicKeys);
1502 :
1503 0 : @override
1504 : Future<void> addSeenPublicKey(
1505 : String publicKey,
1506 : String deviceId,
1507 : ) =>
1508 0 : _seenDeviceKeysBox.put(publicKey.toHiveKey, deviceId);
1509 :
1510 0 : @override
1511 : Future<String?> deviceIdSeen(userId, deviceId) async {
1512 : final raw =
1513 0 : await _seenDeviceIdsBox.get(MultiKey(userId, deviceId).toString());
1514 : if (raw == null) return null;
1515 : return raw as String;
1516 : }
1517 :
1518 0 : @override
1519 : Future<String?> publicKeySeen(String publicKey) async {
1520 0 : final raw = await _seenDeviceKeysBox.get(publicKey.toHiveKey);
1521 : if (raw == null) return null;
1522 : return raw as String;
1523 : }
1524 :
1525 1 : @override
1526 : Future<void> storePresence(String userId, CachedPresence presence) =>
1527 3 : _presencesBox.put(userId, presence.toJson());
1528 :
1529 1 : @override
1530 : Future<CachedPresence?> getPresence(String userId) async {
1531 2 : final rawPresence = await _presencesBox.get(userId);
1532 : if (rawPresence == null) return null;
1533 :
1534 2 : return CachedPresence.fromJson(copyMap(rawPresence));
1535 : }
1536 :
1537 0 : @override
1538 : Future<String> exportDump() {
1539 : // see no need to implement this in a deprecated part
1540 0 : throw UnimplementedError();
1541 : }
1542 :
1543 0 : @override
1544 : Future<bool> importDump(String export) {
1545 : // see no need to implement this in a deprecated part
1546 0 : throw UnimplementedError();
1547 : }
1548 :
1549 0 : @override
1550 : Future<void> storeWellKnown(DiscoveryInformation? discoveryInformation) {
1551 : if (discoveryInformation == null) {
1552 0 : return _clientBox.delete('discovery_information');
1553 : }
1554 0 : return _clientBox.put(
1555 : 'discovery_information',
1556 0 : jsonEncode(discoveryInformation.toJson()),
1557 : );
1558 : }
1559 :
1560 0 : @override
1561 : Future<DiscoveryInformation?> getWellKnown() async {
1562 : final rawDiscoveryInformation =
1563 0 : await _clientBox.get('discovery_information');
1564 : if (rawDiscoveryInformation == null) return null;
1565 0 : return DiscoveryInformation.fromJson(jsonDecode(rawDiscoveryInformation));
1566 : }
1567 :
1568 0 : @override
1569 0 : Future<void> delete() => Hive.deleteFromDisk();
1570 :
1571 0 : @override
1572 : Future<void> markUserProfileAsOutdated(userId) async {
1573 : return;
1574 : }
1575 :
1576 1 : @override
1577 : Future<CachedProfileInformation?> getUserProfile(String userId) async {
1578 : return null;
1579 : }
1580 :
1581 1 : @override
1582 : Future<void> storeUserProfile(
1583 : String userId,
1584 : CachedProfileInformation profile,
1585 : ) async {
1586 : return;
1587 : }
1588 : }
1589 :
1590 1 : dynamic _castValue(dynamic value) {
1591 1 : if (value is Map) {
1592 1 : return convertToJson(value);
1593 : }
1594 1 : if (value is List) {
1595 2 : return value.map(_castValue).toList();
1596 : }
1597 : return value;
1598 : }
1599 :
1600 : /// Hive always gives back an `_InternalLinkedHasMap<dynamic, dynamic>`. This
1601 : /// creates a deep copy of the json and makes sure that the format is always
1602 : /// `Map<String, dynamic>`.
1603 1 : Map<String, dynamic> convertToJson(Map map) {
1604 1 : final copy = Map<String, dynamic>.from(map);
1605 2 : for (final entry in copy.entries) {
1606 4 : copy[entry.key] = _castValue(entry.value);
1607 : }
1608 : return copy;
1609 : }
1610 :
1611 : class MultiKey {
1612 : final List<String> parts;
1613 :
1614 1 : MultiKey(String key1, [String? key2, String? key3])
1615 1 : : parts = [
1616 : key1,
1617 1 : if (key2 != null) key2,
1618 0 : if (key3 != null) key3,
1619 : ];
1620 :
1621 0 : const MultiKey.byParts(this.parts);
1622 :
1623 1 : MultiKey.fromString(String multiKeyString)
1624 5 : : parts = multiKeyString.split('|').map((s) => s.fromHiveKey).toList();
1625 :
1626 1 : @override
1627 5 : String toString() => parts.map((s) => s.toHiveKey).join('|');
1628 :
1629 0 : @override
1630 0 : bool operator ==(other) => parts.toString() == other.toString();
1631 :
1632 0 : @override
1633 0 : int get hashCode => Object.hashAll(parts);
1634 : }
1635 :
1636 : extension HiveKeyExtension on String {
1637 2 : String get toHiveKey => isValidMatrixId
1638 5 : ? '$sigil${Uri.encodeComponent(localpart!)}:${Uri.encodeComponent(domain!)}'
1639 2 : : Uri.encodeComponent(this);
1640 : }
1641 :
1642 : extension FromHiveKeyExtension on String {
1643 2 : String get fromHiveKey => Uri.decodeComponent(this);
1644 : }
|