2020-08-05 22:08:55 +05:30
|
|
|
/*
|
|
|
|
Copyright 2020 Bruno Windels <bruno@windels.cloud>
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2020-08-28 18:06:00 +05:30
|
|
|
import {MEGOLM_ALGORITHM} from "../e2ee/common.js";
|
|
|
|
|
2020-09-14 20:04:07 +05:30
|
|
|
|
|
|
|
function applyTimelineEntries(data, timelineEntries, isInitialSync, isTimelineOpen, ownUserId) {
|
|
|
|
if (timelineEntries.length) {
|
|
|
|
data = timelineEntries.reduce((data, entry) => {
|
|
|
|
return processTimelineEvent(data, entry,
|
|
|
|
isInitialSync, isTimelineOpen, ownUserId);
|
|
|
|
}, data);
|
|
|
|
}
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function applySyncResponse(data, roomResponse, membership) {
|
2020-03-15 01:16:49 +05:30
|
|
|
if (roomResponse.summary) {
|
|
|
|
data = updateSummary(data, roomResponse.summary);
|
|
|
|
}
|
|
|
|
if (membership !== data.membership) {
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.membership = membership;
|
|
|
|
}
|
2020-08-28 00:22:51 +05:30
|
|
|
if (roomResponse.account_data) {
|
|
|
|
data = roomResponse.account_data.events.reduce(processRoomAccountData, data);
|
|
|
|
}
|
2020-09-23 17:56:14 +05:30
|
|
|
const stateEvents = roomResponse?.state?.events;
|
2020-03-15 01:16:49 +05:30
|
|
|
// state comes before timeline
|
2020-09-23 17:56:14 +05:30
|
|
|
if (Array.isArray(stateEvents)) {
|
|
|
|
data = stateEvents.reduce(processStateEvent, data);
|
2020-03-15 01:16:49 +05:30
|
|
|
}
|
2020-09-23 17:56:14 +05:30
|
|
|
const timelineEvents = roomResponse?.timeline?.events;
|
2020-09-14 20:04:07 +05:30
|
|
|
// process state events in timeline
|
|
|
|
// non-state events are handled by applyTimelineEntries
|
|
|
|
// so decryption is handled properly
|
2020-09-23 17:56:14 +05:30
|
|
|
if (Array.isArray(timelineEvents)) {
|
|
|
|
data = timelineEvents.reduce((data, event) => {
|
2020-09-14 20:04:07 +05:30
|
|
|
if (typeof event.state_key === "string") {
|
|
|
|
return processStateEvent(data, event);
|
2020-08-21 18:05:23 +05:30
|
|
|
}
|
2020-09-14 21:15:13 +05:30
|
|
|
return data;
|
2020-08-21 17:15:38 +05:30
|
|
|
}, data);
|
2020-03-15 01:16:49 +05:30
|
|
|
}
|
2020-08-20 20:37:02 +05:30
|
|
|
const unreadNotifications = roomResponse.unread_notifications;
|
|
|
|
if (unreadNotifications) {
|
|
|
|
data = data.cloneIfNeeded();
|
2020-08-21 19:20:32 +05:30
|
|
|
data.highlightCount = unreadNotifications.highlight_count || 0;
|
2020-08-20 20:37:02 +05:30
|
|
|
data.notificationCount = unreadNotifications.notification_count;
|
|
|
|
}
|
2020-03-15 01:16:49 +05:30
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2020-08-28 00:22:51 +05:30
|
|
|
function processRoomAccountData(data, event) {
|
|
|
|
if (event?.type === "m.tag") {
|
|
|
|
let tags = event?.content?.tags;
|
|
|
|
if (!tags || Array.isArray(tags) || typeof tags !== "object") {
|
|
|
|
tags = null;
|
|
|
|
}
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.tags = tags;
|
|
|
|
}
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2020-08-21 18:05:23 +05:30
|
|
|
function processStateEvent(data, event) {
|
2020-03-15 01:16:49 +05:30
|
|
|
if (event.type === "m.room.encryption") {
|
2020-08-28 18:06:00 +05:30
|
|
|
const algorithm = event.content?.algorithm;
|
|
|
|
if (!data.encryption && algorithm === MEGOLM_ALGORITHM) {
|
2020-03-15 01:16:49 +05:30
|
|
|
data = data.cloneIfNeeded();
|
2020-08-28 18:06:00 +05:30
|
|
|
data.encryption = event.content;
|
2020-03-15 01:16:49 +05:30
|
|
|
}
|
2020-08-21 15:25:25 +05:30
|
|
|
} else if (event.type === "m.room.name") {
|
2020-08-20 20:32:51 +05:30
|
|
|
const newName = event.content?.name;
|
2020-03-15 01:16:49 +05:30
|
|
|
if (newName !== data.name) {
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.name = newName;
|
|
|
|
}
|
2020-08-21 15:25:25 +05:30
|
|
|
} else if (event.type === "m.room.avatar") {
|
2020-08-20 20:32:51 +05:30
|
|
|
const newUrl = event.content?.url;
|
|
|
|
if (newUrl !== data.avatarUrl) {
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.avatarUrl = newUrl;
|
|
|
|
}
|
2020-08-21 18:05:23 +05:30
|
|
|
} else if (event.type === "m.room.canonical_alias") {
|
|
|
|
const content = event.content;
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.canonicalAlias = content.alias;
|
|
|
|
}
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2020-09-14 17:31:47 +05:30
|
|
|
function processTimelineEvent(data, eventEntry, isInitialSync, isTimelineOpen, ownUserId) {
|
|
|
|
if (eventEntry.eventType === "m.room.message") {
|
2020-09-14 20:04:07 +05:30
|
|
|
if (!data.lastMessageTimestamp || eventEntry.timestamp > data.lastMessageTimestamp) {
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.lastMessageTimestamp = eventEntry.timestamp;
|
|
|
|
}
|
2020-09-14 17:31:47 +05:30
|
|
|
if (!isInitialSync && eventEntry.sender !== ownUserId && !isTimelineOpen) {
|
2020-09-14 20:04:07 +05:30
|
|
|
data = data.cloneIfNeeded();
|
2020-08-21 17:15:38 +05:30
|
|
|
data.isUnread = true;
|
|
|
|
}
|
2020-09-14 17:31:47 +05:30
|
|
|
const {content} = eventEntry;
|
2020-08-20 20:32:51 +05:30
|
|
|
const body = content?.body;
|
|
|
|
const msgtype = content?.msgtype;
|
2020-09-14 17:31:47 +05:30
|
|
|
if (msgtype === "m.text" && !eventEntry.isEncrypted) {
|
2020-09-14 20:04:07 +05:30
|
|
|
data = data.cloneIfNeeded();
|
2020-03-15 01:16:49 +05:30
|
|
|
data.lastMessageBody = body;
|
|
|
|
}
|
|
|
|
}
|
2020-09-22 21:52:37 +05:30
|
|
|
// store the event key of the last decrypted event so when decryption does succeed,
|
|
|
|
// we can attempt to re-decrypt from this point to update the room summary
|
|
|
|
if (!!data.encryption && eventEntry.isEncrypted && eventEntry.isDecrypted) {
|
|
|
|
let hasLargerEventKey = true;
|
|
|
|
if (data.lastDecryptedEventKey) {
|
|
|
|
try {
|
|
|
|
hasLargerEventKey = eventEntry.compare(data.lastDecryptedEventKey) > 0;
|
|
|
|
} catch (err) {
|
2020-09-23 21:29:42 +05:30
|
|
|
// TODO: load the fragments in between here?
|
|
|
|
// this could happen if an earlier event gets decrypted that
|
|
|
|
// is in a fragment different from the live one and the timeline is not open.
|
|
|
|
// In this case, we will just read too many events once per app load
|
|
|
|
// and then keep the mapping in memory. When eventually an event is decrypted in
|
|
|
|
// the live fragment, this should stop failing and the event key will be written.
|
2020-09-22 21:52:37 +05:30
|
|
|
hasLargerEventKey = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (hasLargerEventKey) {
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
const {fragmentId, entryIndex} = eventEntry;
|
|
|
|
data.lastDecryptedEventKey = {fragmentId, entryIndex};
|
|
|
|
}
|
|
|
|
}
|
2020-03-15 01:16:49 +05:30
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
function updateSummary(data, summary) {
|
|
|
|
const heroes = summary["m.heroes"];
|
2020-08-21 21:41:26 +05:30
|
|
|
const joinCount = summary["m.joined_member_count"];
|
|
|
|
const inviteCount = summary["m.invited_member_count"];
|
2020-08-31 19:39:38 +05:30
|
|
|
// TODO: we could easily calculate if all members are available here and set hasFetchedMembers?
|
|
|
|
// so we can avoid calling /members...
|
|
|
|
// we'd need to do a count query in the roomMembers store though ...
|
2020-08-21 21:41:07 +05:30
|
|
|
if (heroes && Array.isArray(heroes)) {
|
2020-03-15 01:16:49 +05:30
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.heroes = heroes;
|
|
|
|
}
|
|
|
|
if (Number.isInteger(inviteCount)) {
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.inviteCount = inviteCount;
|
|
|
|
}
|
|
|
|
if (Number.isInteger(joinCount)) {
|
|
|
|
data = data.cloneIfNeeded();
|
|
|
|
data.joinCount = joinCount;
|
|
|
|
}
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
class SummaryData {
|
|
|
|
constructor(copy, roomId) {
|
|
|
|
this.roomId = copy ? copy.roomId : roomId;
|
|
|
|
this.name = copy ? copy.name : null;
|
|
|
|
this.lastMessageBody = copy ? copy.lastMessageBody : null;
|
2020-08-21 15:25:47 +05:30
|
|
|
this.lastMessageTimestamp = copy ? copy.lastMessageTimestamp : null;
|
2020-08-21 17:40:53 +05:30
|
|
|
this.isUnread = copy ? copy.isUnread : false;
|
2020-08-28 18:06:00 +05:30
|
|
|
this.encryption = copy ? copy.encryption : null;
|
2020-09-22 21:52:37 +05:30
|
|
|
this.lastDecryptedEventKey = copy ? copy.lastDecryptedEventKey : null;
|
2020-08-21 17:41:53 +05:30
|
|
|
this.isDirectMessage = copy ? copy.isDirectMessage : false;
|
2020-03-15 01:16:49 +05:30
|
|
|
this.membership = copy ? copy.membership : null;
|
|
|
|
this.inviteCount = copy ? copy.inviteCount : 0;
|
|
|
|
this.joinCount = copy ? copy.joinCount : 0;
|
|
|
|
this.heroes = copy ? copy.heroes : null;
|
|
|
|
this.canonicalAlias = copy ? copy.canonicalAlias : null;
|
2020-06-27 02:56:24 +05:30
|
|
|
this.hasFetchedMembers = copy ? copy.hasFetchedMembers : false;
|
2020-08-31 12:23:47 +05:30
|
|
|
this.isTrackingMembers = copy ? copy.isTrackingMembers : false;
|
2020-08-20 20:32:51 +05:30
|
|
|
this.avatarUrl = copy ? copy.avatarUrl : null;
|
2020-08-20 20:37:02 +05:30
|
|
|
this.notificationCount = copy ? copy.notificationCount : 0;
|
|
|
|
this.highlightCount = copy ? copy.highlightCount : 0;
|
2020-08-28 00:22:51 +05:30
|
|
|
this.tags = copy ? copy.tags : null;
|
2020-03-15 01:16:49 +05:30
|
|
|
this.cloned = copy ? true : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
cloneIfNeeded() {
|
|
|
|
if (this.cloned) {
|
|
|
|
return this;
|
|
|
|
} else {
|
|
|
|
return new SummaryData(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
serialize() {
|
|
|
|
const {cloned, ...serializedProps} = this;
|
|
|
|
return serializedProps;
|
|
|
|
}
|
2018-12-21 19:05:24 +05:30
|
|
|
|
2020-09-23 17:56:14 +05:30
|
|
|
applyTimelineEntries(timelineEntries, isInitialSync, isTimelineOpen, ownUserId) {
|
|
|
|
return applyTimelineEntries(this, timelineEntries, isInitialSync, isTimelineOpen, ownUserId);
|
2020-08-21 21:41:07 +05:30
|
|
|
}
|
|
|
|
|
2020-09-23 17:56:14 +05:30
|
|
|
applySyncResponse(roomResponse, membership) {
|
|
|
|
return applySyncResponse(this, roomResponse, membership);
|
2020-08-28 18:06:00 +05:30
|
|
|
}
|
|
|
|
|
2020-08-21 22:33:21 +05:30
|
|
|
get needsHeroes() {
|
2020-09-23 17:56:14 +05:30
|
|
|
return !this.name && !this.canonicalAlias && this.heroes && this.heroes.length > 0;
|
2020-08-21 15:25:47 +05:30
|
|
|
}
|
2020-09-23 17:56:14 +05:30
|
|
|
}
|
2020-08-21 15:25:47 +05:30
|
|
|
|
2020-09-23 17:56:14 +05:30
|
|
|
export class RoomSummary {
|
|
|
|
constructor(roomId) {
|
|
|
|
this._data = new SummaryData(null, roomId);
|
2018-12-21 19:05:24 +05:30
|
|
|
}
|
|
|
|
|
2020-09-23 17:56:14 +05:30
|
|
|
get data() {
|
|
|
|
return this._data;
|
2020-09-22 21:52:37 +05:30
|
|
|
}
|
|
|
|
|
2020-08-21 15:26:10 +05:30
|
|
|
writeClearUnread(txn) {
|
|
|
|
const data = new SummaryData(this._data);
|
|
|
|
data.isUnread = false;
|
|
|
|
data.notificationCount = 0;
|
|
|
|
data.highlightCount = 0;
|
|
|
|
txn.roomSummary.set(data.serialize());
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2020-06-27 02:56:24 +05:30
|
|
|
writeHasFetchedMembers(value, txn) {
|
|
|
|
const data = new SummaryData(this._data);
|
|
|
|
data.hasFetchedMembers = value;
|
|
|
|
txn.roomSummary.set(data.serialize());
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2020-08-31 12:23:47 +05:30
|
|
|
writeIsTrackingMembers(value, txn) {
|
|
|
|
const data = new SummaryData(this._data);
|
|
|
|
data.isTrackingMembers = value;
|
|
|
|
txn.roomSummary.set(data.serialize());
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2020-09-23 17:56:14 +05:30
|
|
|
writeData(data, txn) {
|
2020-03-15 01:16:49 +05:30
|
|
|
if (data !== this._data) {
|
|
|
|
txn.roomSummary.set(data.serialize());
|
|
|
|
return data;
|
2018-12-21 19:05:24 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-23 17:56:14 +05:30
|
|
|
async writeAndApplyData(data, storage) {
|
|
|
|
if (data === this._data) {
|
|
|
|
return;
|
|
|
|
}
|
2020-09-17 21:29:35 +05:30
|
|
|
const txn = await storage.readWriteTxn([
|
2020-09-14 20:04:07 +05:30
|
|
|
storage.storeNames.roomSummary,
|
|
|
|
]);
|
|
|
|
try {
|
|
|
|
txn.roomSummary.set(data.serialize());
|
|
|
|
} catch (err) {
|
|
|
|
txn.abort();
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
await txn.complete();
|
|
|
|
this.applyChanges(data);
|
|
|
|
}
|
|
|
|
|
2020-06-27 02:56:24 +05:30
|
|
|
applyChanges(data) {
|
2020-03-15 01:16:49 +05:30
|
|
|
this._data = data;
|
2020-09-23 17:56:14 +05:30
|
|
|
// clear cloned flag, so cloneIfNeeded makes a copy and
|
|
|
|
// this._data is not modified if any field is changed.
|
|
|
|
this._data.cloned = false;
|
2020-03-15 01:16:49 +05:30
|
|
|
}
|
2018-12-21 19:05:24 +05:30
|
|
|
|
2020-03-15 01:16:49 +05:30
|
|
|
async load(summary) {
|
|
|
|
this._data = new SummaryData(summary);
|
2018-12-21 19:05:24 +05:30
|
|
|
}
|
2020-03-15 01:16:49 +05:30
|
|
|
}
|
2018-12-21 19:05:24 +05:30
|
|
|
|
2020-03-15 01:16:49 +05:30
|
|
|
export function tests() {
|
|
|
|
return {
|
|
|
|
"membership trigger change": function(assert) {
|
|
|
|
const summary = new RoomSummary("id");
|
2020-03-15 02:08:37 +05:30
|
|
|
let written = false;
|
2020-08-27 17:52:59 +05:30
|
|
|
const changes = summary.writeSync({}, "join", false, false, {roomSummary: {set: () => { written = true; }}});
|
2020-03-15 01:16:49 +05:30
|
|
|
assert(changes);
|
2020-03-15 02:08:37 +05:30
|
|
|
assert(written);
|
|
|
|
assert.equal(changes.membership, "join");
|
2019-10-13 11:18:33 +05:30
|
|
|
}
|
2020-03-15 01:16:49 +05:30
|
|
|
}
|
2018-12-21 19:05:24 +05:30
|
|
|
}
|