refactored lossy for clarity, added separate class for last write wins
This commit is contained in:
parent
5161ab0a85
commit
4eec3b294c
|
@ -0,0 +1,57 @@
|
||||||
|
import Debug from "debug";
|
||||||
|
import {Delta} from "../src/delta";
|
||||||
|
import {LastWriteWins} from "../src/last-write-wins";
|
||||||
|
import {Lossless} from "../src/lossless";
|
||||||
|
import {RhizomeNode} from "../src/node";
|
||||||
|
const debug = Debug('test:last-write-wins');
|
||||||
|
|
||||||
|
describe('Last write wins', () => {
|
||||||
|
|
||||||
|
describe('given that two separate writes occur', () => {
|
||||||
|
const node = new RhizomeNode();
|
||||||
|
const lossless = new Lossless(node);
|
||||||
|
|
||||||
|
const lossy = new LastWriteWins(lossless);
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
lossless.ingestDelta(new Delta({
|
||||||
|
creator: 'a',
|
||||||
|
host: 'h',
|
||||||
|
pointers: [{
|
||||||
|
localContext: "vegetable",
|
||||||
|
target: "broccoli",
|
||||||
|
targetContext: "want"
|
||||||
|
}, {
|
||||||
|
localContext: "desire",
|
||||||
|
target: 95,
|
||||||
|
}]
|
||||||
|
}));
|
||||||
|
|
||||||
|
lossless.ingestDelta(new Delta({
|
||||||
|
creator: 'a',
|
||||||
|
host: 'h',
|
||||||
|
pointers: [{
|
||||||
|
localContext: "vegetable",
|
||||||
|
target: "broccoli",
|
||||||
|
targetContext: "want"
|
||||||
|
}, {
|
||||||
|
localContext: "want",
|
||||||
|
target: 90,
|
||||||
|
}]
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('our resolver should return the most recently written value', () => {
|
||||||
|
const result = lossy.resolve(["broccoli"]);
|
||||||
|
debug('result', result);
|
||||||
|
expect(result).toMatchObject({
|
||||||
|
broccoli: {
|
||||||
|
id: "broccoli",
|
||||||
|
properties: {
|
||||||
|
want: 90
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -37,7 +37,7 @@ describe('Lossless', () => {
|
||||||
expect(lossless.view()).toMatchObject({
|
expect(lossless.view()).toMatchObject({
|
||||||
keanu: {
|
keanu: {
|
||||||
referencedAs: ["actor"],
|
referencedAs: ["actor"],
|
||||||
properties: {
|
propertyDeltas: {
|
||||||
roles: [{
|
roles: [{
|
||||||
creator: "a",
|
creator: "a",
|
||||||
host: "h",
|
host: "h",
|
||||||
|
@ -53,7 +53,7 @@ describe('Lossless', () => {
|
||||||
},
|
},
|
||||||
neo: {
|
neo: {
|
||||||
referencedAs: ["role"],
|
referencedAs: ["role"],
|
||||||
properties: {
|
propertyDeltas: {
|
||||||
actor: [{
|
actor: [{
|
||||||
creator: "a",
|
creator: "a",
|
||||||
host: "h",
|
host: "h",
|
||||||
|
@ -69,7 +69,7 @@ describe('Lossless', () => {
|
||||||
},
|
},
|
||||||
the_matrix: {
|
the_matrix: {
|
||||||
referencedAs: ["film"],
|
referencedAs: ["film"],
|
||||||
properties: {
|
propertyDeltas: {
|
||||||
cast: [{
|
cast: [{
|
||||||
creator: "a",
|
creator: "a",
|
||||||
host: "h",
|
host: "h",
|
||||||
|
@ -114,7 +114,7 @@ describe('Lossless', () => {
|
||||||
expect(lossless.view()).toMatchObject({
|
expect(lossless.view()).toMatchObject({
|
||||||
ace: {
|
ace: {
|
||||||
referencedAs: ["1", "14"],
|
referencedAs: ["1", "14"],
|
||||||
properties: {
|
propertyDeltas: {
|
||||||
value: [{
|
value: [{
|
||||||
creator: 'A',
|
creator: 'A',
|
||||||
host: 'H',
|
host: 'H',
|
||||||
|
@ -141,7 +141,7 @@ describe('Lossless', () => {
|
||||||
expect(lossless.view(undefined, filter)).toMatchObject({
|
expect(lossless.view(undefined, filter)).toMatchObject({
|
||||||
ace: {
|
ace: {
|
||||||
referencedAs: ["1"],
|
referencedAs: ["1"],
|
||||||
properties: {
|
propertyDeltas: {
|
||||||
value: [{
|
value: [{
|
||||||
creator: 'A',
|
creator: 'A',
|
||||||
host: 'H',
|
host: 'H',
|
||||||
|
@ -156,7 +156,7 @@ describe('Lossless', () => {
|
||||||
expect(lossless.view(["ace"], filter)).toMatchObject({
|
expect(lossless.view(["ace"], filter)).toMatchObject({
|
||||||
ace: {
|
ace: {
|
||||||
referencedAs: ["1"],
|
referencedAs: ["1"],
|
||||||
properties: {
|
propertyDeltas: {
|
||||||
value: [{
|
value: [{
|
||||||
creator: 'A',
|
creator: 'A',
|
||||||
host: 'H',
|
host: 'H',
|
||||||
|
@ -168,5 +168,7 @@ describe('Lossless', () => {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// TODO: Test with transactions, say A1 -- B -- A2
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,13 +1,60 @@
|
||||||
import {RhizomeNode} from "../src/node.js";
|
import Debug from 'debug';
|
||||||
import {Delta, PointerTarget} from "../src/delta.js";
|
import {Delta, PointerTarget} from "../src/delta.js";
|
||||||
import {Lossless, LosslessViewMany} from "../src/lossless.js";
|
import {lastValueFromDeltas} from "../src/last-write-wins.js";
|
||||||
import {Lossy, lastValueFromLosslessViewOne, valueFromCollapsedDelta } from "../src/lossy.js";
|
import {Lossless, LosslessViewOne} from "../src/lossless.js";
|
||||||
|
import {Lossy, valueFromCollapsedDelta} from "../src/lossy.js";
|
||||||
|
import {RhizomeNode} from "../src/node.js";
|
||||||
|
const debug = Debug('test:lossy');
|
||||||
|
|
||||||
|
type Role = {
|
||||||
|
actor: PointerTarget,
|
||||||
|
film: PointerTarget,
|
||||||
|
role: PointerTarget
|
||||||
|
};
|
||||||
|
|
||||||
|
type Summary = {
|
||||||
|
roles: Role[];
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
function initializer(): Summary {
|
||||||
|
return {
|
||||||
|
roles: []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Add more rigor to this example approach to generating a summary.
|
||||||
|
// it's really not CRDT, it likely depends on the order of the pointers.
|
||||||
|
// TODO: Prove with failing test
|
||||||
|
|
||||||
|
const reducer = (acc: Summary, cur: LosslessViewOne): Summary => {
|
||||||
|
if (cur.referencedAs.includes("role")) {
|
||||||
|
const {delta, value: actor} = lastValueFromDeltas("actor", cur.propertyDeltas["actor"]) ?? {};
|
||||||
|
if (!delta) throw new Error('expected to find delta');
|
||||||
|
if (!actor) throw new Error('expected to find actor');
|
||||||
|
const film = valueFromCollapsedDelta("film", delta);
|
||||||
|
if (!film) throw new Error('expected to find film');
|
||||||
|
acc.roles.push({
|
||||||
|
role: cur.id,
|
||||||
|
actor,
|
||||||
|
film
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
}
|
||||||
|
|
||||||
|
const resolver = (acc: Summary): Summary => {
|
||||||
|
return acc;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
describe('Lossy', () => {
|
describe('Lossy', () => {
|
||||||
describe('se a provided function to resolve entity views', () => {
|
describe('use a provided initializer, reducer, and resolver to resolve entity views', () => {
|
||||||
const node = new RhizomeNode();
|
const node = new RhizomeNode();
|
||||||
const lossless = new Lossless(node);
|
const lossless = new Lossless(node);
|
||||||
const lossy = new Lossy(lossless);
|
|
||||||
|
const lossy = new Lossy(lossless, initializer, reducer, resolver);
|
||||||
|
|
||||||
beforeAll(() => {
|
beforeAll(() => {
|
||||||
lossless.ingestDelta(new Delta({
|
lossless.ingestDelta(new Delta({
|
||||||
|
@ -36,36 +83,8 @@ describe('Lossy', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('example summary', () => {
|
it('example summary', () => {
|
||||||
type Role = {
|
const result = lossy.resolve();
|
||||||
actor: PointerTarget,
|
debug('result', result);
|
||||||
film: PointerTarget,
|
|
||||||
role: PointerTarget
|
|
||||||
};
|
|
||||||
|
|
||||||
type Summary = {
|
|
||||||
roles: Role[];
|
|
||||||
};
|
|
||||||
|
|
||||||
const resolver = (losslessView: LosslessViewMany): Summary => {
|
|
||||||
const roles: Role[] = [];
|
|
||||||
for (const [id, ent] of Object.entries(losslessView)) {
|
|
||||||
if (ent.referencedAs.includes("role")) {
|
|
||||||
const {delta, value: actor} = lastValueFromLosslessViewOne(ent, "actor") ?? {};
|
|
||||||
if (!delta) continue; // TODO: panic
|
|
||||||
if (!actor) continue; // TODO: panic
|
|
||||||
const film = valueFromCollapsedDelta(delta, "film");
|
|
||||||
if (!film) continue; // TODO: panic
|
|
||||||
roles.push({
|
|
||||||
role: id,
|
|
||||||
actor,
|
|
||||||
film
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return {roles};
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = lossy.resolve<Summary>(resolver);
|
|
||||||
expect(result).toEqual({
|
expect(result).toEqual({
|
||||||
roles: [{
|
roles: [{
|
||||||
film: "the_matrix",
|
film: "the_matrix",
|
||||||
|
|
|
@ -6,9 +6,9 @@
|
||||||
import Debug from 'debug';
|
import Debug from 'debug';
|
||||||
import {randomUUID} from "node:crypto";
|
import {randomUUID} from "node:crypto";
|
||||||
import EventEmitter from "node:events";
|
import EventEmitter from "node:events";
|
||||||
import {Delta, DeltaFilter} from "./delta.js";
|
import {Delta} from "./delta.js";
|
||||||
import {Entity, EntityProperties} from "./entity.js";
|
import {Entity, EntityProperties} from "./entity.js";
|
||||||
import {Lossy, ResolvedViewOne, Resolver} from "./lossy.js";
|
import {LastWriteWins, ResolvedViewOne} from './last-write-wins.js';
|
||||||
import {RhizomeNode} from "./node.js";
|
import {RhizomeNode} from "./node.js";
|
||||||
import {DomainEntityID} from "./types.js";
|
import {DomainEntityID} from "./types.js";
|
||||||
const debug = Debug('rz:collection');
|
const debug = Debug('rz:collection');
|
||||||
|
@ -17,7 +17,7 @@ export class Collection {
|
||||||
rhizomeNode?: RhizomeNode;
|
rhizomeNode?: RhizomeNode;
|
||||||
name: string;
|
name: string;
|
||||||
eventStream = new EventEmitter();
|
eventStream = new EventEmitter();
|
||||||
lossy?: Lossy;
|
lossy?: LastWriteWins;
|
||||||
|
|
||||||
constructor(name: string) {
|
constructor(name: string) {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
|
@ -32,7 +32,7 @@ export class Collection {
|
||||||
rhizomeConnect(rhizomeNode: RhizomeNode) {
|
rhizomeConnect(rhizomeNode: RhizomeNode) {
|
||||||
this.rhizomeNode = rhizomeNode;
|
this.rhizomeNode = rhizomeNode;
|
||||||
|
|
||||||
this.lossy = new Lossy(this.rhizomeNode.lossless);
|
this.lossy = new LastWriteWins(this.rhizomeNode.lossless);
|
||||||
|
|
||||||
// Listen for completed transactions, and emit updates to event stream
|
// Listen for completed transactions, and emit updates to event stream
|
||||||
this.rhizomeNode.lossless.eventStream.on("updated", (id) => {
|
this.rhizomeNode.lossless.eventStream.on("updated", (id) => {
|
||||||
|
@ -58,7 +58,6 @@ export class Collection {
|
||||||
newProperties: EntityProperties,
|
newProperties: EntityProperties,
|
||||||
creator: string,
|
creator: string,
|
||||||
host: string,
|
host: string,
|
||||||
resolver?: Resolver
|
|
||||||
): {
|
): {
|
||||||
transactionDelta: Delta | undefined,
|
transactionDelta: Delta | undefined,
|
||||||
deltas: Delta[]
|
deltas: Delta[]
|
||||||
|
@ -67,7 +66,7 @@ export class Collection {
|
||||||
let oldProperties: EntityProperties = {};
|
let oldProperties: EntityProperties = {};
|
||||||
|
|
||||||
if (entityId) {
|
if (entityId) {
|
||||||
const entity = this.resolve(entityId, resolver);
|
const entity = this.resolve(entityId);
|
||||||
if (entity) {
|
if (entity) {
|
||||||
oldProperties = entity.properties;
|
oldProperties = entity.properties;
|
||||||
}
|
}
|
||||||
|
@ -155,7 +154,6 @@ export class Collection {
|
||||||
async put(
|
async put(
|
||||||
entityId: DomainEntityID | undefined,
|
entityId: DomainEntityID | undefined,
|
||||||
properties: EntityProperties,
|
properties: EntityProperties,
|
||||||
resolver?: Resolver
|
|
||||||
): Promise<ResolvedViewOne> {
|
): Promise<ResolvedViewOne> {
|
||||||
if (!this.rhizomeNode) throw new Error('collection not connecte to rhizome');
|
if (!this.rhizomeNode) throw new Error('collection not connecte to rhizome');
|
||||||
|
|
||||||
|
@ -173,7 +171,6 @@ export class Collection {
|
||||||
properties,
|
properties,
|
||||||
this.rhizomeNode?.config.creator,
|
this.rhizomeNode?.config.creator,
|
||||||
this.rhizomeNode?.config.peerId,
|
this.rhizomeNode?.config.peerId,
|
||||||
resolver,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const ingested = new Promise<boolean>((resolve) => {
|
const ingested = new Promise<boolean>((resolve) => {
|
||||||
|
@ -204,21 +201,19 @@ export class Collection {
|
||||||
|
|
||||||
await ingested;
|
await ingested;
|
||||||
|
|
||||||
const res = this.resolve(entityId, resolver);
|
const res = this.resolve(entityId);
|
||||||
if (!res) throw new Error("could not get what we just put!");
|
if (!res) throw new Error("could not get what we just put!");
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
resolve<T = ResolvedViewOne>(
|
resolve(
|
||||||
id: string,
|
id: string
|
||||||
resolver?: Resolver,
|
): ResolvedViewOne | undefined {
|
||||||
deltaFilter?: DeltaFilter
|
|
||||||
): T | undefined {
|
|
||||||
if (!this.rhizomeNode) throw new Error('collection not connected to rhizome');
|
if (!this.rhizomeNode) throw new Error('collection not connected to rhizome');
|
||||||
if (!this.lossy) throw new Error('lossy view not initialized');
|
if (!this.lossy) throw new Error('lossy view not initialized');
|
||||||
|
|
||||||
const res = this.lossy.resolve(resolver, [id], deltaFilter) || {};
|
const res = this.lossy.resolve([id]) || {};
|
||||||
|
|
||||||
return res[id] as T;
|
return res[id];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,104 @@
|
||||||
|
import Debug from 'debug';
|
||||||
|
import {Lossy, valueFromCollapsedDelta} from './lossy.js';
|
||||||
|
|
||||||
|
import {EntityProperties} from "./entity.js";
|
||||||
|
import {CollapsedDelta, Lossless, LosslessViewOne} from "./lossless.js";
|
||||||
|
import {DomainEntityID, PropertyID, PropertyTypes, Timestamp, ViewMany} from "./types.js";
|
||||||
|
|
||||||
|
const debug = Debug('rz:lossy:last-write-wins');
|
||||||
|
|
||||||
|
type TimestampedProperty = {
|
||||||
|
value: PropertyTypes,
|
||||||
|
timeUpdated: Timestamp
|
||||||
|
};
|
||||||
|
|
||||||
|
type TimestampedProperties = {
|
||||||
|
[key: PropertyID]: TimestampedProperty
|
||||||
|
};
|
||||||
|
|
||||||
|
export type LossyViewOne<T = TimestampedProperties> = {
|
||||||
|
id: DomainEntityID;
|
||||||
|
properties: T;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type LossyViewMany<T = TimestampedProperties> = ViewMany<LossyViewOne<T>>;
|
||||||
|
|
||||||
|
export type ResolvedViewOne = LossyViewOne<EntityProperties>;
|
||||||
|
export type ResolvedViewMany = ViewMany<ResolvedViewOne>;
|
||||||
|
|
||||||
|
type Accumulator = LossyViewMany<TimestampedProperties>;
|
||||||
|
type Result = LossyViewMany<EntityProperties>;
|
||||||
|
|
||||||
|
// Function for resolving a value for an entity by last write wins
|
||||||
|
export function lastValueFromDeltas(
|
||||||
|
key: string,
|
||||||
|
deltas?: CollapsedDelta[]
|
||||||
|
): {
|
||||||
|
delta?: CollapsedDelta,
|
||||||
|
value?: string | number,
|
||||||
|
timeUpdated?: number
|
||||||
|
} | undefined {
|
||||||
|
const res: {
|
||||||
|
delta?: CollapsedDelta,
|
||||||
|
value?: string | number,
|
||||||
|
timeUpdated?: number
|
||||||
|
} = {};
|
||||||
|
res.timeUpdated = 0;
|
||||||
|
|
||||||
|
for (const delta of deltas || []) {
|
||||||
|
const value = valueFromCollapsedDelta(key, delta);
|
||||||
|
if (value === undefined) continue;
|
||||||
|
if (res.timeUpdated && delta.timeCreated < res.timeUpdated) continue;
|
||||||
|
res.delta = delta;
|
||||||
|
res.value = value;
|
||||||
|
res.timeUpdated = delta.timeCreated;
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
function initializer(): Accumulator {
|
||||||
|
return {};
|
||||||
|
};
|
||||||
|
|
||||||
|
function reducer(acc: Accumulator, cur: LosslessViewOne): Accumulator {
|
||||||
|
if (!acc[cur.id]) {
|
||||||
|
acc[cur.id] = {id: cur.id, properties: {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const [key, deltas] of Object.entries(cur.propertyDeltas)) {
|
||||||
|
debug(`reducer: looking for value for key ${key}`);
|
||||||
|
const {value, timeUpdated} = lastValueFromDeltas(key, deltas) || {};
|
||||||
|
debug(`reducer: key ${key} value ${value} timeUpdated ${timeUpdated}`);
|
||||||
|
if (!value || !timeUpdated) continue;
|
||||||
|
|
||||||
|
if (timeUpdated > (acc[cur.id].properties[key]?.timeUpdated || 0)) {
|
||||||
|
acc[cur.id].properties[key] = {
|
||||||
|
value,
|
||||||
|
timeUpdated
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
};
|
||||||
|
|
||||||
|
function resolver(cur: Accumulator): Result {
|
||||||
|
const res: Result = {};
|
||||||
|
|
||||||
|
for (const [id, ent] of Object.entries(cur)) {
|
||||||
|
res[id] = {id, properties: {}};
|
||||||
|
for (const [key, {value}] of Object.entries(ent.properties)) {
|
||||||
|
res[id].properties[key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
|
};
|
||||||
|
|
||||||
|
export class LastWriteWins extends Lossy<Accumulator, Result> {
|
||||||
|
constructor(
|
||||||
|
readonly lossless: Lossless,
|
||||||
|
) {
|
||||||
|
super(lossless, initializer, reducer, resolver);
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,10 +3,10 @@
|
||||||
|
|
||||||
import Debug from 'debug';
|
import Debug from 'debug';
|
||||||
import EventEmitter from 'events';
|
import EventEmitter from 'events';
|
||||||
import {Delta, DeltaFilter, DeltaNetworkImage} from './delta.js';
|
import {Delta, DeltaFilter, DeltaID, DeltaNetworkImage} from './delta.js';
|
||||||
|
import {RhizomeNode} from './node.js';
|
||||||
import {Transactions} from './transactions.js';
|
import {Transactions} from './transactions.js';
|
||||||
import {DomainEntityID, PropertyID, PropertyTypes, TransactionID, ViewMany} from "./types.js";
|
import {DomainEntityID, PropertyID, PropertyTypes, TransactionID, ViewMany} from "./types.js";
|
||||||
import {RhizomeNode} from './node.js';
|
|
||||||
const debug = Debug('rz:lossless');
|
const debug = Debug('rz:lossless');
|
||||||
|
|
||||||
export type CollapsedPointer = {[key: PropertyID]: PropertyTypes};
|
export type CollapsedPointer = {[key: PropertyID]: PropertyTypes};
|
||||||
|
@ -16,8 +16,9 @@ export type CollapsedDelta = Omit<DeltaNetworkImage, 'pointers'> & {
|
||||||
};
|
};
|
||||||
|
|
||||||
export type LosslessViewOne = {
|
export type LosslessViewOne = {
|
||||||
|
id: DomainEntityID,
|
||||||
referencedAs: string[];
|
referencedAs: string[];
|
||||||
properties: {
|
propertyDeltas: {
|
||||||
[key: PropertyID]: CollapsedDelta[]
|
[key: PropertyID]: CollapsedDelta[]
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -27,12 +28,9 @@ export type LosslessViewMany = ViewMany<LosslessViewOne>;
|
||||||
class LosslessEntityMap extends Map<DomainEntityID, LosslessEntity> {};
|
class LosslessEntityMap extends Map<DomainEntityID, LosslessEntity> {};
|
||||||
|
|
||||||
class LosslessEntity {
|
class LosslessEntity {
|
||||||
id: DomainEntityID;
|
|
||||||
properties = new Map<PropertyID, Set<Delta>>();
|
properties = new Map<PropertyID, Set<Delta>>();
|
||||||
|
|
||||||
constructor(id: DomainEntityID) {
|
constructor(readonly lossless: Lossless, readonly id: DomainEntityID) {}
|
||||||
this.id = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
addDelta(delta: Delta) {
|
addDelta(delta: Delta) {
|
||||||
const targetContexts = delta.pointers
|
const targetContexts = delta.pointers
|
||||||
|
@ -48,6 +46,7 @@ class LosslessEntity {
|
||||||
}
|
}
|
||||||
|
|
||||||
propertyDeltas.add(delta);
|
propertyDeltas.add(delta);
|
||||||
|
debug(`[${this.lossless.rhizomeNode.config.peerId}]`, `entity ${this.id} added delta:`, JSON.stringify(delta));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,12 +70,12 @@ export class Lossless {
|
||||||
|
|
||||||
constructor(readonly rhizomeNode: RhizomeNode) {
|
constructor(readonly rhizomeNode: RhizomeNode) {
|
||||||
this.transactions = new Transactions(this);
|
this.transactions = new Transactions(this);
|
||||||
this.transactions.eventStream.on("completed", (transactionId) => {
|
this.transactions.eventStream.on("completed", (transactionId, deltaIds) => {
|
||||||
debug(`[${this.rhizomeNode.config.peerId}]`, `Completed transaction ${transactionId}`);
|
debug(`[${this.rhizomeNode.config.peerId}]`, `Completed transaction ${transactionId}`);
|
||||||
const transaction = this.transactions.get(transactionId);
|
const transaction = this.transactions.get(transactionId);
|
||||||
if (!transaction) return;
|
if (!transaction) return;
|
||||||
for (const id of transaction.entityIds) {
|
for (const id of transaction.entityIds) {
|
||||||
this.eventStream.emit("updated", id);
|
this.eventStream.emit("updated", id, deltaIds);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -91,7 +90,7 @@ export class Lossless {
|
||||||
let ent = this.domainEntities.get(target);
|
let ent = this.domainEntities.get(target);
|
||||||
|
|
||||||
if (!ent) {
|
if (!ent) {
|
||||||
ent = new LosslessEntity(target);
|
ent = new LosslessEntity(this, target);
|
||||||
this.domainEntities.set(target, ent);
|
this.domainEntities.set(target, ent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -116,28 +115,49 @@ export class Lossless {
|
||||||
if (!transactionId) {
|
if (!transactionId) {
|
||||||
// No transaction -- we can issue an update event immediately
|
// No transaction -- we can issue an update event immediately
|
||||||
for (const id of targets) {
|
for (const id of targets) {
|
||||||
this.eventStream.emit("updated", id);
|
this.eventStream.emit("updated", id, [delta.id]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return transactionId;
|
return transactionId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
viewSpecific(entityId: DomainEntityID, deltaIds: DeltaID[], deltaFilter?: DeltaFilter): LosslessViewOne | undefined {
|
||||||
|
debug(`[${this.rhizomeNode.config.peerId}]`, `viewSpecific, deltaIds:`, JSON.stringify(deltaIds));
|
||||||
|
const combinedFilter = (delta: Delta) => {
|
||||||
|
debug(`[${this.rhizomeNode.config.peerId}]`, `combinedFilter, deltaIds:`, JSON.stringify(deltaIds));
|
||||||
|
if (!deltaIds.includes(delta.id)) {
|
||||||
|
debug(`[${this.rhizomeNode.config.peerId}]`, `Excluding delta ${delta.id} because it's not in the requested list of deltas`);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!deltaFilter) return true;
|
||||||
|
return deltaFilter(delta);
|
||||||
|
};
|
||||||
|
const res = this.view([entityId], (delta) => combinedFilter(delta));
|
||||||
|
return res[entityId];
|
||||||
|
}
|
||||||
|
|
||||||
view(entityIds?: DomainEntityID[], deltaFilter?: DeltaFilter): LosslessViewMany {
|
view(entityIds?: DomainEntityID[], deltaFilter?: DeltaFilter): LosslessViewMany {
|
||||||
const view: LosslessViewMany = {};
|
const view: LosslessViewMany = {};
|
||||||
entityIds = entityIds ?? Array.from(this.domainEntities.keys());
|
entityIds = entityIds ?? Array.from(this.domainEntities.keys());
|
||||||
|
|
||||||
for (const id of entityIds) {
|
for (const id of entityIds) {
|
||||||
const ent = this.domainEntities.get(id);
|
const ent = this.domainEntities.get(id);
|
||||||
if (!ent) continue;
|
if (!ent) continue;
|
||||||
|
|
||||||
|
|
||||||
const referencedAs = new Set<string>();
|
const referencedAs = new Set<string>();
|
||||||
const properties: {
|
const propertyDeltas: {
|
||||||
[key: PropertyID]: CollapsedDelta[]
|
[key: PropertyID]: CollapsedDelta[]
|
||||||
} = {};
|
} = {};
|
||||||
|
|
||||||
for (const [key, deltas] of ent.properties.entries()) {
|
for (const [key, deltas] of ent.properties.entries()) {
|
||||||
properties[key] = properties[key] || [];
|
propertyDeltas[key] = propertyDeltas[key] || [];
|
||||||
|
|
||||||
for (const delta of deltas) {
|
for (const delta of deltas) {
|
||||||
|
if (deltaFilter && !deltaFilter(delta)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// If this delta is part of a transaction,
|
// If this delta is part of a transaction,
|
||||||
// we need to be able to wait for the whole transaction.
|
// we need to be able to wait for the whole transaction.
|
||||||
if (delta.transactionId) {
|
if (delta.transactionId) {
|
||||||
|
@ -148,11 +168,6 @@ export class Lossless {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (deltaFilter) {
|
|
||||||
const include = deltaFilter(delta);
|
|
||||||
if (!include) continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const pointers: CollapsedPointer[] = [];
|
const pointers: CollapsedPointer[] = [];
|
||||||
|
|
||||||
for (const {localContext, target} of delta.pointers) {
|
for (const {localContext, target} of delta.pointers) {
|
||||||
|
@ -162,20 +177,21 @@ export class Lossless {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const collapsedDelta: CollapsedDelta = {
|
propertyDeltas[key].push({
|
||||||
...delta,
|
...delta,
|
||||||
pointers
|
pointers
|
||||||
};
|
});
|
||||||
|
|
||||||
properties[key].push(collapsedDelta);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
view[ent.id] = {
|
view[ent.id] = {
|
||||||
|
id: ent.id,
|
||||||
referencedAs: Array.from(referencedAs.values()),
|
referencedAs: Array.from(referencedAs.values()),
|
||||||
properties
|
propertyDeltas
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
debug(`[${this.rhizomeNode.config.peerId}]`, `Returning view:`, JSON.stringify(view, null, 2));
|
||||||
return view;
|
return view;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
127
src/lossy.ts
127
src/lossy.ts
|
@ -5,36 +5,20 @@
|
||||||
// We can achieve this via functional expression, encoded as JSON-Logic.
|
// We can achieve this via functional expression, encoded as JSON-Logic.
|
||||||
// Fields in the output can be described as transformations
|
// Fields in the output can be described as transformations
|
||||||
|
|
||||||
// import Debug from 'debug';
|
import Debug from 'debug';
|
||||||
import {DeltaFilter} from "./delta.js";
|
import {DeltaFilter, DeltaID} from "./delta.js";
|
||||||
import {CollapsedDelta, Lossless, LosslessViewMany, LosslessViewOne} from "./lossless.js";
|
import {CollapsedDelta, Lossless, LosslessViewOne} from "./lossless.js";
|
||||||
import {DomainEntityID, PropertyID, PropertyTypes, Timestamp, ViewMany} from "./types.js";
|
import {DomainEntityID} from "./types.js";
|
||||||
// const debug = Debug('rz:lossy');
|
const debug = Debug('rz:lossy');
|
||||||
|
|
||||||
type TimestampedProperty = {
|
export type Initializer<Accumulator> = (v: LosslessViewOne) => Accumulator;
|
||||||
value: PropertyTypes,
|
export type Reducer<Accumulator> = (acc: Accumulator, cur: LosslessViewOne) => Accumulator;
|
||||||
timeUpdated: Timestamp
|
export type Resolver<Accumulator, Result> = (cur: Accumulator) => Result;
|
||||||
};
|
|
||||||
|
|
||||||
export type LossyViewOne<T = TimestampedProperty> = {
|
|
||||||
id: DomainEntityID;
|
|
||||||
properties: {
|
|
||||||
[key: PropertyID]: T
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export type LossyViewMany<T> = ViewMany<LossyViewOne<T>>;
|
|
||||||
|
|
||||||
export type ResolvedViewOne = LossyViewOne<PropertyTypes>;
|
|
||||||
export type ResolvedViewMany = ViewMany<ResolvedViewOne>;
|
|
||||||
|
|
||||||
export type Resolver<T = ResolvedViewMany> =
|
|
||||||
(losslessView: LosslessViewMany) => T;
|
|
||||||
|
|
||||||
// Extract a particular value from a delta's pointers
|
// Extract a particular value from a delta's pointers
|
||||||
export function valueFromCollapsedDelta(
|
export function valueFromCollapsedDelta(
|
||||||
delta: CollapsedDelta,
|
key: string,
|
||||||
key: string
|
delta: CollapsedDelta
|
||||||
): string | number | undefined {
|
): string | number | undefined {
|
||||||
for (const pointer of delta.pointers) {
|
for (const pointer of delta.pointers) {
|
||||||
for (const [k, value] of Object.entries(pointer)) {
|
for (const [k, value] of Object.entries(pointer)) {
|
||||||
|
@ -45,75 +29,56 @@ export function valueFromCollapsedDelta(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Function for resolving a value for an entity by last write wins
|
|
||||||
export function lastValueFromLosslessViewOne(
|
|
||||||
ent: LosslessViewOne,
|
|
||||||
key: string
|
|
||||||
): {
|
|
||||||
delta?: CollapsedDelta,
|
|
||||||
value?: string | number,
|
|
||||||
timeUpdated?: number
|
|
||||||
} | undefined {
|
|
||||||
const res: {
|
|
||||||
delta?: CollapsedDelta,
|
|
||||||
value?: string | number,
|
|
||||||
timeUpdated?: number
|
|
||||||
} = {};
|
|
||||||
res.timeUpdated = 0;
|
|
||||||
|
|
||||||
for (const delta of ent.properties[key] || []) {
|
|
||||||
const value = valueFromCollapsedDelta(delta, key);
|
|
||||||
if (value === undefined) continue;
|
|
||||||
if (delta.timeCreated < res.timeUpdated) continue;
|
|
||||||
res.delta = delta;
|
|
||||||
res.value = value;
|
|
||||||
res.timeUpdated = delta.timeCreated;
|
|
||||||
}
|
|
||||||
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: Incremental updates of lossy models. For example, with last-write-wins,
|
// TODO: Incremental updates of lossy models. For example, with last-write-wins,
|
||||||
// we keep the timeUpdated for each field. A second stage resolver can rearrange
|
// we keep the timeUpdated for each field. A second stage resolver can rearrange
|
||||||
// the data structure to a preferred shape and may discard the timeUpdated info.
|
// the data structure to a preferred shape and may discard the timeUpdated info.
|
||||||
export class Lossy {
|
export class Lossy<Accumulator, Result> {
|
||||||
lossless: Lossless;
|
deltaFilter?: DeltaFilter;
|
||||||
|
accumulator?: Accumulator;
|
||||||
|
|
||||||
constructor(lossless: Lossless) {
|
constructor(
|
||||||
this.lossless = lossless;
|
readonly lossless: Lossless,
|
||||||
|
readonly initializer: Initializer<Accumulator>,
|
||||||
|
readonly reducer: Reducer<Accumulator>,
|
||||||
|
readonly resolver: Resolver<Accumulator, Result>,
|
||||||
|
) {
|
||||||
|
this.lossless.eventStream.on("updated", (id, deltaIds) => {
|
||||||
|
debug(`[${this.lossless.rhizomeNode.config.peerId}] entity ${id} updated, deltaIds:`,
|
||||||
|
JSON.stringify(deltaIds));
|
||||||
|
this.ingestUpdate(id, deltaIds);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
ingestUpdate(id: DomainEntityID, deltaIds: DeltaID[]) {
|
||||||
|
debug(`[${this.lossless.rhizomeNode.config.peerId}] prior to ingesting update, deltaIds:`, deltaIds);
|
||||||
|
const losslessPartial = this.lossless.viewSpecific(id, deltaIds, this.deltaFilter);
|
||||||
|
|
||||||
|
debug(`[${this.lossless.rhizomeNode.config.peerId}] prior to ingesting update, lossless partial:`,
|
||||||
|
JSON.stringify(losslessPartial, null, 2));
|
||||||
|
|
||||||
|
if (!losslessPartial) return;
|
||||||
|
|
||||||
|
const latest = this.accumulator || this.initializer(losslessPartial);
|
||||||
|
this.accumulator = this.reducer(latest, losslessPartial);
|
||||||
|
|
||||||
|
debug(`[${this.lossless.rhizomeNode.config.peerId}] after ingesting update, entity ${id} accumulator:`,
|
||||||
|
JSON.stringify(this.accumulator, null, 2));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Using the lossless view of some given domain entities,
|
// Using the lossless view of some given domain entities,
|
||||||
// apply a filter to the deltas composing that lossless view,
|
// apply a filter to the deltas composing that lossless view,
|
||||||
// and then apply a supplied resolver function which receives
|
// and then apply a supplied resolver function which receives
|
||||||
// the filtered lossless view as input.
|
// the filtered lossless view as input.
|
||||||
// TODO: Cache things!
|
// resolve<T = ResolvedViewOne>(fn?: Resolver<T> | Resolver, entityIds?: DomainEntityID[], deltaFilter?: DeltaFilter): T {
|
||||||
resolve<T = ResolvedViewOne>(fn?: Resolver<T> | Resolver, entityIds?: DomainEntityID[], deltaFilter?: DeltaFilter): T {
|
resolve(entityIds?: DomainEntityID[]): Result | undefined {
|
||||||
if (!fn) {
|
if (!entityIds) {
|
||||||
fn = (v) => this.defaultResolver(v);
|
entityIds = Array.from(this.lossless.domainEntities.keys());
|
||||||
}
|
|
||||||
const losslessView = this.lossless.view(entityIds, deltaFilter);
|
|
||||||
return fn(losslessView) as T;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
defaultResolver(losslessView: LosslessViewMany): ResolvedViewMany {
|
if (!this.accumulator) return undefined;
|
||||||
const resolved: ResolvedViewMany = {};
|
|
||||||
|
|
||||||
// debug(`[${this.lossless.rhizomeNode.config.peerId}]`, 'Default resolver, lossless view', JSON.stringify(losslessView));
|
return this.resolver(this.accumulator);
|
||||||
for (const [id, ent] of Object.entries(losslessView)) {
|
|
||||||
resolved[id] = {id, properties: {}};
|
|
||||||
|
|
||||||
for (const key of Object.keys(ent.properties)) {
|
|
||||||
const {value} = lastValueFromLosslessViewOne(ent, key) || {};
|
|
||||||
|
|
||||||
// debug(`[${this.lossless.rhizomeNode.config.peerId}]`, `[ ${key} ] = ${value}`);
|
|
||||||
resolved[id].properties[key] = value;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
return resolved;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate a rule
|
// Generate a rule
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import Debug from "debug";
|
import Debug from "debug";
|
||||||
import EventEmitter from "events";
|
import EventEmitter from "events";
|
||||||
import {Delta, DeltaID} from "./delta.js";
|
import {Delta, DeltaID} from "./delta.js";
|
||||||
import {DomainEntityID, TransactionID} from "./types.js";
|
|
||||||
import {Lossless} from "./lossless.js";
|
import {Lossless} from "./lossless.js";
|
||||||
|
import {DomainEntityID, TransactionID} from "./types.js";
|
||||||
const debug = Debug('rz:transactions');
|
const debug = Debug('rz:transactions');
|
||||||
|
|
||||||
function getDeltaTransactionId(delta: Delta): TransactionID | undefined {
|
function getDeltaTransactionId(delta: Delta): TransactionID | undefined {
|
||||||
|
@ -53,6 +53,19 @@ export class Transaction {
|
||||||
size?: number;
|
size?: number;
|
||||||
receivedDeltaIds = new Set<DeltaID>();
|
receivedDeltaIds = new Set<DeltaID>();
|
||||||
entityIds = new Set<DomainEntityID>();
|
entityIds = new Set<DomainEntityID>();
|
||||||
|
resolved: Promise<boolean>;
|
||||||
|
|
||||||
|
constructor(readonly transactions: Transactions, readonly id: TransactionID) {
|
||||||
|
this.resolved = new Promise((resolve) => {
|
||||||
|
this.transactions.eventStream.on("completed", (transactionId) => {
|
||||||
|
if (transactionId === this.id) resolve(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
getReceivedDeltaIds() {
|
||||||
|
return Array.from(this.receivedDeltaIds.values());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Transactions {
|
export class Transactions {
|
||||||
|
@ -68,13 +81,14 @@ export class Transactions {
|
||||||
getOrInit(id: TransactionID): Transaction {
|
getOrInit(id: TransactionID): Transaction {
|
||||||
let t = this.transactions.get(id);
|
let t = this.transactions.get(id);
|
||||||
if (!t) {
|
if (!t) {
|
||||||
t = new Transaction();
|
t = new Transaction(this, id);
|
||||||
this.transactions.set(id, t);
|
this.transactions.set(id, t);
|
||||||
}
|
}
|
||||||
return t;
|
return t;
|
||||||
}
|
}
|
||||||
|
|
||||||
ingestDelta(delta: Delta, targets: DomainEntityID[]): TransactionID | undefined {
|
ingestDelta(delta: Delta, targets: DomainEntityID[]): TransactionID | undefined {
|
||||||
|
// This delta may be part of a transaction
|
||||||
{
|
{
|
||||||
const transactionId = getDeltaTransactionId(delta);
|
const transactionId = getDeltaTransactionId(delta);
|
||||||
if (transactionId) {
|
if (transactionId) {
|
||||||
|
@ -83,7 +97,6 @@ export class Transactions {
|
||||||
t.entityIds.add(id);
|
t.entityIds.add(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This delta is part of a transaction
|
|
||||||
// Add this to the delta's data structure for quick reference
|
// Add this to the delta's data structure for quick reference
|
||||||
delta.transactionId = transactionId;
|
delta.transactionId = transactionId;
|
||||||
|
|
||||||
|
@ -92,25 +105,25 @@ export class Transactions {
|
||||||
|
|
||||||
// Notify that the transaction is complete
|
// Notify that the transaction is complete
|
||||||
if (this.isComplete(transactionId)) {
|
if (this.isComplete(transactionId)) {
|
||||||
this.eventStream.emit("completed", transactionId);
|
this.eventStream.emit("completed", t.id, t.getReceivedDeltaIds());
|
||||||
}
|
}
|
||||||
|
|
||||||
return transactionId;
|
return transactionId;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This delta may describe a transaction
|
||||||
{
|
{
|
||||||
const {transactionId, size} = getTransactionSize(delta) || {};
|
const {transactionId, size} = getTransactionSize(delta) || {};
|
||||||
if (transactionId && size) {
|
if (transactionId && size) {
|
||||||
// This delta describes a transaction
|
|
||||||
|
|
||||||
debug(`[${this.lossless.rhizomeNode.config.peerId}]`, `Transaction ${transactionId} has size ${size}`);
|
debug(`[${this.lossless.rhizomeNode.config.peerId}]`, `Transaction ${transactionId} has size ${size}`);
|
||||||
|
|
||||||
this.setSize(transactionId, size as number);
|
this.setSize(transactionId, size as number);
|
||||||
|
|
||||||
// Check if the transaction is complete
|
// Check if the transaction is complete
|
||||||
if (this.isComplete(transactionId)) {
|
if (this.isComplete(transactionId)) {
|
||||||
this.eventStream.emit("completed", transactionId);
|
const t = this.getOrInit(transactionId);
|
||||||
|
this.eventStream.emit("completed", t.id, t.getReceivedDeltaIds());
|
||||||
}
|
}
|
||||||
|
|
||||||
return transactionId;
|
return transactionId;
|
||||||
|
@ -124,8 +137,16 @@ export class Transactions {
|
||||||
}
|
}
|
||||||
|
|
||||||
isComplete(id: TransactionID) {
|
isComplete(id: TransactionID) {
|
||||||
const t = this.getOrInit(id);
|
const t = this.get(id);
|
||||||
return t.size !== undefined && t.receivedDeltaIds.size === t.size;
|
if (!t) return false;
|
||||||
|
if (t.size === undefined) return false;
|
||||||
|
return t.receivedDeltaIds.size === t.size;
|
||||||
|
}
|
||||||
|
|
||||||
|
async waitFor(id: TransactionID) {
|
||||||
|
const t = this.get(id);
|
||||||
|
if (!t) return;
|
||||||
|
await t.resolved;
|
||||||
}
|
}
|
||||||
|
|
||||||
setSize(id: TransactionID, size: number) {
|
setSize(id: TransactionID, size: number) {
|
||||||
|
|
Loading…
Reference in New Issue