mirror of
https://github.com/vector-im/hydrogen-web.git
synced 2025-01-11 12:37:22 +01:00
Avoid creating new fragments when posible
This commit is contained in:
parent
fdfea95d22
commit
1d71665c48
@ -16,6 +16,7 @@ limitations under the License.
|
|||||||
|
|
||||||
import {EventKey} from "../EventKey";
|
import {EventKey} from "../EventKey";
|
||||||
import {EventEntry} from "../entries/EventEntry.js";
|
import {EventEntry} from "../entries/EventEntry.js";
|
||||||
|
import {Direction} from "../Direction";
|
||||||
import {FragmentBoundaryEntry} from "../entries/FragmentBoundaryEntry.js";
|
import {FragmentBoundaryEntry} from "../entries/FragmentBoundaryEntry.js";
|
||||||
import {createEventEntry, directionalAppend} from "./common.js";
|
import {createEventEntry, directionalAppend} from "./common.js";
|
||||||
import {RoomMember, EVENT_TYPE as MEMBER_EVENT_TYPE} from "../../members/RoomMember.js";
|
import {RoomMember, EVENT_TYPE as MEMBER_EVENT_TYPE} from "../../members/RoomMember.js";
|
||||||
@ -211,14 +212,46 @@ export class GapWriter {
|
|||||||
return changedFragments;
|
return changedFragments;
|
||||||
}
|
}
|
||||||
|
|
||||||
async _storeAndUpdate(fragmentEntry, events, key, end, state, txn, log) {
|
/* If searching for overlapping entries in two directions,
|
||||||
const {
|
* combine the results of the two searches.
|
||||||
nonOverlappingEvents,
|
*
|
||||||
neighbourFragmentEntry
|
* @param mainOverlap the result of a search that located an existing fragment.
|
||||||
} = await this._findOverlappingEvents(fragmentEntry, events, txn, log);
|
* @param otherOverlap the result of a search in the opposite direction to mainOverlap.
|
||||||
const {entries, updatedEntries} = await this._storeEvents(nonOverlappingEvents, key, fragmentEntry.direction, state, txn, log);
|
* @param event the event from which the two-directional search occured.
|
||||||
const fragments = await this._updateFragments(fragmentEntry, neighbourFragmentEntry, end, entries, txn);
|
* @param token the new pagination token for mainOverlap.
|
||||||
return {entries, updatedEntries, fragments};
|
*/
|
||||||
|
async _linkOverlapping(mainOverlap, otherOverlap, event, token, state, txn, log) {
|
||||||
|
const fragmentEntry = mainOverlap.neighbourFragmentEntry;
|
||||||
|
const otherEntry = otherOverlap.neighbourFragmentEntry;
|
||||||
|
|
||||||
|
// We're filling the entry from the opposite direction that the search occured
|
||||||
|
// (e.g. searched up, filling down). Thus, the events need to be added in the opposite
|
||||||
|
// order.
|
||||||
|
const allEvents = mainOverlap.nonOverlappingEvents.reverse();
|
||||||
|
allEvents.push(event, ...otherOverlap.nonOverlappingEvents);
|
||||||
|
|
||||||
|
// TODO Very important: can the 'up' and 'down' entries be the same? If that's
|
||||||
|
// the case, we can end up with a self-link (and thus infinite loop).
|
||||||
|
|
||||||
|
let lastKey = await this._findFragmentEdgeEventKey(fragmentEntry, txn);
|
||||||
|
const {entries, updatedEntries} = await this._storeEvents(allEvents, lastKey, fragmentEntry.direction, state, txn, log);
|
||||||
|
const fragments = await this._updateFragments(fragmentEntry, otherEntry, token, entries, txn);
|
||||||
|
const contextEvent = entries.find(e => e.id === event.event_id) || null;
|
||||||
|
return { entries, updatedEntries, fragments, contextEvent };
|
||||||
|
}
|
||||||
|
|
||||||
|
async _createNewFragment(txn) {
|
||||||
|
const maxFragmentKey = await txn.timelineFragments.getMaxFragmentId(this._roomId);
|
||||||
|
const newFragment = {
|
||||||
|
roomId: this._roomId,
|
||||||
|
id: maxFragmentKey + 1,
|
||||||
|
previousId: null,
|
||||||
|
nextId: null,
|
||||||
|
previousToken: null,
|
||||||
|
nextToken: null
|
||||||
|
};
|
||||||
|
txn.timelineFragments.add(newFragment);
|
||||||
|
return newFragment;
|
||||||
}
|
}
|
||||||
|
|
||||||
async writeContext(response, txn, log) {
|
async writeContext(response, txn, log) {
|
||||||
@ -238,36 +271,20 @@ export class GapWriter {
|
|||||||
return { entries: [], updatedEntries: [], fragments: [], contextEvent: new EventEntry(eventEntry, this._fragmentIdComparer) }
|
return { entries: [], updatedEntries: [], fragments: [], contextEvent: new EventEntry(eventEntry, this._fragmentIdComparer) }
|
||||||
}
|
}
|
||||||
|
|
||||||
const maxFragmentKey = await txn.timelineFragments.getMaxFragmentId(this._roomId);
|
const overlapUp = await this._findOverlappingEventsFor(null, null, Direction.Backward, eventsBefore, txn, log);
|
||||||
const newFragment = {
|
const overlapDown = await this._findOverlappingEventsFor(null, null, Direction.Forward, eventsAfter, txn, log);
|
||||||
roomId: this._roomId,
|
if (overlapUp.neighbourFragmentEntry) {
|
||||||
id: maxFragmentKey + 1,
|
return this._linkOverlapping(overlapUp, overlapDown, event, end, state, txn, log);
|
||||||
previousId: null,
|
} else if (overlapDown.neighbourFragmentEntry) {
|
||||||
nextId: null,
|
return this._linkOverlapping(overlapDown, overlapUp, event, start, state, txn, log);
|
||||||
previousToken: null,
|
|
||||||
nextToken: null
|
|
||||||
};
|
|
||||||
txn.timelineFragments.add(newFragment);
|
|
||||||
const eventKey = EventKey.defaultFragmentKey(newFragment.id);
|
|
||||||
|
|
||||||
const startEntry = FragmentBoundaryEntry.start(newFragment, this._fragmentIdComparer);
|
|
||||||
const startFill = await this._storeAndUpdate(startEntry, eventsBefore, eventKey, start, state, txn, log);
|
|
||||||
|
|
||||||
eventsAfter.unshift(event);
|
|
||||||
const endEntry = FragmentBoundaryEntry.end(newFragment, this._fragmentIdComparer);
|
|
||||||
const endFill = await this._storeAndUpdate(endEntry, eventsAfter, eventKey, end, state, txn, log);
|
|
||||||
|
|
||||||
// Both startFill and endFill are throwaway objects, so we
|
|
||||||
// may as well modify one of them in-place instead of returning a third.
|
|
||||||
startFill.entries.push(...endFill.entries);
|
|
||||||
startFill.updatedEntries.push(...endFill.updatedEntries);
|
|
||||||
startFill.contextEvent = startFill.entries.find(e => e.id === event.event_id);
|
|
||||||
startFill.fragments.push(...endFill.fragments);
|
|
||||||
if (!startFill.fragments.includes(newFragment)) {
|
|
||||||
// We created the fragment so it is updated.
|
|
||||||
startFill.fragments.push(newFragment);
|
|
||||||
}
|
}
|
||||||
return startFill;
|
|
||||||
|
// No overlapping fragments found.
|
||||||
|
const newFragment = await this._createNewFragment(txn);
|
||||||
|
// Pretend that we did find an overlapping entry above, and that this entry is for the new fragment.
|
||||||
|
const newEntry = FragmentBoundaryEntry.end(newFragment, this._fragmentIdComparer);
|
||||||
|
overlapUp.neighbourFragmentEntry = newEntry;
|
||||||
|
return this._linkOverlapping(overlapUp, overlapDown, event, end, state, txn, log);
|
||||||
}
|
}
|
||||||
|
|
||||||
async writeFragmentFill(fragmentEntry, response, txn, log) {
|
async writeFragmentFill(fragmentEntry, response, txn, log) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user