mirror of
https://github.com/vector-im/hydrogen-web.git
synced 2025-01-10 20:17:32 +01:00
first draft of persistFragmentFill
This commit is contained in:
parent
89bc0e1696
commit
a6b17cf25a
29
src/matrix/room/timeline/Direction.js
Normal file
29
src/matrix/room/timeline/Direction.js
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
|
||||||
|
const _forward = Object.freeze(new Direction(true));
|
||||||
|
const _backward = Object.freeze(new Direction(false));
|
||||||
|
|
||||||
|
export default class Direction {
|
||||||
|
constructor(isForward) {
|
||||||
|
this._isForward = isForward;
|
||||||
|
}
|
||||||
|
|
||||||
|
get isForward() {
|
||||||
|
return this._isForward;
|
||||||
|
}
|
||||||
|
|
||||||
|
get isBackward() {
|
||||||
|
return !this.isForward;
|
||||||
|
}
|
||||||
|
|
||||||
|
asApiString() {
|
||||||
|
return this.isForward ? "f" : "b";
|
||||||
|
}
|
||||||
|
|
||||||
|
static get Forward() {
|
||||||
|
return _forward;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get Backward() {
|
||||||
|
return _backward;
|
||||||
|
}
|
||||||
|
}
|
@ -3,6 +3,7 @@ const MIN_EVENT_INDEX = Number.MIN_SAFE_INTEGER + 1;
|
|||||||
const MAX_EVENT_INDEX = Number.MAX_SAFE_INTEGER - 1;
|
const MAX_EVENT_INDEX = Number.MAX_SAFE_INTEGER - 1;
|
||||||
const MID_EVENT_INDEX = 0;
|
const MID_EVENT_INDEX = 0;
|
||||||
|
|
||||||
|
// key for events in the timelineEvents store
|
||||||
export default class EventKey {
|
export default class EventKey {
|
||||||
constructor(fragmentId, eventIndex) {
|
constructor(fragmentId, eventIndex) {
|
||||||
this.fragmentId = fragmentId;
|
this.fragmentId = fragmentId;
|
||||||
@ -14,6 +15,18 @@ export default class EventKey {
|
|||||||
return new EventKey(this.fragmentId + 1, MID_EVENT_INDEX);
|
return new EventKey(this.fragmentId + 1, MID_EVENT_INDEX);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nextKeyForDirection(direction) {
|
||||||
|
if (direction.isForward) {
|
||||||
|
return this.nextKey();
|
||||||
|
} else {
|
||||||
|
return this.previousKey();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
previousKey() {
|
||||||
|
return new EventKey(this.fragmentId, this.eventIndex - 1);
|
||||||
|
}
|
||||||
|
|
||||||
nextKey() {
|
nextKey() {
|
||||||
return new EventKey(this.fragmentId, this.eventIndex + 1);
|
return new EventKey(this.fragmentId, this.eventIndex + 1);
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,10 @@
|
|||||||
//entries can be sorted, first by fragment, then by entry index.
|
//entries can be sorted, first by fragment, then by entry index.
|
||||||
|
|
||||||
export default class BaseEntry {
|
export default class BaseEntry {
|
||||||
|
constructor(fragmentIdComparer) {
|
||||||
|
this._fragmentIdComparer = fragmentIdComparer;
|
||||||
|
}
|
||||||
|
|
||||||
get fragmentId() {
|
get fragmentId() {
|
||||||
throw new Error("unimplemented");
|
throw new Error("unimplemented");
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import BaseEntry from "./BaseEntry.js";
|
import BaseEntry from "./BaseEntry.js";
|
||||||
|
import Direction from "../Direction.js";
|
||||||
|
|
||||||
export default class FragmentBoundaryEntry extends BaseEntry {
|
export default class FragmentBoundaryEntry extends BaseEntry {
|
||||||
constructor(fragment, isFragmentStart, fragmentIdComparator) {
|
constructor(fragment, isFragmentStart, fragmentIdComparator) {
|
||||||
@ -15,12 +16,12 @@ export default class FragmentBoundaryEntry extends BaseEntry {
|
|||||||
return new FragmentBoundaryEntry(fragment, false, fragmentIdComparator);
|
return new FragmentBoundaryEntry(fragment, false, fragmentIdComparator);
|
||||||
}
|
}
|
||||||
|
|
||||||
get hasStarted() {
|
get started() {
|
||||||
return this._isFragmentStart;
|
return this._isFragmentStart;
|
||||||
}
|
}
|
||||||
|
|
||||||
get hasEnded() {
|
get hasEnded() {
|
||||||
return !this.hasStarted;
|
return !this.started;
|
||||||
}
|
}
|
||||||
|
|
||||||
get fragment() {
|
get fragment() {
|
||||||
@ -32,7 +33,7 @@ export default class FragmentBoundaryEntry extends BaseEntry {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get entryIndex() {
|
get entryIndex() {
|
||||||
if (this.hasStarted) {
|
if (this.started) {
|
||||||
return Number.MIN_SAFE_INTEGER;
|
return Number.MIN_SAFE_INTEGER;
|
||||||
} else {
|
} else {
|
||||||
return Number.MAX_SAFE_INTEGER;
|
return Number.MAX_SAFE_INTEGER;
|
||||||
@ -40,10 +41,54 @@ export default class FragmentBoundaryEntry extends BaseEntry {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get isGap() {
|
get isGap() {
|
||||||
if (this.hasStarted) {
|
return !!this.token;
|
||||||
return !!this.fragment.nextToken;
|
|
||||||
} else {
|
|
||||||
return !!this.fragment.previousToken;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get token() {
|
||||||
|
if (this.started) {
|
||||||
|
return this.fragment.nextToken;
|
||||||
|
} else {
|
||||||
|
return this.fragment.previousToken;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
set token(token) {
|
||||||
|
if (this.started) {
|
||||||
|
this.fragment.nextToken = token;
|
||||||
|
} else {
|
||||||
|
this.fragment.previousToken = token;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get linkedFragmentId() {
|
||||||
|
if (this.started) {
|
||||||
|
return this.fragment.nextId;
|
||||||
|
} else {
|
||||||
|
return this.fragment.previousId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
set linkedFragmentId(id) {
|
||||||
|
if (this.started) {
|
||||||
|
this.fragment.nextId = id;
|
||||||
|
} else {
|
||||||
|
this.fragment.previousId = id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get direction() {
|
||||||
|
if (this.started) {
|
||||||
|
return Direction.Backward;
|
||||||
|
} else {
|
||||||
|
return Direction.Forward;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
withUpdatedFragment(fragment) {
|
||||||
|
return new FragmentBoundaryEntry(fragment, this._isFragmentStart, this._fragmentIdComparator);
|
||||||
|
}
|
||||||
|
|
||||||
|
createNeighbourEntry(neighbour) {
|
||||||
|
return new FragmentBoundaryEntry(neighbour, !this._isFragmentStart, this._fragmentIdComparator);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
208
src/matrix/room/timeline/persistence/GapPersister.js
Normal file
208
src/matrix/room/timeline/persistence/GapPersister.js
Normal file
@ -0,0 +1,208 @@
|
|||||||
|
import EventKey from "../EventKey.js";
|
||||||
|
import EventEntry from "../entries/EventEntry.js";
|
||||||
|
import {createEventEntry} from "./common.js";
|
||||||
|
|
||||||
|
function directionalAppend(array, value, direction) {
|
||||||
|
if (direction.isForward) {
|
||||||
|
array.push(value);
|
||||||
|
} else {
|
||||||
|
array.splice(0, 0, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default class GapPersister {
|
||||||
|
constructor({roomId, storage, fragmentIdComparer}) {
|
||||||
|
this._roomId = roomId;
|
||||||
|
this._storage = storage;
|
||||||
|
this._fragmentIdComparer = fragmentIdComparer;
|
||||||
|
}
|
||||||
|
async persistFragmentFill(fragmentEntry, response) {
|
||||||
|
const {fragmentId, direction} = fragmentEntry;
|
||||||
|
// assuming that chunk is in chronological order when backwards too?
|
||||||
|
const {chunk, start, end} = response;
|
||||||
|
let entries;
|
||||||
|
|
||||||
|
if (!Array.isArray(chunk)) {
|
||||||
|
throw new Error("Invalid chunk in response");
|
||||||
|
}
|
||||||
|
if (typeof end !== "string") {
|
||||||
|
throw new Error("Invalid end token in response");
|
||||||
|
}
|
||||||
|
|
||||||
|
const txn = await this._storage.readWriteTxn([
|
||||||
|
this._storage.storeNames.timelineEvents,
|
||||||
|
this._storage.storeNames.timelineFragments,
|
||||||
|
]);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// make sure we have the latest fragment from the store
|
||||||
|
const fragment = await txn.timelineFragments.get(fragmentId);
|
||||||
|
if (!fragment) {
|
||||||
|
throw new Error(`Unknown fragment: ${fragmentId}`);
|
||||||
|
}
|
||||||
|
fragmentEntry = fragmentEntry.withUpdatedFragment(fragment);
|
||||||
|
// check that the request was done with the token we are aware of (extra care to avoid timeline corruption)
|
||||||
|
if (fragmentEntry.token !== start) {
|
||||||
|
throw new Error("start is not equal to prev_batch or next_batch");
|
||||||
|
}
|
||||||
|
// find last event in fragment so we get the eventIndex to begin creating keys at
|
||||||
|
let currentKey;
|
||||||
|
if (direction.isBackward) {
|
||||||
|
const [firstEvent] = await txn.timelineEvents.firstEvents(this._roomId, fragmentId, 1);
|
||||||
|
currentKey = new EventKey(firstEvent.fragmentId, firstEvent.eventIndex);
|
||||||
|
} else {
|
||||||
|
const [lastEvent] = await txn.timelineEvents.lastEvents(this._roomId, fragmentId, 1);
|
||||||
|
currentKey = new EventKey(lastEvent.fragmentId, lastEvent.eventIndex);
|
||||||
|
}
|
||||||
|
// find out if any event in chunk is already present using findFirstOrLastOccurringEventId
|
||||||
|
const eventIds = chunk.map(e => e.event_id);
|
||||||
|
const findLast = direction.isBackward;
|
||||||
|
let nonOverlappingEvents = chunk;
|
||||||
|
let neighbourFragmentEntry;
|
||||||
|
const neighbourEventId = await txn.timelineEvents.findFirstOrLastOccurringEventId(this._roomId, eventIds, findLast);
|
||||||
|
if (neighbourEventId) {
|
||||||
|
// trim overlapping events
|
||||||
|
const neighbourEventIndex = chunk.findIndex(e => e.event_id === neighbourEventId);
|
||||||
|
const start = direction.isBackward ? neighbourEventIndex + 1 : 0;
|
||||||
|
const end = direction.isBackward ? chunk.length : neighbourEventIndex;
|
||||||
|
nonOverlappingEvents = chunk.slice(start, end);
|
||||||
|
// get neighbour fragment to link it up later on
|
||||||
|
const neighbourEvent = await txn.timelineEvents.getByEventId(this._roomId, neighbourEventId);
|
||||||
|
const neighbourFragment = await txn.timelineFragments.get(neighbourEvent.fragmentId);
|
||||||
|
neighbourFragmentEntry = fragmentEntry.createNeighbourEntry(neighbourFragment);
|
||||||
|
}
|
||||||
|
|
||||||
|
// create entries for all events in chunk, add them to entries
|
||||||
|
entries = new Array(nonOverlappingEvents.length);
|
||||||
|
const reducer = direction.isBackward ? Array.prototype.reduceRight : Array.prototype.reduce;
|
||||||
|
currentKey = reducer.call(nonOverlappingEvents, (key, event, i) => {
|
||||||
|
key = key.nextKeyForDirection(direction);
|
||||||
|
const eventEntry = createEventEntry(currentKey, event);
|
||||||
|
txn.timelineEvents.insert(eventEntry);
|
||||||
|
entries[i] = new EventEntry(eventEntry, this._fragmentIdComparer);
|
||||||
|
}, currentKey);
|
||||||
|
|
||||||
|
directionalAppend(entries, fragmentEntry, direction);
|
||||||
|
// set `end` as token, and if we found an event in the step before, link up the fragments in the fragment entry
|
||||||
|
if (neighbourFragmentEntry) {
|
||||||
|
fragmentEntry.linkedFragmentId = neighbourFragmentEntry.fragmentId;
|
||||||
|
neighbourFragmentEntry.linkedFragmentId = fragmentEntry.fragmentId;
|
||||||
|
txn.timelineFragments.set(neighbourFragmentEntry.fragment);
|
||||||
|
directionalAppend(entries, neighbourFragmentEntry, direction);
|
||||||
|
|
||||||
|
// update fragmentIdComparer here after linking up fragments?
|
||||||
|
this._fragmentIdComparer.rebuild(await txn.timelineFragments.all());
|
||||||
|
}
|
||||||
|
fragmentEntry.token = end;
|
||||||
|
txn.timelineFragments.set(fragmentEntry.fragment);
|
||||||
|
} catch (err) {
|
||||||
|
txn.abort();
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
|
await txn.complete();
|
||||||
|
|
||||||
|
return entries;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//#ifdef TESTS
|
||||||
|
//import MemoryStorage from "../storage/memory/MemoryStorage.js";
|
||||||
|
|
||||||
|
export function xtests() {
|
||||||
|
const roomId = "!abc:hs.tld";
|
||||||
|
|
||||||
|
// sets sortKey and roomId on an array of entries
|
||||||
|
function createTimeline(roomId, entries) {
|
||||||
|
let key = new SortKey();
|
||||||
|
for (let entry of entries) {
|
||||||
|
if (entry.gap && entry.gap.prev_batch) {
|
||||||
|
key = key.nextKeyWithGap();
|
||||||
|
}
|
||||||
|
entry.sortKey = key;
|
||||||
|
if (entry.gap && entry.gap.next_batch) {
|
||||||
|
key = key.nextKeyWithGap();
|
||||||
|
} else if (!entry.gap) {
|
||||||
|
key = key.nextKey();
|
||||||
|
}
|
||||||
|
entry.roomId = roomId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function areSorted(entries) {
|
||||||
|
for (var i = 1; i < entries.length; i++) {
|
||||||
|
const isSorted = entries[i - 1].sortKey.compare(entries[i].sortKey) < 0;
|
||||||
|
if(!isSorted) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
"test backwards gap fill with overlapping neighbouring event": async function(assert) {
|
||||||
|
const currentPaginationToken = "abc";
|
||||||
|
const gap = {gap: {prev_batch: currentPaginationToken}};
|
||||||
|
const storage = new MemoryStorage({roomTimeline: createTimeline(roomId, [
|
||||||
|
{event: {event_id: "b"}},
|
||||||
|
{gap: {next_batch: "ghi"}},
|
||||||
|
gap,
|
||||||
|
])});
|
||||||
|
const persister = new RoomPersister({roomId, storage});
|
||||||
|
const response = {
|
||||||
|
start: currentPaginationToken,
|
||||||
|
end: "def",
|
||||||
|
chunk: [
|
||||||
|
{event_id: "a"},
|
||||||
|
{event_id: "b"},
|
||||||
|
{event_id: "c"},
|
||||||
|
{event_id: "d"},
|
||||||
|
]
|
||||||
|
};
|
||||||
|
const {newEntries, replacedEntries} = await persister.persistGapFill(gap, response);
|
||||||
|
// should only have taken events up till existing event
|
||||||
|
assert.equal(newEntries.length, 2);
|
||||||
|
assert.equal(newEntries[0].event.event_id, "c");
|
||||||
|
assert.equal(newEntries[1].event.event_id, "d");
|
||||||
|
assert.equal(replacedEntries.length, 2);
|
||||||
|
assert.equal(replacedEntries[0].gap.next_batch, "hij");
|
||||||
|
assert.equal(replacedEntries[1].gap.prev_batch, currentPaginationToken);
|
||||||
|
assert(areSorted(newEntries));
|
||||||
|
assert(areSorted(replacedEntries));
|
||||||
|
},
|
||||||
|
"test backwards gap fill with non-overlapping neighbouring event": async function(assert) {
|
||||||
|
const currentPaginationToken = "abc";
|
||||||
|
const newPaginationToken = "def";
|
||||||
|
const gap = {gap: {prev_batch: currentPaginationToken}};
|
||||||
|
const storage = new MemoryStorage({roomTimeline: createTimeline(roomId, [
|
||||||
|
{event: {event_id: "a"}},
|
||||||
|
{gap: {next_batch: "ghi"}},
|
||||||
|
gap,
|
||||||
|
])});
|
||||||
|
const persister = new RoomPersister({roomId, storage});
|
||||||
|
const response = {
|
||||||
|
start: currentPaginationToken,
|
||||||
|
end: newPaginationToken,
|
||||||
|
chunk: [
|
||||||
|
{event_id: "c"},
|
||||||
|
{event_id: "d"},
|
||||||
|
{event_id: "e"},
|
||||||
|
{event_id: "f"},
|
||||||
|
]
|
||||||
|
};
|
||||||
|
const {newEntries, replacedEntries} = await persister.persistGapFill(gap, response);
|
||||||
|
// should only have taken events up till existing event
|
||||||
|
assert.equal(newEntries.length, 5);
|
||||||
|
assert.equal(newEntries[0].gap.prev_batch, newPaginationToken);
|
||||||
|
assert.equal(newEntries[1].event.event_id, "c");
|
||||||
|
assert.equal(newEntries[2].event.event_id, "d");
|
||||||
|
assert.equal(newEntries[3].event.event_id, "e");
|
||||||
|
assert.equal(newEntries[4].event.event_id, "f");
|
||||||
|
assert(areSorted(newEntries));
|
||||||
|
|
||||||
|
assert.equal(replacedEntries.length, 1);
|
||||||
|
assert.equal(replacedEntries[0].gap.prev_batch, currentPaginationToken);
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//#endif
|
Loading…
x
Reference in New Issue
Block a user