diff options
author | RaindropsSys <contact@minteck.org> | 2023-04-24 14:03:36 +0200 |
---|---|---|
committer | RaindropsSys <contact@minteck.org> | 2023-04-24 14:03:36 +0200 |
commit | 633c92eae865e957121e08de634aeee11a8b3992 (patch) | |
tree | 09d881bee1dae0b6eee49db1dfaf0f500240606c /includes/external/matrix/node_modules/matrix-js-sdk/src/models/beacon.ts | |
parent | c4657e4509733699c0f26a3c900bab47e915d5a0 (diff) | |
download | pluralconnect-633c92eae865e957121e08de634aeee11a8b3992.tar.gz pluralconnect-633c92eae865e957121e08de634aeee11a8b3992.tar.bz2 pluralconnect-633c92eae865e957121e08de634aeee11a8b3992.zip |
Updated 18 files, added 1692 files and deleted includes/system/compare.inc (automated)
Diffstat (limited to 'includes/external/matrix/node_modules/matrix-js-sdk/src/models/beacon.ts')
-rw-r--r-- | includes/external/matrix/node_modules/matrix-js-sdk/src/models/beacon.ts | 209 |
1 files changed, 209 insertions, 0 deletions
diff --git a/includes/external/matrix/node_modules/matrix-js-sdk/src/models/beacon.ts b/includes/external/matrix/node_modules/matrix-js-sdk/src/models/beacon.ts new file mode 100644 index 0000000..3801831 --- /dev/null +++ b/includes/external/matrix/node_modules/matrix-js-sdk/src/models/beacon.ts @@ -0,0 +1,209 @@ +/* +Copyright 2022 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 { MBeaconEventContent } from "../@types/beacon"; +import { BeaconInfoState, BeaconLocationState, parseBeaconContent, parseBeaconInfoContent } from "../content-helpers"; +import { MatrixEvent } from "./event"; +import { sortEventsByLatestContentTimestamp } from "../utils"; +import { TypedEventEmitter } from "./typed-event-emitter"; + +export enum BeaconEvent { + New = "Beacon.new", + Update = "Beacon.update", + LivenessChange = "Beacon.LivenessChange", + Destroy = "Beacon.Destroy", + LocationUpdate = "Beacon.LocationUpdate", +} + +export type BeaconEventHandlerMap = { + [BeaconEvent.Update]: (event: MatrixEvent, beacon: Beacon) => void; + [BeaconEvent.LivenessChange]: (isLive: boolean, beacon: Beacon) => void; + [BeaconEvent.Destroy]: (beaconIdentifier: string) => void; + [BeaconEvent.LocationUpdate]: (locationState: BeaconLocationState) => void; + [BeaconEvent.Destroy]: (beaconIdentifier: string) => void; +}; + +export const isTimestampInDuration = (startTimestamp: number, durationMs: number, timestamp: number): boolean => + timestamp >= startTimestamp && startTimestamp + durationMs >= timestamp; + +// beacon info events are uniquely identified by +// `<roomId>_<state_key>` +export type BeaconIdentifier = string; +export const getBeaconInfoIdentifier = (event: MatrixEvent): BeaconIdentifier => + `${event.getRoomId()}_${event.getStateKey()}`; + +// https://github.com/matrix-org/matrix-spec-proposals/pull/3672 +export class Beacon extends TypedEventEmitter<Exclude<BeaconEvent, BeaconEvent.New>, BeaconEventHandlerMap> { + public readonly roomId: string; + // beaconInfo is assigned by setBeaconInfo in the constructor + // ! to make tsc believe it is definitely assigned + private _beaconInfo!: BeaconInfoState; + private _isLive?: boolean; + private livenessWatchTimeout?: ReturnType<typeof setTimeout>; + private _latestLocationEvent?: MatrixEvent; + + public constructor(private rootEvent: MatrixEvent) { + super(); + this.roomId = this.rootEvent.getRoomId()!; + this.setBeaconInfo(this.rootEvent); + } + + public get isLive(): boolean { + return !!this._isLive; + } + + public get identifier(): BeaconIdentifier { + return getBeaconInfoIdentifier(this.rootEvent); + } + + public get beaconInfoId(): string { + return this.rootEvent.getId()!; + } + + public get beaconInfoOwner(): string { + return this.rootEvent.getStateKey()!; + } + + public get beaconInfoEventType(): string { + return this.rootEvent.getType(); + } + + public get beaconInfo(): BeaconInfoState { + return this._beaconInfo; + } + + public get latestLocationState(): BeaconLocationState | undefined { + return this._latestLocationEvent && parseBeaconContent(this._latestLocationEvent.getContent()); + } + + public get latestLocationEvent(): MatrixEvent | undefined { + return this._latestLocationEvent; + } + + public update(beaconInfoEvent: MatrixEvent): void { + if (getBeaconInfoIdentifier(beaconInfoEvent) !== this.identifier) { + throw new Error("Invalid updating event"); + } + // don't update beacon with an older event + if (beaconInfoEvent.getTs() < this.rootEvent.getTs()) { + return; + } + this.rootEvent = beaconInfoEvent; + this.setBeaconInfo(this.rootEvent); + + this.emit(BeaconEvent.Update, beaconInfoEvent, this); + this.clearLatestLocation(); + } + + public destroy(): void { + if (this.livenessWatchTimeout) { + clearTimeout(this.livenessWatchTimeout); + } + + this._isLive = false; + this.emit(BeaconEvent.Destroy, this.identifier); + } + + /** + * Monitor liveness of a beacon + * Emits BeaconEvent.LivenessChange when beacon expires + */ + public monitorLiveness(): void { + if (this.livenessWatchTimeout) { + clearTimeout(this.livenessWatchTimeout); + } + + this.checkLiveness(); + if (!this.beaconInfo) return; + if (this.isLive) { + const expiryInMs = this.beaconInfo.timestamp! + this.beaconInfo.timeout - Date.now(); + if (expiryInMs > 1) { + this.livenessWatchTimeout = setTimeout(() => { + this.monitorLiveness(); + }, expiryInMs); + } + } else if (this.beaconInfo.timestamp! > Date.now()) { + // beacon start timestamp is in the future + // check liveness again then + this.livenessWatchTimeout = setTimeout(() => { + this.monitorLiveness(); + }, this.beaconInfo.timestamp! - Date.now()); + } + } + + /** + * Process Beacon locations + * Emits BeaconEvent.LocationUpdate + */ + public addLocations(beaconLocationEvents: MatrixEvent[]): void { + // discard locations for beacons that are not live + if (!this.isLive) { + return; + } + + const validLocationEvents = beaconLocationEvents.filter((event) => { + const content = event.getContent<MBeaconEventContent>(); + const parsed = parseBeaconContent(content); + if (!parsed.uri || !parsed.timestamp) return false; // we won't be able to process these + const { timestamp } = parsed; + return ( + this._beaconInfo.timestamp && + // only include positions that were taken inside the beacon's live period + isTimestampInDuration(this._beaconInfo.timestamp, this._beaconInfo.timeout, timestamp) && + // ignore positions older than our current latest location + (!this.latestLocationState || timestamp > this.latestLocationState.timestamp!) + ); + }); + const latestLocationEvent = validLocationEvents.sort(sortEventsByLatestContentTimestamp)?.[0]; + + if (latestLocationEvent) { + this._latestLocationEvent = latestLocationEvent; + this.emit(BeaconEvent.LocationUpdate, this.latestLocationState!); + } + } + + private clearLatestLocation = (): void => { + this._latestLocationEvent = undefined; + this.emit(BeaconEvent.LocationUpdate, this.latestLocationState!); + }; + + private setBeaconInfo(event: MatrixEvent): void { + this._beaconInfo = parseBeaconInfoContent(event.getContent()); + this.checkLiveness(); + } + + private checkLiveness(): void { + const prevLiveness = this.isLive; + + // element web sets a beacon's start timestamp to the senders local current time + // when Alice's system clock deviates slightly from Bob's a beacon Alice intended to be live + // may have a start timestamp in the future from Bob's POV + // handle this by adding 6min of leniency to the start timestamp when it is in the future + if (!this.beaconInfo) return; + const startTimestamp = + this.beaconInfo.timestamp! > Date.now() + ? this.beaconInfo.timestamp! - 360000 /* 6min */ + : this.beaconInfo.timestamp; + this._isLive = + !!this._beaconInfo.live && + !!startTimestamp && + isTimestampInDuration(startTimestamp, this._beaconInfo.timeout, Date.now()); + + if (prevLiveness !== this.isLive) { + this.emit(BeaconEvent.LivenessChange, this.isLive, this); + } + } +} |