-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathcrdt.js
More file actions
53 lines (49 loc) · 1.47 KB
/
crdt.js
File metadata and controls
53 lines (49 loc) · 1.47 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
'use strict';
const delta = (schema) => (initial, modified) => {
const result = {};
for (const key in modified) {
const next = modified[key];
const prev = initial[key];
if (typeof next === 'object' && !Array.isArray(next)) {
const def = schema[key] || {};
result[key] = delta(def)(prev || {}, next);
} else if (Array.isArray(next)) {
const add = next.filter((v) => !prev?.includes(v));
if (add.length > 0) result[key] = { add };
} else if (schema[key] === 'counter') {
result[key] = next - prev;
} else if (prev !== next) {
result[key] = next;
}
}
for (const key in initial) {
if (!(key in modified)) result[key] = { deleted: true };
}
return result;
};
const merge = (schema) => (state, delta) => {
const result = structuredClone(state);
for (const key in delta) {
const value = delta[key];
const res = result[key];
if (value?.deleted) {
delete result[key];
} else if (value?.add) {
const prev = res || [];
result[key] = Array.from(new Set([...prev, ...value.add]));
} else if (typeof value === 'object' && !Array.isArray(value)) {
const def = schema[key] || {};
result[key] = merge(def)(res || {}, value);
} else if (schema[key] === 'counter') {
result[key] = value + (res || 0);
} else {
result[key] = value;
}
}
return result;
};
const crdt = (schema) => ({
delta: delta(schema),
merge: merge(schema),
});
module.exports = crdt;