mirror of
https://github.com/vector-im/hydrogen-web.git
synced 2025-01-10 20:17:32 +01:00
implement subscribing to a single event
This commit is contained in:
parent
7d81306a49
commit
137264edcb
90
src/matrix/room/ObservedEventMap.js
Normal file
90
src/matrix/room/ObservedEventMap.js
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2020 The Matrix.org Foundation C.I.C.
|
||||||
|
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {BaseObservableValue} from "../../observable/ObservableValue.js";
|
||||||
|
|
||||||
|
export class ObservedEventMap {
|
||||||
|
constructor(notifyEmpty) {
|
||||||
|
this._map = new Map();
|
||||||
|
this._notifyEmpty = notifyEmpty;
|
||||||
|
}
|
||||||
|
|
||||||
|
observe(eventId, eventEntry = null) {
|
||||||
|
let observable = this._map.get(eventId);
|
||||||
|
if (!observable) {
|
||||||
|
observable = new ObservedEvent(this, eventEntry);
|
||||||
|
this._map.set(eventId, observable);
|
||||||
|
}
|
||||||
|
return observable;
|
||||||
|
}
|
||||||
|
|
||||||
|
updateEvents(eventEntries) {
|
||||||
|
for (let i = 0; i < eventEntries.length; i += 1) {
|
||||||
|
const entry = eventEntries[i];
|
||||||
|
const observable = this._map.get(entry.id);
|
||||||
|
observable?.update(entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_remove(observable) {
|
||||||
|
this._map.delete(observable.get().id);
|
||||||
|
if (this._map.size === 0) {
|
||||||
|
this._notifyEmpty();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ObservedEvent extends BaseObservableValue {
|
||||||
|
constructor(eventMap, entry) {
|
||||||
|
super();
|
||||||
|
this._eventMap = eventMap;
|
||||||
|
this._entry = entry;
|
||||||
|
// remove subscription in microtask after creating it
|
||||||
|
// otherwise ObservedEvents would easily never get
|
||||||
|
// removed if you never subscribe
|
||||||
|
Promise.resolve().then(() => {
|
||||||
|
if (!this.hasSubscriptions) {
|
||||||
|
this._eventMap.remove(this);
|
||||||
|
this._eventMap = null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
subscribe(handler) {
|
||||||
|
if (!this._eventMap) {
|
||||||
|
throw new Error("ObservedEvent expired, subscribe right after calling room.observeEvent()");
|
||||||
|
}
|
||||||
|
return super.subscribe(handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
onUnsubscribeLast() {
|
||||||
|
this._eventMap._remove(this);
|
||||||
|
this._eventMap = null;
|
||||||
|
super.onUnsubscribeLast();
|
||||||
|
}
|
||||||
|
|
||||||
|
update(entry) {
|
||||||
|
// entries are mostly updated in-place,
|
||||||
|
// apart from when they are created,
|
||||||
|
// but doesn't hurt to reassign
|
||||||
|
this._entry = entry;
|
||||||
|
this.emit(this._entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
get() {
|
||||||
|
return this._entry;
|
||||||
|
}
|
||||||
|
}
|
@ -29,8 +29,8 @@ import {Heroes} from "./members/Heroes.js";
|
|||||||
import {EventEntry} from "./timeline/entries/EventEntry.js";
|
import {EventEntry} from "./timeline/entries/EventEntry.js";
|
||||||
import {EventKey} from "./timeline/EventKey.js";
|
import {EventKey} from "./timeline/EventKey.js";
|
||||||
import {Direction} from "./timeline/Direction.js";
|
import {Direction} from "./timeline/Direction.js";
|
||||||
|
import {ObservedEventMap} from "./ObservedEventMap.js";
|
||||||
import {DecryptionSource} from "../e2ee/common.js";
|
import {DecryptionSource} from "../e2ee/common.js";
|
||||||
|
|
||||||
const EVENT_ENCRYPTED_TYPE = "m.room.encrypted";
|
const EVENT_ENCRYPTED_TYPE = "m.room.encrypted";
|
||||||
|
|
||||||
export class Room extends EventEmitter {
|
export class Room extends EventEmitter {
|
||||||
@ -53,6 +53,7 @@ export class Room extends EventEmitter {
|
|||||||
this._roomEncryption = null;
|
this._roomEncryption = null;
|
||||||
this._getSyncToken = getSyncToken;
|
this._getSyncToken = getSyncToken;
|
||||||
this._clock = clock;
|
this._clock = clock;
|
||||||
|
this._observedEvents = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
_readRetryDecryptCandidateEntries(sinceEventKey, txn) {
|
_readRetryDecryptCandidateEntries(sinceEventKey, txn) {
|
||||||
@ -165,6 +166,9 @@ export class Room extends EventEmitter {
|
|||||||
}
|
}
|
||||||
await writeTxn.complete();
|
await writeTxn.complete();
|
||||||
decryption.applyToEntries(entries);
|
decryption.applyToEntries(entries);
|
||||||
|
if (this._observedEvents) {
|
||||||
|
this._observedEvents.updateEvents(entries);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
return request;
|
return request;
|
||||||
}
|
}
|
||||||
@ -285,6 +289,9 @@ export class Room extends EventEmitter {
|
|||||||
if (this._timeline) {
|
if (this._timeline) {
|
||||||
this._timeline.appendLiveEntries(newTimelineEntries);
|
this._timeline.appendLiveEntries(newTimelineEntries);
|
||||||
}
|
}
|
||||||
|
if (this._observedEvents) {
|
||||||
|
this._observedEvents.updateEvents(newTimelineEntries);
|
||||||
|
}
|
||||||
if (removedPendingEvents) {
|
if (removedPendingEvents) {
|
||||||
this._sendQueue.emitRemovals(removedPendingEvents);
|
this._sendQueue.emitRemovals(removedPendingEvents);
|
||||||
}
|
}
|
||||||
@ -580,6 +587,45 @@ export class Room extends EventEmitter {
|
|||||||
this._summary.applyChanges(changes);
|
this._summary.applyChanges(changes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
observeEvent(eventId) {
|
||||||
|
if (!this._observedEvents) {
|
||||||
|
this._observedEvents = new ObservedEventMap(() => {
|
||||||
|
this._observedEvents = null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
let entry = null;
|
||||||
|
if (this._timeline) {
|
||||||
|
entry = this._timeline.getByEventId(eventId);
|
||||||
|
}
|
||||||
|
const observable = this._observedEvents.observe(eventId, entry);
|
||||||
|
if (!entry) {
|
||||||
|
// update in the background
|
||||||
|
this._readEventById(eventId).then(entry => {
|
||||||
|
observable.update(entry);
|
||||||
|
}).catch(err => {
|
||||||
|
console.warn(`could not load event ${eventId} from storage`, err);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return observable;
|
||||||
|
}
|
||||||
|
|
||||||
|
async _readEventById(eventId) {
|
||||||
|
let stores = [this._storage.storeNames.timelineEvents];
|
||||||
|
if (this.isEncrypted) {
|
||||||
|
stores.push(this._storage.storeNames.inboundGroupSessions);
|
||||||
|
}
|
||||||
|
const txn = this._storage.readTxn(stores);
|
||||||
|
const storageEntry = await txn.timelineEvents.getByEventId(this._roomId, eventId);
|
||||||
|
if (storageEntry) {
|
||||||
|
const entry = new EventEntry(storageEntry, this._fragmentIdComparer);
|
||||||
|
if (entry.eventType === EVENT_ENCRYPTED_TYPE) {
|
||||||
|
const request = this._decryptEntries(DecryptionSource.Timeline, [entry], txn);
|
||||||
|
await request.complete();
|
||||||
|
}
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
dispose() {
|
dispose() {
|
||||||
this._roomEncryption?.dispose();
|
this._roomEncryption?.dispose();
|
||||||
this._timeline?.dispose();
|
this._timeline?.dispose();
|
||||||
|
@ -95,6 +95,15 @@ export class Timeline {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getByEventId(eventId) {
|
||||||
|
for (let i = 0; i < this._remoteEntries.length; i += 1) {
|
||||||
|
const entry = this._remoteEntries.get(i);
|
||||||
|
if (entry.id === eventId) {
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** @public */
|
/** @public */
|
||||||
get entries() {
|
get entries() {
|
||||||
return this._allEntries;
|
return this._allEntries;
|
||||||
|
@ -48,6 +48,10 @@ export class BaseObservable {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get hasSubscriptions() {
|
||||||
|
return this._handlers.size !== 0;
|
||||||
|
}
|
||||||
|
|
||||||
// Add iterator over handlers here
|
// Add iterator over handlers here
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user