Merge branch 'krille/check-code-formatting' into 'master'

Add code formatting CI job

See merge request famedly/famedlysdk!320
This commit is contained in:
Christian Pauly 2020-05-22 10:12:18 +00:00
commit bb75ee00be
16 changed files with 486 additions and 253 deletions

View file

@ -54,6 +54,7 @@ code_analyze:
image: cirrusci/flutter image: cirrusci/flutter
dependencies: [] dependencies: []
script: script:
- flutter format lib/ test/ test_driver/ --set-exit-if-changed
- flutter analyze - flutter analyze
build-api-doc: build-api-doc:

View file

@ -41,7 +41,10 @@ That means for example: "[users] add fetch users endpoint".
- Directories need to be lowercase. - Directories need to be lowercase.
## Code style: ## Code style:
- We recommend using Android Studio for coding. We are using the Android Studio auto styling with ctrl+alt+shift+L. Please use code formatting. You can use VSCode or Android Studio. On other editors you need to run:
```
flutter format lib/**/*/*.dart
```
## Code quality: ## Code quality:
- Don't repeat yourself! Use local variables, functions, classes. - Don't repeat yourself! Use local variables, functions, classes.

View file

@ -84,7 +84,8 @@ class Client {
/// debug: Print debug output? /// debug: Print debug output?
/// database: The database instance to use /// database: The database instance to use
/// enableE2eeRecovery: Enable additional logic to try to recover from bad e2ee sessions /// enableE2eeRecovery: Enable additional logic to try to recover from bad e2ee sessions
Client(this.clientName, {this.debug = false, this.database, this.enableE2eeRecovery = false}) { Client(this.clientName,
{this.debug = false, this.database, this.enableE2eeRecovery = false}) {
onLoginStateChanged.stream.listen((loginState) { onLoginStateChanged.stream.listen((loginState) {
print('LoginState: ${loginState.toString()}'); print('LoginState: ${loginState.toString()}');
}); });
@ -1045,8 +1046,7 @@ class Client {
} }
if (sync['account_data'] is Map<String, dynamic> && if (sync['account_data'] is Map<String, dynamic> &&
sync['account_data']['events'] is List<dynamic>) { sync['account_data']['events'] is List<dynamic>) {
await _handleGlobalEvents( await _handleGlobalEvents(sync['account_data']['events'], 'account_data');
sync['account_data']['events'], 'account_data');
} }
if (sync['device_lists'] is Map<String, dynamic>) { if (sync['device_lists'] is Map<String, dynamic>) {
await _handleDeviceListsEvents(sync['device_lists']); await _handleDeviceListsEvents(sync['device_lists']);
@ -1077,7 +1077,8 @@ class Client {
} }
} }
Future<void> _handleDeviceListsEvents(Map<String, dynamic> deviceLists) async { Future<void> _handleDeviceListsEvents(
Map<String, dynamic> deviceLists) async {
if (deviceLists['changed'] is List) { if (deviceLists['changed'] is List) {
for (final userId in deviceLists['changed']) { for (final userId in deviceLists['changed']) {
if (_userDeviceKeys.containsKey(userId)) { if (_userDeviceKeys.containsKey(userId)) {
@ -1179,7 +1180,8 @@ class Client {
} }
} }
Future<void> _handleRooms(Map<String, dynamic> rooms, Membership membership) async { Future<void> _handleRooms(
Map<String, dynamic> rooms, Membership membership) async {
for (final entry in rooms.entries) { for (final entry in rooms.entries) {
final id = entry.key; final id = entry.key;
final room = entry.value; final room = entry.value;
@ -1252,8 +1254,7 @@ class Client {
if (room['timeline'] is Map<String, dynamic> && if (room['timeline'] is Map<String, dynamic> &&
room['timeline']['events'] is List<dynamic> && room['timeline']['events'] is List<dynamic> &&
room['timeline']['events'].isNotEmpty) { room['timeline']['events'].isNotEmpty) {
await _handleRoomEvents( await _handleRoomEvents(id, room['timeline']['events'], 'timeline');
id, room['timeline']['events'], 'timeline');
handledEvents = true; handledEvents = true;
} }
@ -1318,7 +1319,8 @@ class Client {
} }
} }
Future<void> _handleRoomEvents(String chat_id, List<dynamic> events, String type) async { Future<void> _handleRoomEvents(
String chat_id, List<dynamic> events, String type) async {
for (num i = 0; i < events.length; i++) { for (num i = 0; i < events.length; i++) {
await _handleEvent(events[i], chat_id, type); await _handleEvent(events[i], chat_id, type);
} }
@ -1341,14 +1343,17 @@ class Client {
} }
} }
Future<void> _handleEvent(Map<String, dynamic> event, String roomID, String type) async { Future<void> _handleEvent(
Map<String, dynamic> event, String roomID, String type) async {
if (event['type'] is String && event['content'] is Map<String, dynamic>) { if (event['type'] is String && event['content'] is Map<String, dynamic>) {
// The client must ignore any new m.room.encryption event to prevent // The client must ignore any new m.room.encryption event to prevent
// man-in-the-middle attacks! // man-in-the-middle attacks!
final room = getRoomById(roomID); final room = getRoomById(roomID);
if (room == null || if (room == null ||
(event['type'] == 'm.room.encryption' && room.encrypted && (event['type'] == 'm.room.encryption' &&
event['content']['algorithm'] != room.getState('m.room.encryption')?.content['algorithm'])) { room.encrypted &&
event['content']['algorithm'] !=
room.getState('m.room.encryption')?.content['algorithm'])) {
return; return;
} }
@ -1367,7 +1372,8 @@ class Client {
} }
if (update.eventType == 'm.room.encrypted' && database != null) { if (update.eventType == 'm.room.encrypted' && database != null) {
// the event is still encrytped....let's try fetching the keys from the database! // the event is still encrytped....let's try fetching the keys from the database!
await room.loadInboundGroupSessionKey(event['content']['session_id'], event['content']['sender_key']); await room.loadInboundGroupSessionKey(
event['content']['session_id'], event['content']['sender_key']);
update = update.decrypt(room); update = update.decrypt(room);
} }
if (type != 'ephemeral' && database != null) { if (type != 'ephemeral' && database != null) {
@ -1659,10 +1665,9 @@ class Client {
if (entry.isValid) { if (entry.isValid) {
_userDeviceKeys[userId].deviceKeys[deviceId] = entry; _userDeviceKeys[userId].deviceKeys[deviceId] = entry;
if (deviceId == deviceID && if (deviceId == deviceID &&
entry.ed25519Key == entry.ed25519Key == fingerprintKey) {
fingerprintKey) { // Always trust the own device
// Always trust the own device entry.verified = true;
entry.verified = true;
} }
} }
if (database != null) { if (database != null) {
@ -1670,8 +1675,9 @@ class Client {
id, id,
userId, userId,
deviceId, deviceId,
json.encode( json.encode(_userDeviceKeys[userId]
_userDeviceKeys[userId].deviceKeys[deviceId].toJson()), .deviceKeys[deviceId]
.toJson()),
_userDeviceKeys[userId].deviceKeys[deviceId].verified, _userDeviceKeys[userId].deviceKeys[deviceId].verified,
_userDeviceKeys[userId].deviceKeys[deviceId].blocked, _userDeviceKeys[userId].deviceKeys[deviceId].blocked,
)); ));

View file

@ -19,31 +19,31 @@ class Database extends _$Database {
@override @override
MigrationStrategy get migration => MigrationStrategy( MigrationStrategy get migration => MigrationStrategy(
onCreate: (Migrator m) { onCreate: (Migrator m) {
return m.createAll(); return m.createAll();
}, },
onUpgrade: (Migrator m, int from, int to) async { onUpgrade: (Migrator m, int from, int to) async {
// this appears to be only called once, so multiple consecutive upgrades have to be handled appropriately in here // this appears to be only called once, so multiple consecutive upgrades have to be handled appropriately in here
if (from == 1) { if (from == 1) {
await m.createIndex(userDeviceKeysIndex); await m.createIndex(userDeviceKeysIndex);
await m.createIndex(userDeviceKeysKeyIndex); await m.createIndex(userDeviceKeysKeyIndex);
await m.createIndex(olmSessionsIndex); await m.createIndex(olmSessionsIndex);
await m.createIndex(outboundGroupSessionsIndex); await m.createIndex(outboundGroupSessionsIndex);
await m.createIndex(inboundGroupSessionsIndex); await m.createIndex(inboundGroupSessionsIndex);
await m.createIndex(roomsIndex); await m.createIndex(roomsIndex);
await m.createIndex(eventsIndex); await m.createIndex(eventsIndex);
await m.createIndex(roomStatesIndex); await m.createIndex(roomStatesIndex);
await m.createIndex(accountDataIndex); await m.createIndex(accountDataIndex);
await m.createIndex(roomAccountDataIndex); await m.createIndex(roomAccountDataIndex);
await m.createIndex(presencesIndex); await m.createIndex(presencesIndex);
from++; from++;
} }
if (from == 2) { if (from == 2) {
await m.deleteTable('outbound_group_sessions'); await m.deleteTable('outbound_group_sessions');
await m.createTable(outboundGroupSessions); await m.createTable(outboundGroupSessions);
} }
}, },
); );
Future<DbClient> getClient(String name) async { Future<DbClient> getClient(String name) async {
final res = await dbGetClient(name).get(); final res = await dbGetClient(name).get();
@ -51,7 +51,8 @@ class Database extends _$Database {
return res.first; return res.first;
} }
Future<Map<String, sdk.DeviceKeysList>> getUserDeviceKeys(int clientId) async { Future<Map<String, sdk.DeviceKeysList>> getUserDeviceKeys(
int clientId) async {
final deviceKeys = await getAllUserDeviceKeys(clientId).get(); final deviceKeys = await getAllUserDeviceKeys(clientId).get();
if (deviceKeys.isEmpty) { if (deviceKeys.isEmpty) {
return {}; return {};
@ -59,12 +60,14 @@ class Database extends _$Database {
final deviceKeysKeys = await getAllUserDeviceKeysKeys(clientId).get(); final deviceKeysKeys = await getAllUserDeviceKeysKeys(clientId).get();
final res = <String, sdk.DeviceKeysList>{}; final res = <String, sdk.DeviceKeysList>{};
for (final entry in deviceKeys) { for (final entry in deviceKeys) {
res[entry.userId] = sdk.DeviceKeysList.fromDb(entry, deviceKeysKeys.where((k) => k.userId == entry.userId).toList()); res[entry.userId] = sdk.DeviceKeysList.fromDb(entry,
deviceKeysKeys.where((k) => k.userId == entry.userId).toList());
} }
return res; return res;
} }
Future<Map<String, List<olm.Session>>> getOlmSessions(int clientId, String userId) async { Future<Map<String, List<olm.Session>>> getOlmSessions(
int clientId, String userId) async {
final raw = await getAllOlmSessions(clientId).get(); final raw = await getAllOlmSessions(clientId).get();
if (raw.isEmpty) { if (raw.isEmpty) {
return {}; return {};
@ -85,7 +88,8 @@ class Database extends _$Database {
return res; return res;
} }
Future<DbOutboundGroupSession> getDbOutboundGroupSession(int clientId, String roomId) async { Future<DbOutboundGroupSession> getDbOutboundGroupSession(
int clientId, String roomId) async {
final res = await dbGetOutboundGroupSession(clientId, roomId).get(); final res = await dbGetOutboundGroupSession(clientId, roomId).get();
if (res.isEmpty) { if (res.isEmpty) {
return null; return null;
@ -93,22 +97,28 @@ class Database extends _$Database {
return res.first; return res.first;
} }
Future<List<DbInboundGroupSession>> getDbInboundGroupSessions(int clientId, String roomId) async { Future<List<DbInboundGroupSession>> getDbInboundGroupSessions(
int clientId, String roomId) async {
return await dbGetInboundGroupSessionKeys(clientId, roomId).get(); return await dbGetInboundGroupSessionKeys(clientId, roomId).get();
} }
Future<DbInboundGroupSession> getDbInboundGroupSession(int clientId, String roomId, String sessionId) async { Future<DbInboundGroupSession> getDbInboundGroupSession(
final res = await dbGetInboundGroupSessionKey(clientId, roomId, sessionId).get(); int clientId, String roomId, String sessionId) async {
final res =
await dbGetInboundGroupSessionKey(clientId, roomId, sessionId).get();
if (res.isEmpty) { if (res.isEmpty) {
return null; return null;
} }
return res.first; return res.first;
} }
Future<List<sdk.Room>> getRoomList(sdk.Client client, {bool onlyLeft = false}) async { Future<List<sdk.Room>> getRoomList(sdk.Client client,
final res = await (select(rooms)..where((t) => onlyLeft {bool onlyLeft = false}) async {
? t.membership.equals('leave') final res = await (select(rooms)
: t.membership.equals('leave').not())).get(); ..where((t) => onlyLeft
? t.membership.equals('leave')
: t.membership.equals('leave').not()))
.get();
final resStates = await getAllRoomStates(client.id).get(); final resStates = await getAllRoomStates(client.id).get();
final resAccountData = await getAllRoomAccountData(client.id).get(); final resAccountData = await getAllRoomAccountData(client.id).get();
final roomList = <sdk.Room>[]; final roomList = <sdk.Room>[];
@ -145,11 +155,13 @@ class Database extends _$Database {
/// Stores a RoomUpdate object in the database. Must be called inside of /// Stores a RoomUpdate object in the database. Must be called inside of
/// [transaction]. /// [transaction].
final Set<String> _ensuredRooms = {}; final Set<String> _ensuredRooms = {};
Future<void> storeRoomUpdate(int clientId, sdk.RoomUpdate roomUpdate, [sdk.Room oldRoom]) async { Future<void> storeRoomUpdate(int clientId, sdk.RoomUpdate roomUpdate,
[sdk.Room oldRoom]) async {
final setKey = '${clientId};${roomUpdate.id}'; final setKey = '${clientId};${roomUpdate.id}';
if (roomUpdate.membership != sdk.Membership.leave) { if (roomUpdate.membership != sdk.Membership.leave) {
if (!_ensuredRooms.contains(setKey)) { if (!_ensuredRooms.contains(setKey)) {
await ensureRoomExists(clientId, roomUpdate.id, roomUpdate.membership.toString().split('.').last); await ensureRoomExists(clientId, roomUpdate.id,
roomUpdate.membership.toString().split('.').last);
_ensuredRooms.add(setKey); _ensuredRooms.add(setKey);
} }
} else { } else {
@ -161,24 +173,37 @@ class Database extends _$Database {
var doUpdate = oldRoom == null; var doUpdate = oldRoom == null;
if (!doUpdate) { if (!doUpdate) {
doUpdate = roomUpdate.highlight_count != oldRoom.highlightCount || doUpdate = roomUpdate.highlight_count != oldRoom.highlightCount ||
roomUpdate.notification_count != oldRoom.notificationCount || roomUpdate.notification_count != oldRoom.notificationCount ||
roomUpdate.membership.toString().split('.').last != oldRoom.membership.toString().split('.').last || roomUpdate.membership.toString().split('.').last !=
(roomUpdate.summary?.mJoinedMemberCount != null && oldRoom.membership.toString().split('.').last ||
roomUpdate.summary.mJoinedMemberCount != oldRoom.mInvitedMemberCount) || (roomUpdate.summary?.mJoinedMemberCount != null &&
(roomUpdate.summary?.mInvitedMemberCount != null && roomUpdate.summary.mJoinedMemberCount !=
roomUpdate.summary.mJoinedMemberCount != oldRoom.mJoinedMemberCount) || oldRoom.mInvitedMemberCount) ||
(roomUpdate.summary?.mHeroes != null && (roomUpdate.summary?.mInvitedMemberCount != null &&
roomUpdate.summary.mHeroes.join(',') != oldRoom.mHeroes.join(',')); roomUpdate.summary.mJoinedMemberCount !=
oldRoom.mJoinedMemberCount) ||
(roomUpdate.summary?.mHeroes != null &&
roomUpdate.summary.mHeroes.join(',') !=
oldRoom.mHeroes.join(','));
} }
if (doUpdate) { if (doUpdate) {
await (update(rooms)..where((r) => r.roomId.equals(roomUpdate.id) & r.clientId.equals(clientId))).write(RoomsCompanion( await (update(rooms)
..where((r) =>
r.roomId.equals(roomUpdate.id) & r.clientId.equals(clientId)))
.write(RoomsCompanion(
highlightCount: Value(roomUpdate.highlight_count), highlightCount: Value(roomUpdate.highlight_count),
notificationCount: Value(roomUpdate.notification_count), notificationCount: Value(roomUpdate.notification_count),
membership: Value(roomUpdate.membership.toString().split('.').last), membership: Value(roomUpdate.membership.toString().split('.').last),
joinedMemberCount: roomUpdate.summary?.mJoinedMemberCount != null ? Value(roomUpdate.summary.mJoinedMemberCount) : Value.absent(), joinedMemberCount: roomUpdate.summary?.mJoinedMemberCount != null
invitedMemberCount: roomUpdate.summary?.mInvitedMemberCount != null ? Value(roomUpdate.summary.mInvitedMemberCount) : Value.absent(), ? Value(roomUpdate.summary.mJoinedMemberCount)
heroes: roomUpdate.summary?.mHeroes != null ? Value(roomUpdate.summary.mHeroes.join(',')) : Value.absent(), : Value.absent(),
invitedMemberCount: roomUpdate.summary?.mInvitedMemberCount != null
? Value(roomUpdate.summary.mInvitedMemberCount)
: Value.absent(),
heroes: roomUpdate.summary?.mHeroes != null
? Value(roomUpdate.summary.mHeroes.join(','))
: Value.absent(),
)); ));
} }
@ -193,17 +218,24 @@ class Database extends _$Database {
/// Stores an UserUpdate object in the database. Must be called inside of /// Stores an UserUpdate object in the database. Must be called inside of
/// [transaction]. /// [transaction].
Future<void> storeUserEventUpdate(int clientId, sdk.UserUpdate userUpdate) async { Future<void> storeUserEventUpdate(
int clientId, sdk.UserUpdate userUpdate) async {
if (userUpdate.type == 'account_data') { if (userUpdate.type == 'account_data') {
await storeAccountData(clientId, userUpdate.eventType, json.encode(userUpdate.content['content'])); await storeAccountData(clientId, userUpdate.eventType,
json.encode(userUpdate.content['content']));
} else if (userUpdate.type == 'presence') { } else if (userUpdate.type == 'presence') {
await storePresence(clientId, userUpdate.eventType, userUpdate.content['sender'], json.encode(userUpdate.content['content'])); await storePresence(
clientId,
userUpdate.eventType,
userUpdate.content['sender'],
json.encode(userUpdate.content['content']));
} }
} }
/// Stores an EventUpdate object in the database. Must be called inside of /// Stores an EventUpdate object in the database. Must be called inside of
/// [transaction]. /// [transaction].
Future<void> storeEventUpdate(int clientId, sdk.EventUpdate eventUpdate) async { Future<void> storeEventUpdate(
int clientId, sdk.EventUpdate eventUpdate) async {
if (eventUpdate.type == 'ephemeral') return; if (eventUpdate.type == 'ephemeral') return;
final eventContent = eventUpdate.content; final eventContent = eventUpdate.content;
final type = eventUpdate.type; final type = eventUpdate.type;
@ -227,11 +259,13 @@ class Database extends _$Database {
eventContent['unsigned'] is Map<String, dynamic> && eventContent['unsigned'] is Map<String, dynamic> &&
eventContent['unsigned']['transaction_id'] is String) { eventContent['unsigned']['transaction_id'] is String) {
// status changed and we have an old transaction id --> update event id and stuffs // status changed and we have an old transaction id --> update event id and stuffs
await updateEventStatus(status, eventContent['event_id'], clientId, eventContent['unsigned']['transaction_id'], chatId); await updateEventStatus(status, eventContent['event_id'], clientId,
eventContent['unsigned']['transaction_id'], chatId);
} else { } else {
DbEvent oldEvent; DbEvent oldEvent;
if (type == 'history') { if (type == 'history') {
final allOldEvents = await getEvent(clientId, eventContent['event_id'], chatId).get(); final allOldEvents =
await getEvent(clientId, eventContent['event_id'], chatId).get();
if (allOldEvents.isNotEmpty) { if (allOldEvents.isNotEmpty) {
oldEvent = allOldEvents.first; oldEvent = allOldEvents.first;
} }
@ -241,7 +275,10 @@ class Database extends _$Database {
eventContent['event_id'], eventContent['event_id'],
chatId, chatId,
oldEvent?.sortOrder ?? eventUpdate.sortOrder, oldEvent?.sortOrder ?? eventUpdate.sortOrder,
eventContent['origin_server_ts'] != null ? DateTime.fromMillisecondsSinceEpoch(eventContent['origin_server_ts']) : DateTime.now(), eventContent['origin_server_ts'] != null
? DateTime.fromMillisecondsSinceEpoch(
eventContent['origin_server_ts'])
: DateTime.now(),
eventContent['sender'], eventContent['sender'],
eventContent['type'], eventContent['type'],
json.encode(eventContent['unsigned'] ?? ''), json.encode(eventContent['unsigned'] ?? ''),
@ -256,7 +293,8 @@ class Database extends _$Database {
if (status != -1 && if (status != -1 &&
eventUpdate.content.containsKey('unsigned') && eventUpdate.content.containsKey('unsigned') &&
eventUpdate.content['unsigned']['transaction_id'] is String) { eventUpdate.content['unsigned']['transaction_id'] is String) {
await removeEvent(clientId, eventUpdate.content['unsigned']['transaction_id'], chatId); await removeEvent(clientId,
eventUpdate.content['unsigned']['transaction_id'], chatId);
} }
} }
@ -269,7 +307,10 @@ class Database extends _$Database {
eventContent['event_id'] ?? now.millisecondsSinceEpoch.toString(), eventContent['event_id'] ?? now.millisecondsSinceEpoch.toString(),
chatId, chatId,
eventUpdate.sortOrder ?? 0.0, eventUpdate.sortOrder ?? 0.0,
eventContent['origin_server_ts'] != null ? DateTime.fromMillisecondsSinceEpoch(eventContent['origin_server_ts']) : now, eventContent['origin_server_ts'] != null
? DateTime.fromMillisecondsSinceEpoch(
eventContent['origin_server_ts'])
: now,
eventContent['sender'], eventContent['sender'],
eventContent['type'], eventContent['type'],
json.encode(eventContent['unsigned'] ?? ''), json.encode(eventContent['unsigned'] ?? ''),
@ -287,7 +328,8 @@ class Database extends _$Database {
} }
} }
Future<sdk.Event> getEventById(int clientId, String eventId, sdk.Room room) async { Future<sdk.Event> getEventById(
int clientId, String eventId, sdk.Room room) async {
final event = await getEvent(clientId, eventId, room.id).get(); final event = await getEvent(clientId, eventId, room.id).get();
if (event.isEmpty) { if (event.isEmpty) {
return null; return null;
@ -296,7 +338,9 @@ class Database extends _$Database {
} }
Future<bool> redactMessage(int clientId, sdk.EventUpdate eventUpdate) async { Future<bool> redactMessage(int clientId, sdk.EventUpdate eventUpdate) async {
final events = await getEvent(clientId, eventUpdate.content['redacts'], eventUpdate.roomID).get(); final events = await getEvent(
clientId, eventUpdate.content['redacts'], eventUpdate.roomID)
.get();
var success = false; var success = false;
for (final dbEvent in events) { for (final dbEvent in events) {
final event = sdk.Event.fromDb(dbEvent, null); final event = sdk.Event.fromDb(dbEvent, null);
@ -325,29 +369,44 @@ class Database extends _$Database {
Future<void> forgetRoom(int clientId, String roomId) async { Future<void> forgetRoom(int clientId, String roomId) async {
final setKey = '${clientId};${roomId}'; final setKey = '${clientId};${roomId}';
_ensuredRooms.remove(setKey); _ensuredRooms.remove(setKey);
await (delete(rooms)..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId))).go(); await (delete(rooms)
await (delete(events)..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId))).go(); ..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId)))
await (delete(roomStates)..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId))).go(); .go();
await (delete(roomAccountData)..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId))).go(); await (delete(events)
..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId)))
.go();
await (delete(roomStates)
..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId)))
.go();
await (delete(roomAccountData)
..where((r) => r.roomId.equals(roomId) & r.clientId.equals(clientId)))
.go();
} }
Future<void> clearCache(int clientId) async { Future<void> clearCache(int clientId) async {
await (delete(presences)..where((r) => r.clientId.equals(clientId))).go(); await (delete(presences)..where((r) => r.clientId.equals(clientId))).go();
await (delete(roomAccountData)..where((r) => r.clientId.equals(clientId))).go(); await (delete(roomAccountData)..where((r) => r.clientId.equals(clientId)))
.go();
await (delete(accountData)..where((r) => r.clientId.equals(clientId))).go(); await (delete(accountData)..where((r) => r.clientId.equals(clientId))).go();
await (delete(roomStates)..where((r) => r.clientId.equals(clientId))).go(); await (delete(roomStates)..where((r) => r.clientId.equals(clientId))).go();
await (delete(events)..where((r) => r.clientId.equals(clientId))).go(); await (delete(events)..where((r) => r.clientId.equals(clientId))).go();
await (delete(rooms)..where((r) => r.clientId.equals(clientId))).go(); await (delete(rooms)..where((r) => r.clientId.equals(clientId))).go();
await (delete(outboundGroupSessions)..where((r) => r.clientId.equals(clientId))).go(); await (delete(outboundGroupSessions)
..where((r) => r.clientId.equals(clientId)))
.go();
await storePrevBatch(null, clientId); await storePrevBatch(null, clientId);
} }
Future<void> clear(int clientId) async { Future<void> clear(int clientId) async {
await clearCache(clientId); await clearCache(clientId);
await (delete(inboundGroupSessions)..where((r) => r.clientId.equals(clientId))).go(); await (delete(inboundGroupSessions)
..where((r) => r.clientId.equals(clientId)))
.go();
await (delete(olmSessions)..where((r) => r.clientId.equals(clientId))).go(); await (delete(olmSessions)..where((r) => r.clientId.equals(clientId))).go();
await (delete(userDeviceKeysKey)..where((r) => r.clientId.equals(clientId))).go(); await (delete(userDeviceKeysKey)..where((r) => r.clientId.equals(clientId)))
await (delete(userDeviceKeys)..where((r) => r.clientId.equals(clientId))).go(); .go();
await (delete(userDeviceKeys)..where((r) => r.clientId.equals(clientId)))
.go();
await (delete(clients)..where((r) => r.clientId.equals(clientId))).go(); await (delete(clients)..where((r) => r.clientId.equals(clientId))).go();
} }

View file

@ -42,7 +42,15 @@ class Presence {
final String statusMsg; final String statusMsg;
final DateTime time; final DateTime time;
Presence({this.sender, this.displayname, this.avatarUrl, this.currentlyActive, this.lastActiveAgo, this.presence, this.statusMsg, this.time}); Presence(
{this.sender,
this.displayname,
this.avatarUrl,
this.currentlyActive,
this.lastActiveAgo,
this.presence,
this.statusMsg,
this.time});
Presence.fromJson(Map<String, dynamic> json) Presence.fromJson(Map<String, dynamic> json)
: sender = json['sender'], : sender = json['sender'],
@ -66,15 +74,16 @@ class Presence {
return Presence( return Presence(
sender: dbEntry.sender, sender: dbEntry.sender,
displayname: content['displayname'], displayname: content['displayname'],
avatarUrl: content['avatar_url'] != null ? Uri.parse(content['avatar_url']) : null, avatarUrl: content['avatar_url'] != null
? Uri.parse(content['avatar_url'])
: null,
currentlyActive: content['currently_active'], currentlyActive: content['currently_active'],
lastActiveAgo: content['last_active_ago'], lastActiveAgo: content['last_active_ago'],
time: DateTime.fromMillisecondsSinceEpoch( time: DateTime.fromMillisecondsSinceEpoch(
DateTime.now().millisecondsSinceEpoch - DateTime.now().millisecondsSinceEpoch -
(content['last_active_ago'] ?? 0)), (content['last_active_ago'] ?? 0)),
presence: PresenceType.values.firstWhere( presence: PresenceType.values.firstWhere(
(e) => (e) => e.toString() == "PresenceType.${content['presence']}",
e.toString() == "PresenceType.${content['presence']}",
orElse: () => null), orElse: () => null),
statusMsg: content['status_msg'], statusMsg: content['status_msg'],
); );

View file

@ -112,7 +112,8 @@ class Room {
} }
Future<void> updateSortOrder() async { Future<void> updateSortOrder() async {
await client.database?.updateRoomSortOrder(_oldestSortOrder, _newestSortOrder, client.id, id); await client.database?.updateRoomSortOrder(
_oldestSortOrder, _newestSortOrder, client.id, id);
} }
/// Clears the existing outboundGroupSession, tries to create a new one and /// Clears the existing outboundGroupSession, tries to create a new one and
@ -165,10 +166,12 @@ class Room {
Future<void> _storeOutboundGroupSession() async { Future<void> _storeOutboundGroupSession() async {
if (_outboundGroupSession == null) return; if (_outboundGroupSession == null) return;
await client.database?.storeOutboundGroupSession( await client.database?.storeOutboundGroupSession(
client.id, id, _outboundGroupSession.pickle(client.userID), client.id,
json.encode(_outboundGroupSessionDevices), _outboundGroupSessionCreationTime, id,
_outboundGroupSessionSentMessages _outboundGroupSession.pickle(client.userID),
); json.encode(_outboundGroupSessionDevices),
_outboundGroupSessionCreationTime,
_outboundGroupSessionSentMessages);
return; return;
} }
@ -190,21 +193,27 @@ class Room {
} }
// next check if it needs to be rotated // next check if it needs to be rotated
final encryptionContent = getState('m.room.encryption')?.content; final encryptionContent = getState('m.room.encryption')?.content;
final maxMessages = encryptionContent != null && encryptionContent['rotation_period_msgs'] is int final maxMessages = encryptionContent != null &&
? encryptionContent['rotation_period_msgs'] encryptionContent['rotation_period_msgs'] is int
: 100; ? encryptionContent['rotation_period_msgs']
final maxAge = encryptionContent != null && encryptionContent['rotation_period_ms'] is int : 100;
? encryptionContent['rotation_period_ms'] final maxAge = encryptionContent != null &&
: 604800000; // default of one week encryptionContent['rotation_period_ms'] is int
? encryptionContent['rotation_period_ms']
: 604800000; // default of one week
if (_outboundGroupSessionSentMessages >= maxMessages || if (_outboundGroupSessionSentMessages >= maxMessages ||
_outboundGroupSessionCreationTime.add(Duration(milliseconds: maxAge)).isBefore(DateTime.now())) { _outboundGroupSessionCreationTime
.add(Duration(milliseconds: maxAge))
.isBefore(DateTime.now())) {
wipe = true; wipe = true;
} }
if (!wipe) { if (!wipe) {
return false; return false;
} }
} }
if (!wipe && _outboundGroupSessionDevices == null && _outboundGroupSession == null) { if (!wipe &&
_outboundGroupSessionDevices == null &&
_outboundGroupSession == null) {
return true; // let's just short-circuit out of here, no need to do DB stuff return true; // let's just short-circuit out of here, no need to do DB stuff
} }
_outboundGroupSessionDevices = null; _outboundGroupSessionDevices = null;
@ -250,8 +259,12 @@ class Room {
indexes: {}, indexes: {},
key: client.userID, key: client.userID,
); );
client.database?.storeInboundGroupSession(client.id, id, sessionId, client.database?.storeInboundGroupSession(
inboundGroupSession.pickle(client.userID), json.encode(content), client.id,
id,
sessionId,
inboundGroupSession.pickle(client.userID),
json.encode(content),
json.encode({}), json.encode({}),
); );
_tryAgainDecryptLastMessage(); _tryAgainDecryptLastMessage();
@ -398,8 +411,7 @@ class Room {
states.forEach((final String key, final entry) { states.forEach((final String key, final entry) {
if (!entry.containsKey('')) return; if (!entry.containsKey('')) return;
final Event state = entry['']; final Event state = entry[''];
if (state.sortOrder != null && if (state.sortOrder != null && state.sortOrder > lastSortOrder) {
state.sortOrder > lastSortOrder) {
lastSortOrder = state.sortOrder; lastSortOrder = state.sortOrder;
lastEvent = state; lastEvent = state;
} }
@ -436,7 +448,8 @@ class Room {
this.roomAccountData = const {}, this.roomAccountData = const {},
double newestSortOrder = 0.0, double newestSortOrder = 0.0,
double oldestSortOrder = 0.0, double oldestSortOrder = 0.0,
}) : _newestSortOrder = newestSortOrder, _oldestSortOrder = oldestSortOrder; }) : _newestSortOrder = newestSortOrder,
_oldestSortOrder = oldestSortOrder;
/// The default count of how much events should be requested when requesting the /// The default count of how much events should be requested when requesting the
/// history of this room. /// history of this room.
@ -524,7 +537,8 @@ class Room {
name = name.replaceAll(RegExp(r'[^\w-]'), ''); name = name.replaceAll(RegExp(r'[^\w-]'), '');
return name.toLowerCase(); return name.toLowerCase();
}; };
final addEmotePack = (String packName, Map<String, dynamic> content, [String packNameOverride]) { final addEmotePack = (String packName, Map<String, dynamic> content,
[String packNameOverride]) {
if (!(content['short'] is Map)) { if (!(content['short'] is Map)) {
return; return;
} }
@ -572,7 +586,10 @@ class Room {
final stateKey = stateKeyEntry.key; final stateKey = stateKeyEntry.key;
final event = room.getState('im.ponies.room_emotes', stateKey); final event = room.getState('im.ponies.room_emotes', stateKey);
if (event != null && stateKeyEntry.value is Map) { if (event != null && stateKeyEntry.value is Map) {
addEmotePack(room.canonicalAlias.isEmpty ? room.id : canonicalAlias, event.content, stateKeyEntry.value['name']); addEmotePack(
room.canonicalAlias.isEmpty ? room.id : canonicalAlias,
event.content,
stateKeyEntry.value['name']);
} }
} }
} }
@ -583,7 +600,11 @@ class Room {
/// Sends a normal text message to this room. Returns the event ID generated /// Sends a normal text message to this room. Returns the event ID generated
/// by the server for this message. /// by the server for this message.
Future<String> sendTextEvent(String message, {String txid, Event inReplyTo, bool parseMarkdown = true, Map<String, Map<String, String>> emotePacks}) { Future<String> sendTextEvent(String message,
{String txid,
Event inReplyTo,
bool parseMarkdown = true,
Map<String, Map<String, String>> emotePacks}) {
final event = <String, dynamic>{ final event = <String, dynamic>{
'msgtype': 'm.text', 'msgtype': 'm.text',
'body': message, 'body': message,
@ -836,7 +857,11 @@ class Room {
final sortOrder = newSortOrder; final sortOrder = newSortOrder;
// Display a *sending* event and store it. // Display a *sending* event and store it.
var eventUpdate = EventUpdate(type: 'timeline', roomID: id, eventType: type, sortOrder: sortOrder, var eventUpdate = EventUpdate(
type: 'timeline',
roomID: id,
eventType: type,
sortOrder: sortOrder,
content: { content: {
'type': type, 'type': type,
'event_id': messageID, 'event_id': messageID,
@ -994,7 +1019,8 @@ class Room {
final dbActions = <Future<dynamic> Function()>[]; final dbActions = <Future<dynamic> Function()>[];
if (client.database != null) { if (client.database != null) {
dbActions.add(() => client.database.setRoomPrevBatch(prev_batch, client.id, id)); dbActions.add(
() => client.database.setRoomPrevBatch(prev_batch, client.id, id));
} }
if (!(resp['chunk'] is List<dynamic> && if (!(resp['chunk'] is List<dynamic> &&
@ -1012,7 +1038,8 @@ class Room {
).decrypt(this); ).decrypt(this);
client.onEvent.add(eventUpdate); client.onEvent.add(eventUpdate);
if (client.database != null) { if (client.database != null) {
dbActions.add(() => client.database.storeEventUpdate(client.id, eventUpdate)); dbActions.add(
() => client.database.storeEventUpdate(client.id, eventUpdate));
} }
} }
} }
@ -1028,11 +1055,13 @@ class Room {
).decrypt(this); ).decrypt(this);
client.onEvent.add(eventUpdate); client.onEvent.add(eventUpdate);
if (client.database != null) { if (client.database != null) {
dbActions.add(() => client.database.storeEventUpdate(client.id, eventUpdate)); dbActions.add(
() => client.database.storeEventUpdate(client.id, eventUpdate));
} }
} }
if (client.database != null) { if (client.database != null) {
dbActions.add(() => client.database.setRoomPrevBatch(resp['end'], client.id, id)); dbActions.add(
() => client.database.setRoomPrevBatch(resp['end'], client.id, id));
} }
await client.database?.transaction(() async { await client.database?.transaction(() async {
for (final f in dbActions) { for (final f in dbActions) {
@ -1105,12 +1134,12 @@ class Room {
/// Returns a Room from a json String which comes normally from the store. If the /// Returns a Room from a json String which comes normally from the store. If the
/// state are also given, the method will await them. /// state are also given, the method will await them.
static Future<Room> getRoomFromTableRow( static Future<Room> getRoomFromTableRow(
DbRoom row, // either Map<String, dynamic> or DbRoom DbRoom row, // either Map<String, dynamic> or DbRoom
Client matrix, Client matrix, {
{ dynamic states, // DbRoomState, as iterator and optionally as future
dynamic states, // DbRoomState, as iterator and optionally as future dynamic
dynamic roomAccountData, // DbRoomAccountData, as iterator and optionally as future roomAccountData, // DbRoomAccountData, as iterator and optionally as future
}) async { }) async {
final newRoom = Room( final newRoom = Room(
id: row.roomId, id: row.roomId,
membership: Membership.values membership: Membership.values
@ -1136,7 +1165,8 @@ class Room {
rawStates = states; rawStates = states;
} }
for (final rawState in rawStates) { for (final rawState in rawStates) {
final newState = Event.fromDb(rawState, newRoom);; final newState = Event.fromDb(rawState, newRoom);
;
newRoom.setState(newState); newRoom.setState(newState);
} }
} }
@ -1179,7 +1209,8 @@ class Room {
await events[i].loadSession(); await events[i].loadSession();
events[i] = events[i].decrypted; events[i] = events[i].decrypted;
if (events[i].type != EventTypes.Encrypted) { if (events[i].type != EventTypes.Encrypted) {
await client.database.storeEventUpdate(client.id, await client.database.storeEventUpdate(
client.id,
EventUpdate( EventUpdate(
eventType: events[i].typeKey, eventType: events[i].typeKey,
content: events[i].toJson(), content: events[i].toJson(),
@ -1301,7 +1332,8 @@ class Room {
'content': resp, 'content': resp,
'state_key': mxID, 'state_key': mxID,
}; };
await client.database.storeEventUpdate(client.id, await client.database.storeEventUpdate(
client.id,
EventUpdate( EventUpdate(
content: content, content: content,
roomID: id, roomID: id,
@ -1748,7 +1780,8 @@ class Room {
var users = await requestParticipants(); var users = await requestParticipants();
for (final user in users) { for (final user in users) {
if (client.userDeviceKeys.containsKey(user.id)) { if (client.userDeviceKeys.containsKey(user.id)) {
for (var deviceKeyEntry in client.userDeviceKeys[user.id].deviceKeys.values) { for (var deviceKeyEntry
in client.userDeviceKeys[user.id].deviceKeys.values) {
deviceKeys.add(deviceKeyEntry); deviceKeys.add(deviceKeyEntry);
} }
} }
@ -1762,14 +1795,14 @@ class Room {
if (_restoredOutboundGroupSession || client.database == null) { if (_restoredOutboundGroupSession || client.database == null) {
return; return;
} }
final outboundSession = await client.database.getDbOutboundGroupSession(client.id, id); final outboundSession =
await client.database.getDbOutboundGroupSession(client.id, id);
if (outboundSession != null) { if (outboundSession != null) {
try { try {
_outboundGroupSession = olm.OutboundGroupSession(); _outboundGroupSession = olm.OutboundGroupSession();
_outboundGroupSession.unpickle( _outboundGroupSession.unpickle(client.userID, outboundSession.pickle);
client.userID, outboundSession.pickle);
_outboundGroupSessionDevices = _outboundGroupSessionDevices =
List<String>.from(json.decode(outboundSession.deviceIds)); List<String>.from(json.decode(outboundSession.deviceIds));
_outboundGroupSessionCreationTime = outboundSession.creationTime; _outboundGroupSessionCreationTime = outboundSession.creationTime;
_outboundGroupSessionSentMessages = outboundSession.sentMessages; _outboundGroupSessionSentMessages = outboundSession.sentMessages;
} catch (e) { } catch (e) {
@ -1825,46 +1858,53 @@ class Room {
Future<void> requestSessionKey(String sessionId, String senderKey) async { Future<void> requestSessionKey(String sessionId, String senderKey) async {
final users = await requestParticipants(); final users = await requestParticipants();
await client.sendToDevice( await client.sendToDevice(
[], [],
'm.room_key_request', 'm.room_key_request',
{ {
'action': 'request_cancellation', 'action': 'request_cancellation',
'request_id': base64.encode(utf8.encode(sessionId)), 'request_id': base64.encode(utf8.encode(sessionId)),
'requesting_device_id': client.deviceID, 'requesting_device_id': client.deviceID,
},
encrypted: false,
toUsers: users);
await client.sendToDevice(
[],
'm.room_key_request',
{
'action': 'request',
'body': {
'algorithm': 'm.megolm.v1.aes-sha2',
'room_id': id,
'sender_key': senderKey,
'session_id': sessionId,
}, },
'request_id': base64.encode(utf8.encode(sessionId)), encrypted: false,
'requesting_device_id': client.deviceID, toUsers: users);
}, await client.sendToDevice(
encrypted: false, [],
toUsers: users); 'm.room_key_request',
{
'action': 'request',
'body': {
'algorithm': 'm.megolm.v1.aes-sha2',
'room_id': id,
'sender_key': senderKey,
'session_id': sessionId,
},
'request_id': base64.encode(utf8.encode(sessionId)),
'requesting_device_id': client.deviceID,
},
encrypted: false,
toUsers: users);
} }
Future<void> loadInboundGroupSessionKey(String sessionId, [String senderKey]) async { Future<void> loadInboundGroupSessionKey(String sessionId,
if (sessionId == null || inboundGroupSessions.containsKey(sessionId)) return; // nothing to do [String senderKey]) async {
final session = await client.database.getDbInboundGroupSession(client.id, id, sessionId); if (sessionId == null || inboundGroupSessions.containsKey(sessionId)) {
return;
} // nothing to do
final session = await client.database
.getDbInboundGroupSession(client.id, id, sessionId);
if (session == null) { if (session == null) {
// no session found, let's request it! // no session found, let's request it!
if (client.enableE2eeRecovery && !_requestedSessionIds.contains(sessionId) && senderKey != null) { if (client.enableE2eeRecovery &&
!_requestedSessionIds.contains(sessionId) &&
senderKey != null) {
unawaited(requestSessionKey(sessionId, senderKey)); unawaited(requestSessionKey(sessionId, senderKey));
_requestedSessionIds.add(sessionId); _requestedSessionIds.add(sessionId);
} }
return; return;
} }
try { try {
_inboundGroupSessions[sessionId] = SessionKey.fromDb(session, client.userID); _inboundGroupSessions[sessionId] =
SessionKey.fromDb(session, client.userID);
} catch (e) { } catch (e) {
print('[LibOlm] Could not unpickle inboundGroupSession: ' + e.toString()); print('[LibOlm] Could not unpickle inboundGroupSession: ' + e.toString());
} }
@ -1887,7 +1927,8 @@ class Room {
/// Returns a m.bad.encrypted event if it fails and does nothing if the event /// Returns a m.bad.encrypted event if it fails and does nothing if the event
/// was not encrypted. /// was not encrypted.
Event decryptGroupMessage(Event event) { Event decryptGroupMessage(Event event) {
if (event.type != EventTypes.Encrypted || event.content['ciphertext'] == null) return event; if (event.type != EventTypes.Encrypted ||
event.content['ciphertext'] == null) return event;
Map<String, dynamic> decryptedPayload; Map<String, dynamic> decryptedPayload;
try { try {
if (!client.encryptionEnabled) { if (!client.encryptionEnabled) {
@ -1905,7 +1946,9 @@ class Room {
.decrypt(event.content['ciphertext']); .decrypt(event.content['ciphertext']);
final messageIndexKey = final messageIndexKey =
event.eventId + event.time.millisecondsSinceEpoch.toString(); event.eventId + event.time.millisecondsSinceEpoch.toString();
if (inboundGroupSessions[sessionId].indexes.containsKey(messageIndexKey) && if (inboundGroupSessions[sessionId]
.indexes
.containsKey(messageIndexKey) &&
inboundGroupSessions[sessionId].indexes[messageIndexKey] != inboundGroupSessions[sessionId].indexes[messageIndexKey] !=
decryptResult.message_index) { decryptResult.message_index) {
if ((_outboundGroupSession?.session_id() ?? '') == sessionId) { if ((_outboundGroupSession?.session_id() ?? '') == sessionId) {
@ -1919,11 +1962,17 @@ class Room {
// the entry should always exist. In the case it doesn't, the following // the entry should always exist. In the case it doesn't, the following
// line *could* throw an error. As that is a future, though, and we call // line *could* throw an error. As that is a future, though, and we call
// it un-awaited here, nothing happens, which is exactly the result we want // it un-awaited here, nothing happens, which is exactly the result we want
client.database?.updateInboundGroupSessionIndexes(json.encode(inboundGroupSessions[sessionId].indexes), client.id, id, sessionId); client.database?.updateInboundGroupSessionIndexes(
json.encode(inboundGroupSessions[sessionId].indexes),
client.id,
id,
sessionId);
decryptedPayload = json.decode(decryptResult.plaintext); decryptedPayload = json.decode(decryptResult.plaintext);
} catch (exception) { } catch (exception) {
// alright, if this was actually by our own outbound group session, we might as well clear it // alright, if this was actually by our own outbound group session, we might as well clear it
if (client.enableE2eeRecovery && (_outboundGroupSession?.session_id() ?? '') == event.content['session_id']) { if (client.enableE2eeRecovery &&
(_outboundGroupSession?.session_id() ?? '') ==
event.content['session_id']) {
clearOutboundGroupSession(wipe: true); clearOutboundGroupSession(wipe: true);
} }
if (exception.toString() == DecryptError.UNKNOWN_SESSION) { if (exception.toString() == DecryptError.UNKNOWN_SESSION) {

View file

@ -43,7 +43,8 @@ class EventUpdate {
// the order where to stort this event // the order where to stort this event
final double sortOrder; final double sortOrder;
EventUpdate({this.eventType, this.roomID, this.type, this.content, this.sortOrder}); EventUpdate(
{this.eventType, this.roomID, this.type, this.content, this.sortOrder});
EventUpdate decrypt(Room room) { EventUpdate decrypt(Room room) {
if (eventType != 'm.room.encrypted') { if (eventType != 'm.room.encrypted') {

View file

@ -83,7 +83,9 @@ class Timeline {
// as r.limitedTimeline can be "null" sometimes, we need to check for == true // as r.limitedTimeline can be "null" sometimes, we need to check for == true
// as after receiving a limited timeline room update new events are expected // as after receiving a limited timeline room update new events are expected
// to be received via the onEvent stream, it is unneeded to call sortAndUpdate // to be received via the onEvent stream, it is unneeded to call sortAndUpdate
roomSub ??= room.client.onRoomUpdate.stream.where((r) => r.id == room.id && r.limitedTimeline == true).listen((r) => events.clear()); roomSub ??= room.client.onRoomUpdate.stream
.where((r) => r.id == room.id && r.limitedTimeline == true)
.listen((r) => events.clear());
sessionIdReceivedSub ??= sessionIdReceivedSub ??=
room.onSessionKeyReceived.stream.listen(_sessionKeyReceived); room.onSessionKeyReceived.stream.listen(_sessionKeyReceived);
} }
@ -129,8 +131,8 @@ class Timeline {
if (eventUpdate.eventType == 'm.room.redaction') { if (eventUpdate.eventType == 'm.room.redaction') {
final eventId = _findEvent(event_id: eventUpdate.content['redacts']); final eventId = _findEvent(event_id: eventUpdate.content['redacts']);
if (eventId != null) { if (eventId != null) {
events[eventId] events[eventId].setRedactionEvent(Event.fromJson(
.setRedactionEvent(Event.fromJson(eventUpdate.content, room, eventUpdate.sortOrder)); eventUpdate.content, room, eventUpdate.sortOrder));
} }
} else if (eventUpdate.content['status'] == -2) { } else if (eventUpdate.content['status'] == -2) {
var i = _findEvent(event_id: eventUpdate.content['event_id']); var i = _findEvent(event_id: eventUpdate.content['event_id']);
@ -146,18 +148,23 @@ class Timeline {
: null); : null);
if (i < events.length) { if (i < events.length) {
events[i] = Event.fromJson(eventUpdate.content, room, eventUpdate.sortOrder); events[i] = Event.fromJson(
eventUpdate.content, room, eventUpdate.sortOrder);
} }
} else { } else {
Event newEvent; Event newEvent;
var senderUser = room.getState('m.room.member', eventUpdate.content['sender'])?.asUser ?? await room.client.database var senderUser = room
?.getUser(room.client.id, eventUpdate.content['sender'], room); .getState('m.room.member', eventUpdate.content['sender'])
?.asUser ??
await room.client.database?.getUser(
room.client.id, eventUpdate.content['sender'], room);
if (senderUser != null) { if (senderUser != null) {
eventUpdate.content['displayname'] = senderUser.displayName; eventUpdate.content['displayname'] = senderUser.displayName;
eventUpdate.content['avatar_url'] = senderUser.avatarUrl.toString(); eventUpdate.content['avatar_url'] = senderUser.avatarUrl.toString();
} }
newEvent = Event.fromJson(eventUpdate.content, room, eventUpdate.sortOrder); newEvent =
Event.fromJson(eventUpdate.content, room, eventUpdate.sortOrder);
if (eventUpdate.type == 'history' && if (eventUpdate.type == 'history' &&
events.indexWhere( events.indexWhere(

View file

@ -10,7 +10,8 @@ class DeviceKeysList {
bool outdated = true; bool outdated = true;
Map<String, DeviceKeys> deviceKeys = {}; Map<String, DeviceKeys> deviceKeys = {};
DeviceKeysList.fromDb(DbUserDeviceKey dbEntry, List<DbUserDeviceKeysKey> childEntries) { DeviceKeysList.fromDb(
DbUserDeviceKey dbEntry, List<DbUserDeviceKeysKey> childEntries) {
userId = dbEntry.userId; userId = dbEntry.userId;
outdated = dbEntry.outdated; outdated = dbEntry.outdated;
deviceKeys = {}; deviceKeys = {};
@ -67,11 +68,16 @@ class DeviceKeys {
String get curve25519Key => keys['curve25519:$deviceId']; String get curve25519Key => keys['curve25519:$deviceId'];
String get ed25519Key => keys['ed25519:$deviceId']; String get ed25519Key => keys['ed25519:$deviceId'];
bool get isValid => userId != null && deviceId != null && curve25519Key != null && ed25519Key != null; bool get isValid =>
userId != null &&
deviceId != null &&
curve25519Key != null &&
ed25519Key != null;
Future<void> setVerified(bool newVerified, Client client) { Future<void> setVerified(bool newVerified, Client client) {
verified = newVerified; verified = newVerified;
return client.database?.setVerifiedUserDeviceKey(newVerified, client.id, userId, deviceId); return client.database
?.setVerifiedUserDeviceKey(newVerified, client.id, userId, deviceId);
} }
Future<void> setBlocked(bool newBlocked, Client client) { Future<void> setBlocked(bool newBlocked, Client client) {
@ -82,7 +88,8 @@ class DeviceKeys {
room.clearOutboundGroupSession(); room.clearOutboundGroupSession();
} }
} }
return client.database?.setBlockedUserDeviceKey(newBlocked, client.id, userId, deviceId); return client.database
?.setBlockedUserDeviceKey(newBlocked, client.id, userId, deviceId);
} }
DeviceKeys({ DeviceKeys({
@ -101,7 +108,9 @@ class DeviceKeys {
userId = dbEntry.userId; userId = dbEntry.userId;
deviceId = dbEntry.deviceId; deviceId = dbEntry.deviceId;
algorithms = content['algorithms'].cast<String>(); algorithms = content['algorithms'].cast<String>();
keys = content['keys'] != null ? Map<String, String>.from(content['keys']) : null; keys = content['keys'] != null
? Map<String, String>.from(content['keys'])
: null;
signatures = content['signatures'] != null signatures = content['signatures'] != null
? Map<String, dynamic>.from(content['signatures']) ? Map<String, dynamic>.from(content['signatures'])
: null; : null;
@ -147,7 +156,8 @@ class DeviceKeys {
} }
KeyVerification startVerification(Client client) { KeyVerification startVerification(Client client) {
final request = KeyVerification(client: client, userId: userId, deviceId: deviceId); final request =
KeyVerification(client: client, userId: userId, deviceId: deviceId);
request.start(); request.start();
client.addKeyVerificationRequest(request); client.addKeyVerificationRequest(request);
return request; return request;

View file

@ -42,7 +42,14 @@ import '../room.dart';
| | | |
*/ */
enum KeyVerificationState { askAccept, waitingAccept, askSas, waitingSas, done, error } enum KeyVerificationState {
askAccept,
waitingAccept,
askSas,
waitingSas,
done,
error
}
List<String> _intersect(List<String> a, List<dynamic> b) { List<String> _intersect(List<String> a, List<dynamic> b) {
final res = <String>[]; final res = <String>[];
@ -76,7 +83,8 @@ List<int> _bytesToInt(Uint8List bytes, int totalBits) {
final VERIFICATION_METHODS = [_KeyVerificationMethodSas.type]; final VERIFICATION_METHODS = [_KeyVerificationMethodSas.type];
_KeyVerificationMethod _makeVerificationMethod(String type, KeyVerification request) { _KeyVerificationMethod _makeVerificationMethod(
String type, KeyVerification request) {
if (type == _KeyVerificationMethodSas.type) { if (type == _KeyVerificationMethodSas.type) {
return _KeyVerificationMethodSas(request: request); return _KeyVerificationMethodSas(request: request);
} }
@ -89,7 +97,7 @@ class KeyVerification {
final Room room; final Room room;
final String userId; final String userId;
void Function() onUpdate; void Function() onUpdate;
String get deviceId => _deviceId; String get deviceId => _deviceId;
String _deviceId; String _deviceId;
bool startedVerification = false; bool startedVerification = false;
_KeyVerificationMethod method; _KeyVerificationMethod method;
@ -104,7 +112,8 @@ class KeyVerification {
String canceledCode; String canceledCode;
String canceledReason; String canceledReason;
KeyVerification({this.client, this.room, this.userId, String deviceId, this.onUpdate}) { KeyVerification(
{this.client, this.room, this.userId, String deviceId, this.onUpdate}) {
lastActivity = DateTime.now(); lastActivity = DateTime.now();
_deviceId ??= deviceId; _deviceId ??= deviceId;
} }
@ -115,9 +124,10 @@ class KeyVerification {
} }
static String getTransactionId(Map<String, dynamic> payload) { static String getTransactionId(Map<String, dynamic> payload) {
return payload['transaction_id'] ?? ( return payload['transaction_id'] ??
payload['m.relates_to'] is Map ? payload['m.relates_to']['event_id'] : null (payload['m.relates_to'] is Map
); ? payload['m.relates_to']['event_id']
: null);
} }
Future<void> start() async { Future<void> start() async {
@ -132,7 +142,8 @@ class KeyVerification {
setState(KeyVerificationState.waitingAccept); setState(KeyVerificationState.waitingAccept);
} }
Future<void> handlePayload(String type, Map<String, dynamic> payload, [String eventId]) async { Future<void> handlePayload(String type, Map<String, dynamic> payload,
[String eventId]) async {
print('[Key Verification] Received type ${type}: ' + payload.toString()); print('[Key Verification] Received type ${type}: ' + payload.toString());
try { try {
switch (type) { switch (type) {
@ -141,13 +152,16 @@ class KeyVerification {
transactionId ??= eventId ?? payload['transaction_id']; transactionId ??= eventId ?? payload['transaction_id'];
// verify the timestamp // verify the timestamp
final now = DateTime.now(); final now = DateTime.now();
final verifyTime = DateTime.fromMillisecondsSinceEpoch(payload['timestamp']); final verifyTime =
if (now.subtract(Duration(minutes: 10)).isAfter(verifyTime) || now.add(Duration(minutes: 5)).isBefore(verifyTime)) { DateTime.fromMillisecondsSinceEpoch(payload['timestamp']);
if (now.subtract(Duration(minutes: 10)).isAfter(verifyTime) ||
now.add(Duration(minutes: 5)).isBefore(verifyTime)) {
await cancel('m.timeout'); await cancel('m.timeout');
return; return;
} }
// verify it has a method we can use // verify it has a method we can use
possibleMethods = _intersect(VERIFICATION_METHODS, payload['methods']); possibleMethods =
_intersect(VERIFICATION_METHODS, payload['methods']);
if (possibleMethods.isEmpty) { if (possibleMethods.isEmpty) {
// reject it outright // reject it outright
await cancel('m.unknown_method'); await cancel('m.unknown_method');
@ -156,7 +170,8 @@ class KeyVerification {
setState(KeyVerificationState.askAccept); setState(KeyVerificationState.askAccept);
break; break;
case 'm.key.verification.ready': case 'm.key.verification.ready':
possibleMethods = _intersect(VERIFICATION_METHODS, payload['methods']); possibleMethods =
_intersect(VERIFICATION_METHODS, payload['methods']);
if (possibleMethods.isEmpty) { if (possibleMethods.isEmpty) {
// reject it outright // reject it outright
await cancel('m.unknown_method'); await cancel('m.unknown_method');
@ -214,7 +229,8 @@ class KeyVerification {
/// called when the user accepts an incoming verification /// called when the user accepts an incoming verification
Future<void> acceptVerification() async { Future<void> acceptVerification() async {
if (!(await verifyLastStep(['m.key.verification.request', 'm.key.verification.start']))) { if (!(await verifyLastStep(
['m.key.verification.request', 'm.key.verification.start']))) {
return; return;
} }
setState(KeyVerificationState.waitingAccept); setState(KeyVerificationState.waitingAccept);
@ -231,7 +247,8 @@ class KeyVerification {
/// called when the user rejects an incoming verification /// called when the user rejects an incoming verification
Future<void> rejectVerification() async { Future<void> rejectVerification() async {
if (!(await verifyLastStep(['m.key.verification.request', 'm.key.verification.start']))) { if (!(await verifyLastStep(
['m.key.verification.request', 'm.key.verification.start']))) {
return; return;
} }
await cancel('m.user'); await cancel('m.user');
@ -251,7 +268,9 @@ class KeyVerification {
List<int> get sasNumbers { List<int> get sasNumbers {
if (method is _KeyVerificationMethodSas) { if (method is _KeyVerificationMethodSas) {
return _bytesToInt((method as _KeyVerificationMethodSas).makeSas(5), 13).map((n) => n + 1000).toList(); return _bytesToInt((method as _KeyVerificationMethodSas).makeSas(5), 13)
.map((n) => n + 1000)
.toList();
} }
return []; return [];
} }
@ -265,13 +284,15 @@ class KeyVerification {
List<KeyVerificationEmoji> get sasEmojis { List<KeyVerificationEmoji> get sasEmojis {
if (method is _KeyVerificationMethodSas) { if (method is _KeyVerificationMethodSas) {
final numbers = _bytesToInt((method as _KeyVerificationMethodSas).makeSas(6), 6); final numbers =
_bytesToInt((method as _KeyVerificationMethodSas).makeSas(6), 6);
return numbers.map((n) => KeyVerificationEmoji(n)).toList().sublist(0, 7); return numbers.map((n) => KeyVerificationEmoji(n)).toList().sublist(0, 7);
} }
return []; return [];
} }
Future<void> verifyKeys(Map<String, String> keys, Future<bool> Function(String, DeviceKeys) verifier) async { Future<void> verifyKeys(Map<String, String> keys,
Future<bool> Function(String, DeviceKeys) verifier) async {
final verifiedDevices = <String>[]; final verifiedDevices = <String>[];
if (!client.userDeviceKeys.containsKey(userId)) { if (!client.userDeviceKeys.containsKey(userId)) {
@ -282,8 +303,10 @@ class KeyVerification {
final keyId = entry.key; final keyId = entry.key;
final verifyDeviceId = keyId.substring('ed25519:'.length); final verifyDeviceId = keyId.substring('ed25519:'.length);
final keyInfo = entry.value; final keyInfo = entry.value;
if (client.userDeviceKeys[userId].deviceKeys.containsKey(verifyDeviceId)) { if (client.userDeviceKeys[userId].deviceKeys
if (!(await verifier(keyInfo, client.userDeviceKeys[userId].deviceKeys[verifyDeviceId]))) { .containsKey(verifyDeviceId)) {
if (!(await verifier(keyInfo,
client.userDeviceKeys[userId].deviceKeys[verifyDeviceId]))) {
await cancel('m.key_mismatch'); await cancel('m.key_mismatch');
return; return;
} }
@ -295,12 +318,14 @@ class KeyVerification {
} }
// okay, we reached this far, so all the devices are verified! // okay, we reached this far, so all the devices are verified!
for (final verifyDeviceId in verifiedDevices) { for (final verifyDeviceId in verifiedDevices) {
await client.userDeviceKeys[userId].deviceKeys[verifyDeviceId].setVerified(true, client); await client.userDeviceKeys[userId].deviceKeys[verifyDeviceId]
.setVerified(true, client);
} }
} }
Future<bool> verifyActivity() async { Future<bool> verifyActivity() async {
if (lastActivity != null && lastActivity.add(Duration(minutes: 10)).isAfter(DateTime.now())) { if (lastActivity != null &&
lastActivity.add(Duration(minutes: 10)).isAfter(DateTime.now())) {
lastActivity = DateTime.now(); lastActivity = DateTime.now();
return true; return true;
} }
@ -351,7 +376,8 @@ class KeyVerification {
if (['m.key.verification.request'].contains(type)) { if (['m.key.verification.request'].contains(type)) {
payload['msgtype'] = type; payload['msgtype'] = type;
payload['to'] = userId; payload['to'] = userId;
payload['body'] = 'Attempting verification request. (${type}) Apparently your client doesn\'t support this'; payload['body'] =
'Attempting verification request. (${type}) Apparently your client doesn\'t support this';
type = 'm.room.message'; type = 'm.room.message';
} }
final newTransactionId = await room.sendEvent(payload, type: type); final newTransactionId = await room.sendEvent(payload, type: type);
@ -360,7 +386,8 @@ class KeyVerification {
client.addKeyVerificationRequest(this); client.addKeyVerificationRequest(this);
} }
} else { } else {
await client.sendToDevice([client.userDeviceKeys[userId].deviceKeys[deviceId]], type, payload); await client.sendToDevice(
[client.userDeviceKeys[userId].deviceKeys[deviceId]], type, payload);
} }
} }
@ -385,6 +412,7 @@ abstract class _KeyVerificationMethod {
bool validateStart(Map<String, dynamic> payload) { bool validateStart(Map<String, dynamic> payload) {
return false; return false;
} }
Future<void> sendStart(); Future<void> sendStart();
void dispose() {} void dispose() {}
} }
@ -395,7 +423,8 @@ const KNOWN_MESSAGE_AUTHENTIFICATION_CODES = ['hkdf-hmac-sha256'];
const KNOWN_AUTHENTICATION_TYPES = ['emoji', 'decimal']; const KNOWN_AUTHENTICATION_TYPES = ['emoji', 'decimal'];
class _KeyVerificationMethodSas extends _KeyVerificationMethod { class _KeyVerificationMethodSas extends _KeyVerificationMethod {
_KeyVerificationMethodSas({KeyVerification request}) : super(request: request); _KeyVerificationMethodSas({KeyVerification request})
: super(request: request);
static String type = 'm.sas.v1'; static String type = 'm.sas.v1';
@ -419,7 +448,8 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
try { try {
switch (type) { switch (type) {
case 'm.key.verification.start': case 'm.key.verification.start':
if (!(await request.verifyLastStep(['m.key.verification.request', 'm.key.verification.start']))) { if (!(await request.verifyLastStep(
['m.key.verification.request', 'm.key.verification.start']))) {
return; // abort return; // abort
} }
if (!validateStart(payload)) { if (!validateStart(payload)) {
@ -439,7 +469,8 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
await _sendKey(); await _sendKey();
break; break;
case 'm.key.verification.key': case 'm.key.verification.key':
if (!(await request.verifyLastStep(['m.key.verification.accept', 'm.key.verification.start']))) { if (!(await request.verifyLastStep(
['m.key.verification.accept', 'm.key.verification.start']))) {
return; return;
} }
_handleKey(payload); _handleKey(payload);
@ -506,7 +537,8 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
if (payload['method'] != type) { if (payload['method'] != type) {
return false; return false;
} }
final possibleKeyAgreementProtocols = _intersect(KNOWN_KEY_AGREEMENT_PROTOCOLS, payload['key_agreement_protocols']); final possibleKeyAgreementProtocols = _intersect(
KNOWN_KEY_AGREEMENT_PROTOCOLS, payload['key_agreement_protocols']);
if (possibleKeyAgreementProtocols.isEmpty) { if (possibleKeyAgreementProtocols.isEmpty) {
return false; return false;
} }
@ -516,12 +548,15 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
return false; return false;
} }
hash = possibleHashes.first; hash = possibleHashes.first;
final possibleMessageAuthenticationCodes = _intersect(KNOWN_MESSAGE_AUTHENTIFICATION_CODES, payload['message_authentication_codes']); final possibleMessageAuthenticationCodes = _intersect(
KNOWN_MESSAGE_AUTHENTIFICATION_CODES,
payload['message_authentication_codes']);
if (possibleMessageAuthenticationCodes.isEmpty) { if (possibleMessageAuthenticationCodes.isEmpty) {
return false; return false;
} }
messageAuthenticationCode = possibleMessageAuthenticationCodes.first; messageAuthenticationCode = possibleMessageAuthenticationCodes.first;
final possibleAuthenticationTypes = _intersect(KNOWN_AUTHENTICATION_TYPES, payload['short_authentication_string']); final possibleAuthenticationTypes = _intersect(
KNOWN_AUTHENTICATION_TYPES, payload['short_authentication_string']);
if (possibleAuthenticationTypes.isEmpty) { if (possibleAuthenticationTypes.isEmpty) {
return false; return false;
} }
@ -544,7 +579,8 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
} }
bool _handleAccept(Map<String, dynamic> payload) { bool _handleAccept(Map<String, dynamic> payload) {
if (!KNOWN_KEY_AGREEMENT_PROTOCOLS.contains(payload['key_agreement_protocol'])) { if (!KNOWN_KEY_AGREEMENT_PROTOCOLS
.contains(payload['key_agreement_protocol'])) {
return false; return false;
} }
keyAgreementProtocol = payload['key_agreement_protocol']; keyAgreementProtocol = payload['key_agreement_protocol'];
@ -552,11 +588,13 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
return false; return false;
} }
hash = payload['hash']; hash = payload['hash'];
if (!KNOWN_MESSAGE_AUTHENTIFICATION_CODES.contains(payload['message_authentication_code'])) { if (!KNOWN_MESSAGE_AUTHENTIFICATION_CODES
.contains(payload['message_authentication_code'])) {
return false; return false;
} }
messageAuthenticationCode = payload['message_authentication_code']; messageAuthenticationCode = payload['message_authentication_code'];
final possibleAuthenticationTypes = _intersect(KNOWN_AUTHENTICATION_TYPES, payload['short_authentication_string']); final possibleAuthenticationTypes = _intersect(
KNOWN_AUTHENTICATION_TYPES, payload['short_authentication_string']);
if (possibleAuthenticationTypes.isEmpty) { if (possibleAuthenticationTypes.isEmpty) {
return false; return false;
} }
@ -585,13 +623,23 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
Uint8List makeSas(int bytes) { Uint8List makeSas(int bytes) {
var sasInfo = ''; var sasInfo = '';
if (keyAgreementProtocol == 'curve25519-hkdf-sha256') { if (keyAgreementProtocol == 'curve25519-hkdf-sha256') {
final ourInfo = '${client.userID}|${client.deviceID}|${sas.get_pubkey()}|'; final ourInfo =
final theirInfo = '${request.userId}|${request.deviceId}|${theirPublicKey}|'; '${client.userID}|${client.deviceID}|${sas.get_pubkey()}|';
sasInfo = 'MATRIX_KEY_VERIFICATION_SAS|' + (request.startedVerification ? ourInfo + theirInfo : theirInfo + ourInfo) + request.transactionId; final theirInfo =
'${request.userId}|${request.deviceId}|${theirPublicKey}|';
sasInfo = 'MATRIX_KEY_VERIFICATION_SAS|' +
(request.startedVerification
? ourInfo + theirInfo
: theirInfo + ourInfo) +
request.transactionId;
} else if (keyAgreementProtocol == 'curve25519') { } else if (keyAgreementProtocol == 'curve25519') {
final ourInfo = client.userID + client.deviceID; final ourInfo = client.userID + client.deviceID;
final theirInfo = request.userId + request.deviceId; final theirInfo = request.userId + request.deviceId;
sasInfo = 'MATRIX_KEY_VERIFICATION_SAS' + (request.startedVerification ? ourInfo + theirInfo : theirInfo + ourInfo) + request.transactionId; sasInfo = 'MATRIX_KEY_VERIFICATION_SAS' +
(request.startedVerification
? ourInfo + theirInfo
: theirInfo + ourInfo) +
request.transactionId;
} else { } else {
throw 'Unknown key agreement protocol'; throw 'Unknown key agreement protocol';
} }
@ -600,9 +648,11 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
Future<void> _sendMac() async { Future<void> _sendMac() async {
final baseInfo = 'MATRIX_KEY_VERIFICATION_MAC' + final baseInfo = 'MATRIX_KEY_VERIFICATION_MAC' +
client.userID + client.deviceID + client.userID +
request.userId + request.deviceId + client.deviceID +
request.transactionId; request.userId +
request.deviceId +
request.transactionId;
final mac = <String, String>{}; final mac = <String, String>{};
final keyList = <String>[]; final keyList = <String>[];
@ -610,7 +660,8 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
// for now it is just our device key, once we have cross-signing // for now it is just our device key, once we have cross-signing
// we would also add the cross signing key here // we would also add the cross signing key here
final deviceKeyId = 'ed25519:${client.deviceID}'; final deviceKeyId = 'ed25519:${client.deviceID}';
mac[deviceKeyId] = _calculateMac(client.fingerprintKey, baseInfo + deviceKeyId); mac[deviceKeyId] =
_calculateMac(client.fingerprintKey, baseInfo + deviceKeyId);
keyList.add(deviceKeyId); keyList.add(deviceKeyId);
keyList.sort(); keyList.sort();
@ -624,13 +675,16 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
Future<void> _processMac() async { Future<void> _processMac() async {
final payload = macPayload; final payload = macPayload;
final baseInfo = 'MATRIX_KEY_VERIFICATION_MAC' + final baseInfo = 'MATRIX_KEY_VERIFICATION_MAC' +
request.userId + request.deviceId + request.userId +
client.userID + client.deviceID + request.deviceId +
request.transactionId; client.userID +
client.deviceID +
request.transactionId;
final keyList = payload['mac'].keys.toList(); final keyList = payload['mac'].keys.toList();
keyList.sort(); keyList.sort();
if (payload['keys'] != _calculateMac(keyList.join(','), baseInfo + 'KEY_IDS')) { if (payload['keys'] !=
_calculateMac(keyList.join(','), baseInfo + 'KEY_IDS')) {
await request.cancel('m.key_mismatch'); await request.cancel('m.key_mismatch');
return; return;
} }
@ -646,7 +700,9 @@ class _KeyVerificationMethodSas extends _KeyVerificationMethod {
} }
} }
await request.verifyKeys(mac, (String mac, DeviceKeys device) async { await request.verifyKeys(mac, (String mac, DeviceKeys device) async {
return mac == _calculateMac(device.ed25519Key, baseInfo + 'ed25519:' + device.deviceId); return mac ==
_calculateMac(
device.ed25519Key, baseInfo + 'ed25519:' + device.deviceId);
}); });
await request.send('m.key.verification.done', {}); await request.send('m.key.verification.done', {});
if (request.state != KeyVerificationState.error) { if (request.state != KeyVerificationState.error) {

View file

@ -13,11 +13,12 @@ class LinebreakSyntax extends InlineSyntax {
class SpoilerSyntax extends TagSyntax { class SpoilerSyntax extends TagSyntax {
Map<String, String> reasonMap = <String, String>{}; Map<String, String> reasonMap = <String, String>{};
SpoilerSyntax() : super( SpoilerSyntax()
r'\|\|(?:([^\|]+)\|(?!\|))?', : super(
requiresDelimiterRun: true, r'\|\|(?:([^\|]+)\|(?!\|))?',
end: r'\|\|', requiresDelimiterRun: true,
); end: r'\|\|',
);
@override @override
bool onMatch(InlineParser parser, Match match) { bool onMatch(InlineParser parser, Match match) {
@ -31,7 +32,8 @@ class SpoilerSyntax extends TagSyntax {
@override @override
bool onMatchEnd(InlineParser parser, Match match, TagState state) { bool onMatchEnd(InlineParser parser, Match match, TagState state) {
final element = Element('span', state.children); final element = Element('span', state.children);
element.attributes['data-mx-spoiler'] = htmlEscape.convert(reasonMap[match.input] ?? ''); element.attributes['data-mx-spoiler'] =
htmlEscape.convert(reasonMap[match.input] ?? '');
parser.addNode(element); parser.addNode(element);
return true; return true;
} }
@ -88,14 +90,33 @@ class PillSyntax extends InlineSyntax {
String markdown(String text, [Map<String, Map<String, String>> emotePacks]) { String markdown(String text, [Map<String, Map<String, String>> emotePacks]) {
emotePacks ??= <String, Map<String, String>>{}; emotePacks ??= <String, Map<String, String>>{};
var ret = markdownToHtml(text, var ret = markdownToHtml(
text,
extensionSet: ExtensionSet.commonMark, extensionSet: ExtensionSet.commonMark,
inlineSyntaxes: [StrikethroughSyntax(), LinebreakSyntax(), SpoilerSyntax(), EmoteSyntax(emotePacks), PillSyntax()], inlineSyntaxes: [
StrikethroughSyntax(),
LinebreakSyntax(),
SpoilerSyntax(),
EmoteSyntax(emotePacks),
PillSyntax()
],
); );
var stripPTags = '<p>'.allMatches(ret).length <= 1; var stripPTags = '<p>'.allMatches(ret).length <= 1;
if (stripPTags) { if (stripPTags) {
final otherBlockTags = ['table', 'pre', 'ol', 'ul', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'blockquote']; final otherBlockTags = [
'table',
'pre',
'ol',
'ul',
'h1',
'h2',
'h3',
'h4',
'h5',
'h6',
'blockquote'
];
for (final tag in otherBlockTags) { for (final tag in otherBlockTags) {
// we check for the close tag as the opening one might have attributes // we check for the close tag as the opening one might have attributes
if (ret.contains('</${tag}>')) { if (ret.contains('</${tag}>')) {

View file

@ -26,7 +26,8 @@ class RoomKeyRequest extends ToDeviceEvent {
var message = session.content; var message = session.content;
message['forwarding_curve25519_key_chain'] = forwardedKeys; message['forwarding_curve25519_key_chain'] = forwardedKeys;
message['session_key'] = session.inboundGroupSession.export_session(session.inboundGroupSession.first_known_index()); message['session_key'] = session.inboundGroupSession
.export_session(session.inboundGroupSession.first_known_index());
await client.sendToDevice( await client.sendToDevice(
[requestingDevice], [requestingDevice],
'm.forwarded_room_key', 'm.forwarded_room_key',

View file

@ -20,9 +20,8 @@ class SessionKey {
SessionKey.fromDb(DbInboundGroupSession dbEntry, String key) : key = key { SessionKey.fromDb(DbInboundGroupSession dbEntry, String key) : key = key {
final parsedContent = Event.getMapFromPayload(dbEntry.content); final parsedContent = Event.getMapFromPayload(dbEntry.content);
final parsedIndexes = Event.getMapFromPayload(dbEntry.indexes); final parsedIndexes = Event.getMapFromPayload(dbEntry.indexes);
content = parsedContent != null content =
? Map<String, dynamic>.from(parsedContent) parsedContent != null ? Map<String, dynamic>.from(parsedContent) : null;
: null;
indexes = parsedIndexes != null indexes = parsedIndexes != null
? Map<String, int>.from(parsedIndexes) ? Map<String, int>.from(parsedIndexes)
: <String, int>{}; : <String, int>{};

View file

@ -15,12 +15,15 @@ void main() {
}, },
}; };
test('simple markdown', () { test('simple markdown', () {
expect(markdown('hey *there* how are **you** doing?'), 'hey <em>there</em> how are <strong>you</strong> doing?'); expect(markdown('hey *there* how are **you** doing?'),
'hey <em>there</em> how are <strong>you</strong> doing?');
expect(markdown('wha ~~strike~~ works!'), 'wha <del>strike</del> works!'); expect(markdown('wha ~~strike~~ works!'), 'wha <del>strike</del> works!');
}); });
test('spoilers', () { test('spoilers', () {
expect(markdown('Snape killed ||Dumbledoor||'), 'Snape killed <span data-mx-spoiler="">Dumbledoor</span>'); expect(markdown('Snape killed ||Dumbledoor||'),
expect(markdown('Snape killed ||Story|Dumbledoor||'), 'Snape killed <span data-mx-spoiler="Story">Dumbledoor</span>'); 'Snape killed <span data-mx-spoiler="">Dumbledoor</span>');
expect(markdown('Snape killed ||Story|Dumbledoor||'),
'Snape killed <span data-mx-spoiler="Story">Dumbledoor</span>');
}); });
test('multiple paragraphs', () { test('multiple paragraphs', () {
expect(markdown('Heya!\n\nBeep'), '<p>Heya!</p>\n<p>Beep</p>'); expect(markdown('Heya!\n\nBeep'), '<p>Heya!</p>\n<p>Beep</p>');
@ -32,16 +35,22 @@ void main() {
expect(markdown('foxies\ncute'), 'foxies<br />\ncute'); expect(markdown('foxies\ncute'), 'foxies<br />\ncute');
}); });
test('emotes', () { test('emotes', () {
expect(markdown(':fox:', emotePacks), '<img src="mxc:&#47;&#47;roomfox" alt=":fox:" title=":fox:" height="32" vertical-align="middle" />'); expect(markdown(':fox:', emotePacks),
expect(markdown(':user~fox:', emotePacks), '<img src="mxc:&#47;&#47;userfox" alt=":fox:" title=":fox:" height="32" vertical-align="middle" />'); '<img src="mxc:&#47;&#47;roomfox" alt=":fox:" title=":fox:" height="32" vertical-align="middle" />');
expect(markdown(':raccoon:', emotePacks), '<img src="mxc:&#47;&#47;raccoon" alt=":raccoon:" title=":raccoon:" height="32" vertical-align="middle" />'); expect(markdown(':user~fox:', emotePacks),
'<img src="mxc:&#47;&#47;userfox" alt=":fox:" title=":fox:" height="32" vertical-align="middle" />');
expect(markdown(':raccoon:', emotePacks),
'<img src="mxc:&#47;&#47;raccoon" alt=":raccoon:" title=":raccoon:" height="32" vertical-align="middle" />');
expect(markdown(':invalid:', emotePacks), ':invalid:'); expect(markdown(':invalid:', emotePacks), ':invalid:');
expect(markdown(':room~invalid:', emotePacks), ':room~invalid:'); expect(markdown(':room~invalid:', emotePacks), ':room~invalid:');
}); });
test('pills', () { test('pills', () {
expect(markdown('Hey @sorunome:sorunome.de!'), 'Hey <a href="https://matrix.to/#/@sorunome:sorunome.de">@sorunome:sorunome.de</a>!'); expect(markdown('Hey @sorunome:sorunome.de!'),
expect(markdown('#fox:sorunome.de: you all are awesome'), '<a href="https://matrix.to/#/#fox:sorunome.de">#fox:sorunome.de</a>: you all are awesome'); 'Hey <a href="https://matrix.to/#/@sorunome:sorunome.de">@sorunome:sorunome.de</a>!');
expect(markdown('!blah:example.org'), '<a href="https://matrix.to/#/!blah:example.org">!blah:example.org</a>'); expect(markdown('#fox:sorunome.de: you all are awesome'),
'<a href="https://matrix.to/#/#fox:sorunome.de">#fox:sorunome.de</a>: you all are awesome');
expect(markdown('!blah:example.org'),
'<a href="https://matrix.to/#/!blah:example.org">!blah:example.org</a>');
}); });
}); });
} }

View file

@ -128,7 +128,8 @@ void test() async {
await Future.delayed(Duration(seconds: 5)); await Future.delayed(Duration(seconds: 5));
assert(room.outboundGroupSession != null); assert(room.outboundGroupSession != null);
var currentSessionIdA = room.outboundGroupSession.session_id(); var currentSessionIdA = room.outboundGroupSession.session_id();
assert(room.inboundGroupSessions.containsKey(room.outboundGroupSession.session_id())); assert(room.inboundGroupSessions
.containsKey(room.outboundGroupSession.session_id()));
assert(testClientA.olmSessions[testClientB.identityKey].length == 1); assert(testClientA.olmSessions[testClientB.identityKey].length == 1);
assert(testClientB.olmSessions[testClientA.identityKey].length == 1); assert(testClientB.olmSessions[testClientA.identityKey].length == 1);
assert(testClientA.olmSessions[testClientB.identityKey].first.session_id() == assert(testClientA.olmSessions[testClientB.identityKey].first.session_id() ==
@ -236,7 +237,8 @@ void test() async {
assert(restoredRoom.outboundGroupSession.session_id() == assert(restoredRoom.outboundGroupSession.session_id() ==
room.outboundGroupSession.session_id()); room.outboundGroupSession.session_id());
assert(restoredRoom.inboundGroupSessions.length == 4); assert(restoredRoom.inboundGroupSessions.length == 4);
assert(restoredRoom.inboundGroupSessions.length == room.inboundGroupSessions.length); assert(restoredRoom.inboundGroupSessions.length ==
room.inboundGroupSessions.length);
for (var i = 0; i < restoredRoom.inboundGroupSessions.length; i++) { for (var i = 0; i < restoredRoom.inboundGroupSessions.length; i++) {
assert(restoredRoom.inboundGroupSessions.keys.toList()[i] == assert(restoredRoom.inboundGroupSessions.keys.toList()[i] ==
room.inboundGroupSessions.keys.toList()[i]); room.inboundGroupSessions.keys.toList()[i]);