@liveblocks/core 3.16.0-flow2 → 3.16.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs +1072 -880
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +258 -116
- package/dist/index.d.ts +258 -116
- package/dist/index.js +985 -793
- package/dist/index.js.map +1 -1
- package/package.json +2 -2
package/dist/index.js
CHANGED
|
@@ -6,7 +6,7 @@ var __export = (target, all) => {
|
|
|
6
6
|
|
|
7
7
|
// src/version.ts
|
|
8
8
|
var PKG_NAME = "@liveblocks/core";
|
|
9
|
-
var PKG_VERSION = "3.16.0
|
|
9
|
+
var PKG_VERSION = "3.16.0";
|
|
10
10
|
var PKG_FORMAT = "esm";
|
|
11
11
|
|
|
12
12
|
// src/dupe-detection.ts
|
|
@@ -3219,10 +3219,26 @@ var ServerMsgCode = Object.freeze({
|
|
|
3219
3219
|
COMMENT_REACTION_ADDED: 405,
|
|
3220
3220
|
COMMENT_REACTION_REMOVED: 406,
|
|
3221
3221
|
COMMENT_METADATA_UPDATED: 409,
|
|
3222
|
+
// For Feeds
|
|
3223
|
+
FEEDS_LIST: 500,
|
|
3224
|
+
FEEDS_ADDED: 501,
|
|
3225
|
+
FEEDS_UPDATED: 502,
|
|
3226
|
+
FEED_DELETED: 503,
|
|
3227
|
+
FEED_MESSAGES_LIST: 504,
|
|
3228
|
+
FEED_MESSAGES_ADDED: 505,
|
|
3229
|
+
FEED_MESSAGES_UPDATED: 506,
|
|
3230
|
+
FEED_MESSAGES_DELETED: 507,
|
|
3231
|
+
FEED_REQUEST_FAILED: 508,
|
|
3222
3232
|
// Error codes
|
|
3223
3233
|
REJECT_STORAGE_OP: 299
|
|
3224
3234
|
// Sent if a mutation was not allowed on the server (i.e. due to permissions, limit exceeded, etc)
|
|
3225
3235
|
});
|
|
3236
|
+
var FeedRequestErrorCode = {
|
|
3237
|
+
INTERNAL: "INTERNAL",
|
|
3238
|
+
FEED_ALREADY_EXISTS: "FEED_ALREADY_EXISTS",
|
|
3239
|
+
FEED_NOT_FOUND: "FEED_NOT_FOUND",
|
|
3240
|
+
FEED_MESSAGE_NOT_FOUND: "FEED_MESSAGE_NOT_FOUND"
|
|
3241
|
+
};
|
|
3226
3242
|
|
|
3227
3243
|
// src/types/IWebSocket.ts
|
|
3228
3244
|
var WebsocketCloseCodes = /* @__PURE__ */ ((WebsocketCloseCodes2) => {
|
|
@@ -5186,6 +5202,7 @@ var Permission = /* @__PURE__ */ ((Permission2) => {
|
|
|
5186
5202
|
Permission2["PresenceWrite"] = "room:presence:write";
|
|
5187
5203
|
Permission2["CommentsWrite"] = "comments:write";
|
|
5188
5204
|
Permission2["CommentsRead"] = "comments:read";
|
|
5205
|
+
Permission2["FeedsWrite"] = "feeds:write";
|
|
5189
5206
|
return Permission2;
|
|
5190
5207
|
})(Permission || {});
|
|
5191
5208
|
function canWriteStorage(scopes) {
|
|
@@ -6302,6 +6319,7 @@ var AbstractCrdt = class {
|
|
|
6302
6319
|
}
|
|
6303
6320
|
/**
|
|
6304
6321
|
* @internal
|
|
6322
|
+
*
|
|
6305
6323
|
* Return an snapshot of this Live tree for use in DevTools.
|
|
6306
6324
|
*/
|
|
6307
6325
|
toTreeNode(key) {
|
|
@@ -6311,14 +6329,6 @@ var AbstractCrdt = class {
|
|
|
6311
6329
|
}
|
|
6312
6330
|
return this.#cachedTreeNode;
|
|
6313
6331
|
}
|
|
6314
|
-
/**
|
|
6315
|
-
* @private
|
|
6316
|
-
* Returns true if the cached immutable snapshot exists and is
|
|
6317
|
-
* reference-equal to the given value. Does not trigger a recompute.
|
|
6318
|
-
*/
|
|
6319
|
-
immutableIs(value) {
|
|
6320
|
-
return this.#cachedImmutable !== void 0 && this.#cachedImmutable === value;
|
|
6321
|
-
}
|
|
6322
6332
|
/**
|
|
6323
6333
|
* Return an immutable snapshot of this Live node and its children.
|
|
6324
6334
|
*/
|
|
@@ -7866,652 +7876,248 @@ var LiveMap = class _LiveMap extends AbstractCrdt {
|
|
|
7866
7876
|
}
|
|
7867
7877
|
};
|
|
7868
7878
|
|
|
7869
|
-
// src/
|
|
7870
|
-
|
|
7871
|
-
|
|
7872
|
-
|
|
7873
|
-
|
|
7874
|
-
|
|
7875
|
-
|
|
7876
|
-
|
|
7877
|
-
|
|
7878
|
-
|
|
7879
|
-
|
|
7880
|
-
|
|
7881
|
-
|
|
7882
|
-
|
|
7883
|
-
|
|
7884
|
-
|
|
7885
|
-
|
|
7886
|
-
|
|
7887
|
-
|
|
7888
|
-
|
|
7889
|
-
|
|
7890
|
-
|
|
7891
|
-
|
|
7892
|
-
|
|
7893
|
-
|
|
7894
|
-
|
|
7895
|
-
|
|
7896
|
-
|
|
7897
|
-
|
|
7898
|
-
|
|
7899
|
-
|
|
7900
|
-
|
|
7901
|
-
|
|
7902
|
-
|
|
7903
|
-
}
|
|
7904
|
-
function liveListToJson(value) {
|
|
7905
|
-
return lsonListToJson(value.toArray());
|
|
7906
|
-
}
|
|
7907
|
-
function lsonToJson(value) {
|
|
7908
|
-
if (value instanceof LiveObject) {
|
|
7909
|
-
return liveObjectToJson(value);
|
|
7910
|
-
} else if (value instanceof LiveList) {
|
|
7911
|
-
return liveListToJson(value);
|
|
7912
|
-
} else if (value instanceof LiveMap) {
|
|
7913
|
-
return liveMapToJson(value);
|
|
7914
|
-
} else if (value instanceof LiveRegister) {
|
|
7915
|
-
return value.data;
|
|
7916
|
-
}
|
|
7917
|
-
if (Array.isArray(value)) {
|
|
7918
|
-
return lsonListToJson(value);
|
|
7919
|
-
} else if (isPlainObject(value)) {
|
|
7920
|
-
return lsonObjectToJson(value);
|
|
7921
|
-
}
|
|
7922
|
-
return value;
|
|
7923
|
-
}
|
|
7924
|
-
function deepLiveify(value, config) {
|
|
7925
|
-
if (Array.isArray(value)) {
|
|
7926
|
-
return new LiveList(value.map((v) => deepLiveify(v, config)));
|
|
7927
|
-
} else if (isPlainObject(value)) {
|
|
7928
|
-
const init = {};
|
|
7929
|
-
const locals = {};
|
|
7930
|
-
for (const key in value) {
|
|
7931
|
-
const val = value[key];
|
|
7932
|
-
if (val === void 0) {
|
|
7933
|
-
continue;
|
|
7934
|
-
}
|
|
7935
|
-
const subConfig = isPlainObject(config) ? config[key] : config;
|
|
7936
|
-
if (subConfig === false) {
|
|
7937
|
-
locals[key] = val;
|
|
7938
|
-
} else if (subConfig === "atomic") {
|
|
7939
|
-
init[key] = val;
|
|
7879
|
+
// src/crdts/LiveObject.ts
|
|
7880
|
+
var MAX_LIVE_OBJECT_SIZE = 128 * 1024;
|
|
7881
|
+
var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
7882
|
+
#map;
|
|
7883
|
+
/**
|
|
7884
|
+
* Tracks unacknowledged local changes per property to preserve optimistic
|
|
7885
|
+
* updates. Maps property keys to their pending operation IDs.
|
|
7886
|
+
*
|
|
7887
|
+
* INVARIANT: Only locally-generated opIds are ever stored here. Remote opIds
|
|
7888
|
+
* are only compared against (to detect ACKs), never stored.
|
|
7889
|
+
*
|
|
7890
|
+
* When a local change is made, the opId is stored here. When a remote op
|
|
7891
|
+
* arrives for the same key:
|
|
7892
|
+
* - If no entry exists → apply remote op
|
|
7893
|
+
* - If opId matches → it's an ACK, clear the entry
|
|
7894
|
+
* - If opId differs → ignore remote op to preserve optimistic update
|
|
7895
|
+
*/
|
|
7896
|
+
#unackedOpsByKey;
|
|
7897
|
+
/**
|
|
7898
|
+
* Enable or disable detection of too large LiveObjects.
|
|
7899
|
+
* When enabled, throws an error if LiveObject static data exceeds 128KB, which
|
|
7900
|
+
* is the maximum value the server will be able to accept.
|
|
7901
|
+
* By default, this behavior is disabled to avoid the runtime performance
|
|
7902
|
+
* overhead on every LiveObject.set() or LiveObject.update() call.
|
|
7903
|
+
*
|
|
7904
|
+
* @experimental
|
|
7905
|
+
*/
|
|
7906
|
+
static detectLargeObjects = false;
|
|
7907
|
+
static #buildRootAndParentToChildren(nodes) {
|
|
7908
|
+
const parentToChildren = /* @__PURE__ */ new Map();
|
|
7909
|
+
let root = null;
|
|
7910
|
+
for (const node of nodes) {
|
|
7911
|
+
if (isRootStorageNode(node)) {
|
|
7912
|
+
root = node[1];
|
|
7940
7913
|
} else {
|
|
7941
|
-
|
|
7914
|
+
const crdt = node[1];
|
|
7915
|
+
const children = parentToChildren.get(crdt.parentId);
|
|
7916
|
+
if (children !== void 0) {
|
|
7917
|
+
children.push(node);
|
|
7918
|
+
} else {
|
|
7919
|
+
parentToChildren.set(crdt.parentId, [node]);
|
|
7920
|
+
}
|
|
7942
7921
|
}
|
|
7943
7922
|
}
|
|
7944
|
-
|
|
7945
|
-
|
|
7946
|
-
lo.setLocal(key, locals[key]);
|
|
7923
|
+
if (root === null) {
|
|
7924
|
+
throw new Error("Root can't be null");
|
|
7947
7925
|
}
|
|
7948
|
-
return
|
|
7949
|
-
} else {
|
|
7950
|
-
return value;
|
|
7926
|
+
return [root, parentToChildren];
|
|
7951
7927
|
}
|
|
7952
|
-
|
|
7953
|
-
|
|
7954
|
-
|
|
7955
|
-
|
|
7956
|
-
|
|
7957
|
-
|
|
7958
|
-
|
|
7959
|
-
|
|
7960
|
-
return reconcileLiveList(live, json, config);
|
|
7961
|
-
} else if (isLiveMap(live) && isPlainObject(json)) {
|
|
7962
|
-
return reconcileLiveMap(live, config);
|
|
7963
|
-
} else {
|
|
7964
|
-
return deepLiveify(json, config);
|
|
7928
|
+
/** @private Do not use this API directly */
|
|
7929
|
+
static _fromItems(nodes, pool) {
|
|
7930
|
+
const [root, parentToChildren] = _LiveObject.#buildRootAndParentToChildren(nodes);
|
|
7931
|
+
return _LiveObject._deserialize(
|
|
7932
|
+
["root", root],
|
|
7933
|
+
parentToChildren,
|
|
7934
|
+
pool
|
|
7935
|
+
);
|
|
7965
7936
|
}
|
|
7966
|
-
}
|
|
7967
|
-
|
|
7968
|
-
|
|
7969
|
-
|
|
7970
|
-
|
|
7971
|
-
|
|
7972
|
-
|
|
7973
|
-
|
|
7974
|
-
const newVal = jsonObj[key];
|
|
7975
|
-
if (newVal === void 0) {
|
|
7976
|
-
liveObj.delete(key);
|
|
7977
|
-
continue;
|
|
7978
|
-
}
|
|
7979
|
-
const subConfig = isPlainObject(config) ? config[key] : config;
|
|
7980
|
-
if (subConfig === false) {
|
|
7981
|
-
liveObj.setLocal(key, newVal);
|
|
7982
|
-
} else if (subConfig === "atomic") {
|
|
7983
|
-
const curVal = liveObj.get(key);
|
|
7984
|
-
if (curVal !== newVal) {
|
|
7985
|
-
liveObj.set(key, newVal);
|
|
7986
|
-
}
|
|
7987
|
-
} else {
|
|
7988
|
-
const curVal = liveObj.get(key);
|
|
7989
|
-
if (curVal === void 0) {
|
|
7990
|
-
liveObj.set(key, deepLiveify(newVal, subConfig));
|
|
7991
|
-
} else if (isLiveStructure(curVal)) {
|
|
7992
|
-
const next = reconcile(curVal, newVal, subConfig);
|
|
7993
|
-
if (next !== curVal) {
|
|
7994
|
-
liveObj.set(key, next);
|
|
7995
|
-
}
|
|
7996
|
-
} else if (curVal !== newVal) {
|
|
7997
|
-
liveObj.set(key, deepLiveify(newVal, subConfig));
|
|
7937
|
+
constructor(obj = {}) {
|
|
7938
|
+
super();
|
|
7939
|
+
this.#unackedOpsByKey = /* @__PURE__ */ new Map();
|
|
7940
|
+
const o = compactObject(obj);
|
|
7941
|
+
for (const key of Object.keys(o)) {
|
|
7942
|
+
const value = o[key];
|
|
7943
|
+
if (isLiveNode(value)) {
|
|
7944
|
+
value._setParentLink(this, key);
|
|
7998
7945
|
}
|
|
7999
7946
|
}
|
|
7947
|
+
this.#map = new Map(Object.entries(o));
|
|
8000
7948
|
}
|
|
8001
|
-
|
|
8002
|
-
|
|
8003
|
-
|
|
8004
|
-
|
|
8005
|
-
}
|
|
8006
|
-
|
|
8007
|
-
|
|
8008
|
-
|
|
8009
|
-
|
|
8010
|
-
|
|
8011
|
-
|
|
8012
|
-
|
|
8013
|
-
|
|
8014
|
-
|
|
8015
|
-
|
|
7949
|
+
/** @internal */
|
|
7950
|
+
_toOps(parentId, parentKey) {
|
|
7951
|
+
if (this._id === void 0) {
|
|
7952
|
+
throw new Error("Cannot serialize item is not attached");
|
|
7953
|
+
}
|
|
7954
|
+
const ops = [];
|
|
7955
|
+
const op = {
|
|
7956
|
+
type: OpCode.CREATE_OBJECT,
|
|
7957
|
+
id: this._id,
|
|
7958
|
+
parentId,
|
|
7959
|
+
parentKey,
|
|
7960
|
+
data: {}
|
|
7961
|
+
};
|
|
7962
|
+
ops.push(op);
|
|
7963
|
+
for (const [key, value] of this.#map) {
|
|
7964
|
+
if (isLiveNode(value)) {
|
|
7965
|
+
for (const childOp of value._toOps(this._id, key)) {
|
|
7966
|
+
ops.push(childOp);
|
|
7967
|
+
}
|
|
7968
|
+
} else {
|
|
7969
|
+
op.data[key] = value;
|
|
8016
7970
|
}
|
|
8017
|
-
} else if (curVal !== newVal) {
|
|
8018
|
-
liveList.set(i, deepLiveify(newVal, config));
|
|
8019
7971
|
}
|
|
7972
|
+
return ops;
|
|
8020
7973
|
}
|
|
8021
|
-
|
|
8022
|
-
|
|
8023
|
-
|
|
8024
|
-
|
|
8025
|
-
|
|
7974
|
+
/** @internal */
|
|
7975
|
+
static _deserialize([id, item], parentToChildren, pool) {
|
|
7976
|
+
const liveObj = new _LiveObject(item.data);
|
|
7977
|
+
liveObj._attach(id, pool);
|
|
7978
|
+
return this._deserializeChildren(liveObj, parentToChildren, pool);
|
|
8026
7979
|
}
|
|
8027
|
-
|
|
8028
|
-
|
|
8029
|
-
|
|
8030
|
-
|
|
8031
|
-
|
|
8032
|
-
let nextEnd = next.length - 1;
|
|
8033
|
-
let prevNode = prev[0];
|
|
8034
|
-
let nextNode = next[0];
|
|
8035
|
-
outer: {
|
|
8036
|
-
while (prevNode === nextNode) {
|
|
8037
|
-
++i;
|
|
8038
|
-
if (i > prevEnd || i > nextEnd) {
|
|
8039
|
-
break outer;
|
|
8040
|
-
}
|
|
8041
|
-
prevNode = prev[i];
|
|
8042
|
-
nextNode = next[i];
|
|
7980
|
+
/** @internal */
|
|
7981
|
+
static _deserializeChildren(liveObj, parentToChildren, pool) {
|
|
7982
|
+
const children = parentToChildren.get(nn(liveObj._id));
|
|
7983
|
+
if (children === void 0) {
|
|
7984
|
+
return liveObj;
|
|
8043
7985
|
}
|
|
8044
|
-
|
|
8045
|
-
|
|
8046
|
-
|
|
8047
|
-
|
|
8048
|
-
|
|
8049
|
-
if (i > prevEnd || i > nextEnd) {
|
|
8050
|
-
break outer;
|
|
7986
|
+
for (const node of children) {
|
|
7987
|
+
const child = deserializeToLson(node, parentToChildren, pool);
|
|
7988
|
+
const crdt = node[1];
|
|
7989
|
+
if (isLiveStructure(child)) {
|
|
7990
|
+
child._setParentLink(liveObj, crdt.parentKey);
|
|
8051
7991
|
}
|
|
8052
|
-
|
|
8053
|
-
|
|
7992
|
+
liveObj.#map.set(crdt.parentKey, child);
|
|
7993
|
+
liveObj.invalidate();
|
|
8054
7994
|
}
|
|
7995
|
+
return liveObj;
|
|
8055
7996
|
}
|
|
8056
|
-
|
|
8057
|
-
|
|
8058
|
-
|
|
8059
|
-
|
|
8060
|
-
|
|
7997
|
+
/** @internal */
|
|
7998
|
+
_attach(id, pool) {
|
|
7999
|
+
super._attach(id, pool);
|
|
8000
|
+
for (const [_key, value] of this.#map) {
|
|
8001
|
+
if (isLiveNode(value)) {
|
|
8002
|
+
value._attach(pool.generateId(), pool);
|
|
8061
8003
|
}
|
|
8062
8004
|
}
|
|
8063
|
-
}
|
|
8064
|
-
|
|
8065
|
-
|
|
8066
|
-
|
|
8067
|
-
|
|
8005
|
+
}
|
|
8006
|
+
/** @internal */
|
|
8007
|
+
_attachChild(op, source) {
|
|
8008
|
+
if (this._pool === void 0) {
|
|
8009
|
+
throw new Error("Can't attach child if managed pool is not present");
|
|
8068
8010
|
}
|
|
8069
|
-
|
|
8070
|
-
|
|
8071
|
-
|
|
8072
|
-
|
|
8073
|
-
|
|
8074
|
-
if (isLiveObject(liveListNode) && isPlainObject(prevNode) && isPlainObject(nextNode)) {
|
|
8075
|
-
legacy_patchLiveObject(liveListNode, prevNode, nextNode);
|
|
8076
|
-
} else {
|
|
8077
|
-
liveList.set(i, deepLiveify(nextNode));
|
|
8011
|
+
const { id, opId, parentKey: key } = op;
|
|
8012
|
+
const child = creationOpToLson(op);
|
|
8013
|
+
if (this._pool.getNode(id) !== void 0) {
|
|
8014
|
+
if (this.#unackedOpsByKey.get(key) === opId) {
|
|
8015
|
+
this.#unackedOpsByKey.delete(key);
|
|
8078
8016
|
}
|
|
8079
|
-
|
|
8017
|
+
return { modified: false };
|
|
8080
8018
|
}
|
|
8081
|
-
|
|
8082
|
-
|
|
8083
|
-
|
|
8019
|
+
if (source === 0 /* LOCAL */) {
|
|
8020
|
+
this.#unackedOpsByKey.set(key, nn(opId));
|
|
8021
|
+
} else if (this.#unackedOpsByKey.get(key) === void 0) {
|
|
8022
|
+
} else if (this.#unackedOpsByKey.get(key) === opId) {
|
|
8023
|
+
this.#unackedOpsByKey.delete(key);
|
|
8024
|
+
return { modified: false };
|
|
8025
|
+
} else {
|
|
8026
|
+
return { modified: false };
|
|
8084
8027
|
}
|
|
8085
|
-
|
|
8086
|
-
|
|
8087
|
-
|
|
8088
|
-
|
|
8028
|
+
const thisId = nn(this._id);
|
|
8029
|
+
const previousValue = this.#map.get(key);
|
|
8030
|
+
let reverse;
|
|
8031
|
+
if (isLiveNode(previousValue)) {
|
|
8032
|
+
reverse = previousValue._toOps(thisId, key);
|
|
8033
|
+
previousValue._detach();
|
|
8034
|
+
} else if (previousValue === void 0) {
|
|
8035
|
+
reverse = [{ type: OpCode.DELETE_OBJECT_KEY, id: thisId, key }];
|
|
8036
|
+
} else {
|
|
8037
|
+
reverse = [
|
|
8038
|
+
{
|
|
8039
|
+
type: OpCode.UPDATE_OBJECT,
|
|
8040
|
+
id: thisId,
|
|
8041
|
+
data: { [key]: previousValue }
|
|
8042
|
+
}
|
|
8043
|
+
];
|
|
8089
8044
|
}
|
|
8090
|
-
|
|
8091
|
-
|
|
8092
|
-
|
|
8093
|
-
|
|
8094
|
-
|
|
8095
|
-
if (nonSerializableValue) {
|
|
8096
|
-
error2(
|
|
8097
|
-
`New state path: '${nonSerializableValue.path}' value: '${String(
|
|
8098
|
-
nonSerializableValue.value
|
|
8099
|
-
)}' is not serializable.
|
|
8100
|
-
Only serializable value can be synced with Liveblocks.`
|
|
8101
|
-
);
|
|
8102
|
-
return;
|
|
8045
|
+
this.#map.set(key, child);
|
|
8046
|
+
this.invalidate();
|
|
8047
|
+
if (isLiveStructure(child)) {
|
|
8048
|
+
child._setParentLink(this, key);
|
|
8049
|
+
child._attach(id, this._pool);
|
|
8103
8050
|
}
|
|
8051
|
+
return {
|
|
8052
|
+
reverse,
|
|
8053
|
+
modified: {
|
|
8054
|
+
node: this,
|
|
8055
|
+
type: "LiveObject",
|
|
8056
|
+
updates: { [key]: { type: "update" } }
|
|
8057
|
+
}
|
|
8058
|
+
};
|
|
8104
8059
|
}
|
|
8105
|
-
|
|
8106
|
-
|
|
8107
|
-
|
|
8108
|
-
|
|
8109
|
-
|
|
8110
|
-
|
|
8111
|
-
|
|
8112
|
-
|
|
8113
|
-
|
|
8114
|
-
|
|
8115
|
-
|
|
8116
|
-
|
|
8117
|
-
|
|
8118
|
-
|
|
8119
|
-
|
|
8120
|
-
|
|
8121
|
-
|
|
8122
|
-
|
|
8123
|
-
|
|
8124
|
-
|
|
8125
|
-
|
|
8126
|
-
if (next[key] === void 0) {
|
|
8127
|
-
root.delete(key);
|
|
8060
|
+
/** @internal */
|
|
8061
|
+
_detachChild(child) {
|
|
8062
|
+
if (child) {
|
|
8063
|
+
const id = nn(this._id);
|
|
8064
|
+
const parentKey = nn(child._parentKey);
|
|
8065
|
+
const reverse = child._toOps(id, parentKey);
|
|
8066
|
+
for (const [key, value] of this.#map) {
|
|
8067
|
+
if (value === child) {
|
|
8068
|
+
this.#map.delete(key);
|
|
8069
|
+
this.invalidate();
|
|
8070
|
+
}
|
|
8071
|
+
}
|
|
8072
|
+
child._detach();
|
|
8073
|
+
const storageUpdate = {
|
|
8074
|
+
node: this,
|
|
8075
|
+
type: "LiveObject",
|
|
8076
|
+
updates: {
|
|
8077
|
+
[parentKey]: { type: "delete" }
|
|
8078
|
+
}
|
|
8079
|
+
};
|
|
8080
|
+
return { modified: storageUpdate, reverse };
|
|
8128
8081
|
}
|
|
8082
|
+
return { modified: false };
|
|
8129
8083
|
}
|
|
8130
|
-
|
|
8131
|
-
|
|
8084
|
+
/** @internal */
|
|
8085
|
+
_detach() {
|
|
8086
|
+
super._detach();
|
|
8087
|
+
for (const value of this.#map.values()) {
|
|
8088
|
+
if (isLiveNode(value)) {
|
|
8089
|
+
value._detach();
|
|
8090
|
+
}
|
|
8091
|
+
}
|
|
8132
8092
|
}
|
|
8133
|
-
|
|
8134
|
-
|
|
8135
|
-
|
|
8136
|
-
|
|
8137
|
-
if (
|
|
8138
|
-
|
|
8139
|
-
} else {
|
|
8140
|
-
path.push(node.parent.key);
|
|
8093
|
+
/** @internal */
|
|
8094
|
+
_apply(op, isLocal) {
|
|
8095
|
+
if (op.type === OpCode.UPDATE_OBJECT) {
|
|
8096
|
+
return this.#applyUpdate(op, isLocal);
|
|
8097
|
+
} else if (op.type === OpCode.DELETE_OBJECT_KEY) {
|
|
8098
|
+
return this.#applyDeleteObjectKey(op, isLocal);
|
|
8141
8099
|
}
|
|
8142
|
-
|
|
8100
|
+
return super._apply(op, isLocal);
|
|
8143
8101
|
}
|
|
8144
|
-
|
|
8145
|
-
|
|
8146
|
-
|
|
8147
|
-
|
|
8148
|
-
|
|
8149
|
-
|
|
8150
|
-
);
|
|
8151
|
-
}
|
|
8152
|
-
function legacy_patchImmutableObjectWithUpdate(state, update) {
|
|
8153
|
-
const path = getParentsPath(update.node);
|
|
8154
|
-
return legacy_patchImmutableNode(state, path, update);
|
|
8155
|
-
}
|
|
8156
|
-
function legacy_patchImmutableNode(state, path, update) {
|
|
8157
|
-
const pathItem = path.pop();
|
|
8158
|
-
if (pathItem === void 0) {
|
|
8159
|
-
switch (update.type) {
|
|
8160
|
-
case "LiveObject": {
|
|
8161
|
-
if (!isJsonObject(state)) {
|
|
8162
|
-
throw new Error(
|
|
8163
|
-
"Internal: received update on LiveObject but state was not an object"
|
|
8164
|
-
);
|
|
8165
|
-
}
|
|
8166
|
-
const newState = Object.assign({}, state);
|
|
8167
|
-
for (const key in update.updates) {
|
|
8168
|
-
if (update.updates[key]?.type === "update") {
|
|
8169
|
-
const val = update.node.get(key);
|
|
8170
|
-
if (val !== void 0) {
|
|
8171
|
-
newState[key] = lsonToJson(val);
|
|
8172
|
-
}
|
|
8173
|
-
} else if (update.updates[key]?.type === "delete") {
|
|
8174
|
-
delete newState[key];
|
|
8175
|
-
}
|
|
8176
|
-
}
|
|
8177
|
-
return newState;
|
|
8178
|
-
}
|
|
8179
|
-
case "LiveList": {
|
|
8180
|
-
if (!Array.isArray(state)) {
|
|
8181
|
-
throw new Error(
|
|
8182
|
-
"Internal: received update on LiveList but state was not an array"
|
|
8183
|
-
);
|
|
8184
|
-
}
|
|
8185
|
-
let newState = state.map((x) => x);
|
|
8186
|
-
for (const listUpdate of update.updates) {
|
|
8187
|
-
if (listUpdate.type === "set") {
|
|
8188
|
-
newState = newState.map(
|
|
8189
|
-
(item, index) => index === listUpdate.index ? lsonToJson(listUpdate.item) : item
|
|
8190
|
-
);
|
|
8191
|
-
} else if (listUpdate.type === "insert") {
|
|
8192
|
-
if (listUpdate.index === newState.length) {
|
|
8193
|
-
newState.push(lsonToJson(listUpdate.item));
|
|
8194
|
-
} else {
|
|
8195
|
-
newState = [
|
|
8196
|
-
...newState.slice(0, listUpdate.index),
|
|
8197
|
-
lsonToJson(listUpdate.item),
|
|
8198
|
-
...newState.slice(listUpdate.index)
|
|
8199
|
-
];
|
|
8200
|
-
}
|
|
8201
|
-
} else if (listUpdate.type === "delete") {
|
|
8202
|
-
newState.splice(listUpdate.index, 1);
|
|
8203
|
-
} else if (listUpdate.type === "move") {
|
|
8204
|
-
if (listUpdate.previousIndex > listUpdate.index) {
|
|
8205
|
-
newState = [
|
|
8206
|
-
...newState.slice(0, listUpdate.index),
|
|
8207
|
-
lsonToJson(listUpdate.item),
|
|
8208
|
-
...newState.slice(listUpdate.index, listUpdate.previousIndex),
|
|
8209
|
-
...newState.slice(listUpdate.previousIndex + 1)
|
|
8210
|
-
];
|
|
8211
|
-
} else {
|
|
8212
|
-
newState = [
|
|
8213
|
-
...newState.slice(0, listUpdate.previousIndex),
|
|
8214
|
-
...newState.slice(
|
|
8215
|
-
listUpdate.previousIndex + 1,
|
|
8216
|
-
listUpdate.index + 1
|
|
8217
|
-
),
|
|
8218
|
-
lsonToJson(listUpdate.item),
|
|
8219
|
-
...newState.slice(listUpdate.index + 1)
|
|
8220
|
-
];
|
|
8221
|
-
}
|
|
8222
|
-
}
|
|
8223
|
-
}
|
|
8224
|
-
return newState;
|
|
8225
|
-
}
|
|
8226
|
-
case "LiveMap": {
|
|
8227
|
-
if (!isJsonObject(state)) {
|
|
8228
|
-
throw new Error(
|
|
8229
|
-
"Internal: received update on LiveMap but state was not an object"
|
|
8230
|
-
);
|
|
8231
|
-
}
|
|
8232
|
-
const newState = Object.assign({}, state);
|
|
8233
|
-
for (const key in update.updates) {
|
|
8234
|
-
if (update.updates[key]?.type === "update") {
|
|
8235
|
-
const value = update.node.get(key);
|
|
8236
|
-
if (value !== void 0) {
|
|
8237
|
-
newState[key] = lsonToJson(value);
|
|
8238
|
-
}
|
|
8239
|
-
} else if (update.updates[key]?.type === "delete") {
|
|
8240
|
-
delete newState[key];
|
|
8241
|
-
}
|
|
8242
|
-
}
|
|
8243
|
-
return newState;
|
|
8102
|
+
/** @internal */
|
|
8103
|
+
_serialize() {
|
|
8104
|
+
const data = {};
|
|
8105
|
+
for (const [key, value] of this.#map) {
|
|
8106
|
+
if (!isLiveNode(value)) {
|
|
8107
|
+
data[key] = value;
|
|
8244
8108
|
}
|
|
8245
8109
|
}
|
|
8246
|
-
|
|
8247
|
-
|
|
8248
|
-
|
|
8249
|
-
|
|
8250
|
-
|
|
8251
|
-
|
|
8252
|
-
|
|
8253
|
-
);
|
|
8254
|
-
return newArray;
|
|
8255
|
-
} else if (isJsonObject(state)) {
|
|
8256
|
-
const node = state[pathItem];
|
|
8257
|
-
if (node === void 0) {
|
|
8258
|
-
return state;
|
|
8110
|
+
if (this.parent.type === "HasParent" && this.parent.node._id) {
|
|
8111
|
+
return {
|
|
8112
|
+
type: CrdtType.OBJECT,
|
|
8113
|
+
parentId: this.parent.node._id,
|
|
8114
|
+
parentKey: this.parent.key,
|
|
8115
|
+
data
|
|
8116
|
+
};
|
|
8259
8117
|
} else {
|
|
8260
|
-
const stateAsObj = state;
|
|
8261
8118
|
return {
|
|
8262
|
-
|
|
8263
|
-
|
|
8264
|
-
};
|
|
8265
|
-
}
|
|
8266
|
-
} else {
|
|
8267
|
-
return state;
|
|
8268
|
-
}
|
|
8269
|
-
}
|
|
8270
|
-
|
|
8271
|
-
// src/crdts/LiveObject.ts
|
|
8272
|
-
var MAX_LIVE_OBJECT_SIZE = 128 * 1024;
|
|
8273
|
-
var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
8274
|
-
#synced;
|
|
8275
|
-
#local = /* @__PURE__ */ new Map();
|
|
8276
|
-
/**
|
|
8277
|
-
* Tracks unacknowledged local changes per property to preserve optimistic
|
|
8278
|
-
* updates. Maps property keys to their pending operation IDs.
|
|
8279
|
-
*
|
|
8280
|
-
* INVARIANT: Only locally-generated opIds are ever stored here. Remote opIds
|
|
8281
|
-
* are only compared against (to detect ACKs), never stored.
|
|
8282
|
-
*
|
|
8283
|
-
* When a local change is made, the opId is stored here. When a remote op
|
|
8284
|
-
* arrives for the same key:
|
|
8285
|
-
* - If no entry exists → apply remote op
|
|
8286
|
-
* - If opId matches → it's an ACK, clear the entry
|
|
8287
|
-
* - If opId differs → ignore remote op to preserve optimistic update
|
|
8288
|
-
*/
|
|
8289
|
-
#unackedOpsByKey;
|
|
8290
|
-
/**
|
|
8291
|
-
* Enable or disable detection of too large LiveObjects.
|
|
8292
|
-
* When enabled, throws an error if LiveObject static data exceeds 128KB, which
|
|
8293
|
-
* is the maximum value the server will be able to accept.
|
|
8294
|
-
* By default, this behavior is disabled to avoid the runtime performance
|
|
8295
|
-
* overhead on every LiveObject.set() or LiveObject.update() call.
|
|
8296
|
-
*
|
|
8297
|
-
* @experimental
|
|
8298
|
-
*/
|
|
8299
|
-
static detectLargeObjects = false;
|
|
8300
|
-
static #buildRootAndParentToChildren(nodes) {
|
|
8301
|
-
const parentToChildren = /* @__PURE__ */ new Map();
|
|
8302
|
-
let root = null;
|
|
8303
|
-
for (const node of nodes) {
|
|
8304
|
-
if (isRootStorageNode(node)) {
|
|
8305
|
-
root = node[1];
|
|
8306
|
-
} else {
|
|
8307
|
-
const crdt = node[1];
|
|
8308
|
-
const children = parentToChildren.get(crdt.parentId);
|
|
8309
|
-
if (children !== void 0) {
|
|
8310
|
-
children.push(node);
|
|
8311
|
-
} else {
|
|
8312
|
-
parentToChildren.set(crdt.parentId, [node]);
|
|
8313
|
-
}
|
|
8314
|
-
}
|
|
8315
|
-
}
|
|
8316
|
-
if (root === null) {
|
|
8317
|
-
throw new Error("Root can't be null");
|
|
8318
|
-
}
|
|
8319
|
-
return [root, parentToChildren];
|
|
8320
|
-
}
|
|
8321
|
-
/** @private Do not use this API directly */
|
|
8322
|
-
static _fromItems(nodes, pool) {
|
|
8323
|
-
const [root, parentToChildren] = _LiveObject.#buildRootAndParentToChildren(nodes);
|
|
8324
|
-
return _LiveObject._deserialize(
|
|
8325
|
-
["root", root],
|
|
8326
|
-
parentToChildren,
|
|
8327
|
-
pool
|
|
8328
|
-
);
|
|
8329
|
-
}
|
|
8330
|
-
constructor(obj = {}) {
|
|
8331
|
-
super();
|
|
8332
|
-
this.#unackedOpsByKey = /* @__PURE__ */ new Map();
|
|
8333
|
-
const o = compactObject(obj);
|
|
8334
|
-
for (const key of Object.keys(o)) {
|
|
8335
|
-
const value = o[key];
|
|
8336
|
-
if (isLiveNode(value)) {
|
|
8337
|
-
value._setParentLink(this, key);
|
|
8338
|
-
}
|
|
8339
|
-
}
|
|
8340
|
-
this.#synced = new Map(Object.entries(o));
|
|
8341
|
-
}
|
|
8342
|
-
/** @internal */
|
|
8343
|
-
_toOps(parentId, parentKey) {
|
|
8344
|
-
if (this._id === void 0) {
|
|
8345
|
-
throw new Error("Cannot serialize item is not attached");
|
|
8346
|
-
}
|
|
8347
|
-
const ops = [];
|
|
8348
|
-
const op = {
|
|
8349
|
-
type: OpCode.CREATE_OBJECT,
|
|
8350
|
-
id: this._id,
|
|
8351
|
-
parentId,
|
|
8352
|
-
parentKey,
|
|
8353
|
-
data: {}
|
|
8354
|
-
};
|
|
8355
|
-
ops.push(op);
|
|
8356
|
-
for (const [key, value] of this.#synced) {
|
|
8357
|
-
if (isLiveNode(value)) {
|
|
8358
|
-
for (const childOp of value._toOps(this._id, key)) {
|
|
8359
|
-
ops.push(childOp);
|
|
8360
|
-
}
|
|
8361
|
-
} else {
|
|
8362
|
-
op.data[key] = value;
|
|
8363
|
-
}
|
|
8364
|
-
}
|
|
8365
|
-
return ops;
|
|
8366
|
-
}
|
|
8367
|
-
/** @internal */
|
|
8368
|
-
static _deserialize([id, item], parentToChildren, pool) {
|
|
8369
|
-
const liveObj = new _LiveObject(item.data);
|
|
8370
|
-
liveObj._attach(id, pool);
|
|
8371
|
-
return this._deserializeChildren(liveObj, parentToChildren, pool);
|
|
8372
|
-
}
|
|
8373
|
-
/** @internal */
|
|
8374
|
-
static _deserializeChildren(liveObj, parentToChildren, pool) {
|
|
8375
|
-
const children = parentToChildren.get(nn(liveObj._id));
|
|
8376
|
-
if (children === void 0) {
|
|
8377
|
-
return liveObj;
|
|
8378
|
-
}
|
|
8379
|
-
for (const node of children) {
|
|
8380
|
-
const child = deserializeToLson(node, parentToChildren, pool);
|
|
8381
|
-
const crdt = node[1];
|
|
8382
|
-
if (isLiveStructure(child)) {
|
|
8383
|
-
child._setParentLink(liveObj, crdt.parentKey);
|
|
8384
|
-
}
|
|
8385
|
-
liveObj.#synced.set(crdt.parentKey, child);
|
|
8386
|
-
liveObj.invalidate();
|
|
8387
|
-
}
|
|
8388
|
-
return liveObj;
|
|
8389
|
-
}
|
|
8390
|
-
/** @internal */
|
|
8391
|
-
_attach(id, pool) {
|
|
8392
|
-
super._attach(id, pool);
|
|
8393
|
-
for (const [_key, value] of this.#synced) {
|
|
8394
|
-
if (isLiveNode(value)) {
|
|
8395
|
-
value._attach(pool.generateId(), pool);
|
|
8396
|
-
}
|
|
8397
|
-
}
|
|
8398
|
-
}
|
|
8399
|
-
/** @internal */
|
|
8400
|
-
_attachChild(op, source) {
|
|
8401
|
-
if (this._pool === void 0) {
|
|
8402
|
-
throw new Error("Can't attach child if managed pool is not present");
|
|
8403
|
-
}
|
|
8404
|
-
const { id, opId, parentKey: key } = op;
|
|
8405
|
-
const child = creationOpToLson(op);
|
|
8406
|
-
if (this._pool.getNode(id) !== void 0) {
|
|
8407
|
-
if (this.#unackedOpsByKey.get(key) === opId) {
|
|
8408
|
-
this.#unackedOpsByKey.delete(key);
|
|
8409
|
-
}
|
|
8410
|
-
return { modified: false };
|
|
8411
|
-
}
|
|
8412
|
-
if (source === 0 /* LOCAL */) {
|
|
8413
|
-
this.#unackedOpsByKey.set(key, nn(opId));
|
|
8414
|
-
} else if (this.#unackedOpsByKey.get(key) === void 0) {
|
|
8415
|
-
} else if (this.#unackedOpsByKey.get(key) === opId) {
|
|
8416
|
-
this.#unackedOpsByKey.delete(key);
|
|
8417
|
-
return { modified: false };
|
|
8418
|
-
} else {
|
|
8419
|
-
return { modified: false };
|
|
8420
|
-
}
|
|
8421
|
-
const thisId = nn(this._id);
|
|
8422
|
-
const previousValue = this.#synced.get(key);
|
|
8423
|
-
let reverse;
|
|
8424
|
-
if (isLiveNode(previousValue)) {
|
|
8425
|
-
reverse = previousValue._toOps(thisId, key);
|
|
8426
|
-
previousValue._detach();
|
|
8427
|
-
} else if (previousValue === void 0) {
|
|
8428
|
-
reverse = [{ type: OpCode.DELETE_OBJECT_KEY, id: thisId, key }];
|
|
8429
|
-
} else {
|
|
8430
|
-
reverse = [
|
|
8431
|
-
{
|
|
8432
|
-
type: OpCode.UPDATE_OBJECT,
|
|
8433
|
-
id: thisId,
|
|
8434
|
-
data: { [key]: previousValue }
|
|
8435
|
-
}
|
|
8436
|
-
];
|
|
8437
|
-
}
|
|
8438
|
-
this.#local.delete(key);
|
|
8439
|
-
this.#synced.set(key, child);
|
|
8440
|
-
this.invalidate();
|
|
8441
|
-
if (isLiveStructure(child)) {
|
|
8442
|
-
child._setParentLink(this, key);
|
|
8443
|
-
child._attach(id, this._pool);
|
|
8444
|
-
}
|
|
8445
|
-
return {
|
|
8446
|
-
reverse,
|
|
8447
|
-
modified: {
|
|
8448
|
-
node: this,
|
|
8449
|
-
type: "LiveObject",
|
|
8450
|
-
updates: { [key]: { type: "update" } }
|
|
8451
|
-
}
|
|
8452
|
-
};
|
|
8453
|
-
}
|
|
8454
|
-
/** @internal */
|
|
8455
|
-
_detachChild(child) {
|
|
8456
|
-
if (child) {
|
|
8457
|
-
const id = nn(this._id);
|
|
8458
|
-
const parentKey = nn(child._parentKey);
|
|
8459
|
-
const reverse = child._toOps(id, parentKey);
|
|
8460
|
-
for (const [key, value] of this.#synced) {
|
|
8461
|
-
if (value === child) {
|
|
8462
|
-
this.#synced.delete(key);
|
|
8463
|
-
this.invalidate();
|
|
8464
|
-
}
|
|
8465
|
-
}
|
|
8466
|
-
child._detach();
|
|
8467
|
-
const storageUpdate = {
|
|
8468
|
-
node: this,
|
|
8469
|
-
type: "LiveObject",
|
|
8470
|
-
updates: {
|
|
8471
|
-
[parentKey]: { type: "delete" }
|
|
8472
|
-
}
|
|
8473
|
-
};
|
|
8474
|
-
return { modified: storageUpdate, reverse };
|
|
8475
|
-
}
|
|
8476
|
-
return { modified: false };
|
|
8477
|
-
}
|
|
8478
|
-
/** @internal */
|
|
8479
|
-
_detach() {
|
|
8480
|
-
super._detach();
|
|
8481
|
-
for (const value of this.#synced.values()) {
|
|
8482
|
-
if (isLiveNode(value)) {
|
|
8483
|
-
value._detach();
|
|
8484
|
-
}
|
|
8485
|
-
}
|
|
8486
|
-
}
|
|
8487
|
-
/** @internal */
|
|
8488
|
-
_apply(op, isLocal) {
|
|
8489
|
-
if (op.type === OpCode.UPDATE_OBJECT) {
|
|
8490
|
-
return this.#applyUpdate(op, isLocal);
|
|
8491
|
-
} else if (op.type === OpCode.DELETE_OBJECT_KEY) {
|
|
8492
|
-
return this.#applyDeleteObjectKey(op, isLocal);
|
|
8493
|
-
}
|
|
8494
|
-
return super._apply(op, isLocal);
|
|
8495
|
-
}
|
|
8496
|
-
/** @internal */
|
|
8497
|
-
_serialize() {
|
|
8498
|
-
const data = {};
|
|
8499
|
-
for (const [key, value] of this.#synced) {
|
|
8500
|
-
if (!isLiveNode(value)) {
|
|
8501
|
-
data[key] = value;
|
|
8502
|
-
}
|
|
8503
|
-
}
|
|
8504
|
-
if (this.parent.type === "HasParent" && this.parent.node._id) {
|
|
8505
|
-
return {
|
|
8506
|
-
type: CrdtType.OBJECT,
|
|
8507
|
-
parentId: this.parent.node._id,
|
|
8508
|
-
parentKey: this.parent.key,
|
|
8509
|
-
data
|
|
8510
|
-
};
|
|
8511
|
-
} else {
|
|
8512
|
-
return {
|
|
8513
|
-
type: CrdtType.OBJECT,
|
|
8514
|
-
data
|
|
8119
|
+
type: CrdtType.OBJECT,
|
|
8120
|
+
data
|
|
8515
8121
|
};
|
|
8516
8122
|
}
|
|
8517
8123
|
}
|
|
@@ -8525,7 +8131,7 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8525
8131
|
data: {}
|
|
8526
8132
|
};
|
|
8527
8133
|
for (const key in op.data) {
|
|
8528
|
-
const oldValue = this.#
|
|
8134
|
+
const oldValue = this.#map.get(key);
|
|
8529
8135
|
if (isLiveNode(oldValue)) {
|
|
8530
8136
|
for (const childOp of oldValue._toOps(id, key)) {
|
|
8531
8137
|
reverse.push(childOp);
|
|
@@ -8553,14 +8159,13 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8553
8159
|
} else {
|
|
8554
8160
|
continue;
|
|
8555
8161
|
}
|
|
8556
|
-
const oldValue = this.#
|
|
8162
|
+
const oldValue = this.#map.get(key);
|
|
8557
8163
|
if (isLiveNode(oldValue)) {
|
|
8558
8164
|
oldValue._detach();
|
|
8559
8165
|
}
|
|
8560
8166
|
isModified = true;
|
|
8561
8167
|
updateDelta[key] = { type: "update" };
|
|
8562
|
-
this.#
|
|
8563
|
-
this.#synced.set(key, value);
|
|
8168
|
+
this.#map.set(key, value);
|
|
8564
8169
|
this.invalidate();
|
|
8565
8170
|
}
|
|
8566
8171
|
if (Object.keys(reverseUpdate.data).length !== 0) {
|
|
@@ -8577,7 +8182,7 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8577
8182
|
}
|
|
8578
8183
|
#applyDeleteObjectKey(op, isLocal) {
|
|
8579
8184
|
const key = op.key;
|
|
8580
|
-
const oldValue = this.#
|
|
8185
|
+
const oldValue = this.#map.get(key);
|
|
8581
8186
|
if (oldValue === void 0) {
|
|
8582
8187
|
return { modified: false };
|
|
8583
8188
|
}
|
|
@@ -8598,8 +8203,7 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8598
8203
|
}
|
|
8599
8204
|
];
|
|
8600
8205
|
}
|
|
8601
|
-
this.#
|
|
8602
|
-
this.#synced.delete(key);
|
|
8206
|
+
this.#map.delete(key);
|
|
8603
8207
|
this.invalidate();
|
|
8604
8208
|
return {
|
|
8605
8209
|
modified: {
|
|
@@ -8612,23 +8216,11 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8612
8216
|
reverse
|
|
8613
8217
|
};
|
|
8614
8218
|
}
|
|
8615
|
-
/** @private */
|
|
8616
|
-
keys() {
|
|
8617
|
-
const result = new Set(this.#synced.keys());
|
|
8618
|
-
for (const key of this.#local.keys()) {
|
|
8619
|
-
result.add(key);
|
|
8620
|
-
}
|
|
8621
|
-
return result;
|
|
8622
|
-
}
|
|
8623
8219
|
/**
|
|
8624
8220
|
* Transform the LiveObject into a javascript object
|
|
8625
8221
|
*/
|
|
8626
8222
|
toObject() {
|
|
8627
|
-
|
|
8628
|
-
for (const [key, value] of this.#local) {
|
|
8629
|
-
result[key] = value;
|
|
8630
|
-
}
|
|
8631
|
-
return result;
|
|
8223
|
+
return Object.fromEntries(this.#map);
|
|
8632
8224
|
}
|
|
8633
8225
|
/**
|
|
8634
8226
|
* Adds or updates a property with a specified key and a value.
|
|
@@ -8636,109 +8228,49 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8636
8228
|
* @param value The value of the property to add
|
|
8637
8229
|
*/
|
|
8638
8230
|
set(key, value) {
|
|
8639
|
-
this.update({ [key]: value });
|
|
8640
|
-
}
|
|
8641
|
-
/**
|
|
8642
|
-
* @experimental
|
|
8643
|
-
*
|
|
8644
|
-
* Sets a local-only property that is not synchronized over the wire.
|
|
8645
|
-
* The value will be visible via get(), toObject(), and toImmutable() on
|
|
8646
|
-
* this client only. Other clients and the server will see `undefined`
|
|
8647
|
-
* for this key.
|
|
8648
|
-
*
|
|
8649
|
-
* Caveat: this method will not add changes to the undo/redo stack.
|
|
8650
|
-
*/
|
|
8651
|
-
setLocal(key, value) {
|
|
8652
8231
|
this._pool?.assertStorageIsWritable();
|
|
8653
|
-
|
|
8654
|
-
this.#local.set(key, value);
|
|
8655
|
-
this.invalidate();
|
|
8656
|
-
if (this._pool !== void 0 && this._id !== void 0) {
|
|
8657
|
-
const ops = deleteResult?.[0] ?? [];
|
|
8658
|
-
const reverse = deleteResult?.[1] ?? [];
|
|
8659
|
-
const storageUpdates = deleteResult?.[2] ?? /* @__PURE__ */ new Map();
|
|
8660
|
-
const existing = storageUpdates.get(this._id);
|
|
8661
|
-
storageUpdates.set(this._id, {
|
|
8662
|
-
node: this,
|
|
8663
|
-
type: "LiveObject",
|
|
8664
|
-
updates: {
|
|
8665
|
-
...existing?.updates,
|
|
8666
|
-
[key]: { type: "update" }
|
|
8667
|
-
}
|
|
8668
|
-
});
|
|
8669
|
-
this._pool.dispatch(ops, reverse, storageUpdates);
|
|
8670
|
-
}
|
|
8232
|
+
this.update({ [key]: value });
|
|
8671
8233
|
}
|
|
8672
8234
|
/**
|
|
8673
8235
|
* Returns a specified property from the LiveObject.
|
|
8674
8236
|
* @param key The key of the property to get
|
|
8675
8237
|
*/
|
|
8676
8238
|
get(key) {
|
|
8677
|
-
return this.#
|
|
8239
|
+
return this.#map.get(key);
|
|
8678
8240
|
}
|
|
8679
8241
|
/**
|
|
8680
|
-
*
|
|
8681
|
-
*
|
|
8682
|
-
* #synced or pool/id are unavailable. Does NOT dispatch.
|
|
8242
|
+
* Deletes a key from the LiveObject
|
|
8243
|
+
* @param key The key of the property to delete
|
|
8683
8244
|
*/
|
|
8684
|
-
|
|
8245
|
+
delete(key) {
|
|
8685
8246
|
this._pool?.assertStorageIsWritable();
|
|
8686
|
-
const
|
|
8687
|
-
|
|
8688
|
-
const oldValue2 = this.#local.get(k);
|
|
8689
|
-
this.#local.delete(k);
|
|
8690
|
-
this.invalidate();
|
|
8691
|
-
if (this._pool !== void 0 && this._id !== void 0) {
|
|
8692
|
-
const storageUpdates2 = /* @__PURE__ */ new Map();
|
|
8693
|
-
storageUpdates2.set(this._id, {
|
|
8694
|
-
node: this,
|
|
8695
|
-
type: "LiveObject",
|
|
8696
|
-
updates: {
|
|
8697
|
-
[k]: {
|
|
8698
|
-
type: "delete",
|
|
8699
|
-
deletedItem: oldValue2
|
|
8700
|
-
}
|
|
8701
|
-
}
|
|
8702
|
-
});
|
|
8703
|
-
return [[], [], storageUpdates2];
|
|
8704
|
-
}
|
|
8705
|
-
return null;
|
|
8706
|
-
}
|
|
8707
|
-
this.#local.delete(k);
|
|
8708
|
-
const oldValue = this.#synced.get(k);
|
|
8247
|
+
const keyAsString = key;
|
|
8248
|
+
const oldValue = this.#map.get(keyAsString);
|
|
8709
8249
|
if (oldValue === void 0) {
|
|
8710
|
-
return
|
|
8250
|
+
return;
|
|
8711
8251
|
}
|
|
8712
8252
|
if (this._pool === void 0 || this._id === void 0) {
|
|
8713
8253
|
if (isLiveNode(oldValue)) {
|
|
8714
8254
|
oldValue._detach();
|
|
8715
8255
|
}
|
|
8716
|
-
this.#
|
|
8256
|
+
this.#map.delete(keyAsString);
|
|
8717
8257
|
this.invalidate();
|
|
8718
|
-
return
|
|
8258
|
+
return;
|
|
8719
8259
|
}
|
|
8720
|
-
const ops = [
|
|
8721
|
-
{
|
|
8722
|
-
type: OpCode.DELETE_OBJECT_KEY,
|
|
8723
|
-
key: k,
|
|
8724
|
-
id: this._id,
|
|
8725
|
-
opId: this._pool.generateOpId()
|
|
8726
|
-
}
|
|
8727
|
-
];
|
|
8728
8260
|
let reverse;
|
|
8729
8261
|
if (isLiveNode(oldValue)) {
|
|
8730
8262
|
oldValue._detach();
|
|
8731
|
-
reverse = oldValue._toOps(this._id,
|
|
8263
|
+
reverse = oldValue._toOps(this._id, keyAsString);
|
|
8732
8264
|
} else {
|
|
8733
8265
|
reverse = [
|
|
8734
8266
|
{
|
|
8735
8267
|
type: OpCode.UPDATE_OBJECT,
|
|
8736
|
-
data: { [
|
|
8268
|
+
data: { [keyAsString]: oldValue },
|
|
8737
8269
|
id: this._id
|
|
8738
8270
|
}
|
|
8739
8271
|
];
|
|
8740
8272
|
}
|
|
8741
|
-
this.#
|
|
8273
|
+
this.#map.delete(keyAsString);
|
|
8742
8274
|
this.invalidate();
|
|
8743
8275
|
const storageUpdates = /* @__PURE__ */ new Map();
|
|
8744
8276
|
storageUpdates.set(this._id, {
|
|
@@ -8748,18 +8280,18 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8748
8280
|
[key]: { type: "delete", deletedItem: oldValue }
|
|
8749
8281
|
}
|
|
8750
8282
|
});
|
|
8751
|
-
|
|
8752
|
-
|
|
8753
|
-
|
|
8754
|
-
|
|
8755
|
-
|
|
8756
|
-
|
|
8757
|
-
|
|
8758
|
-
|
|
8759
|
-
|
|
8760
|
-
|
|
8761
|
-
|
|
8762
|
-
|
|
8283
|
+
this._pool.dispatch(
|
|
8284
|
+
[
|
|
8285
|
+
{
|
|
8286
|
+
type: OpCode.DELETE_OBJECT_KEY,
|
|
8287
|
+
key: keyAsString,
|
|
8288
|
+
id: this._id,
|
|
8289
|
+
opId: this._pool.generateOpId()
|
|
8290
|
+
}
|
|
8291
|
+
],
|
|
8292
|
+
reverse,
|
|
8293
|
+
storageUpdates
|
|
8294
|
+
);
|
|
8763
8295
|
}
|
|
8764
8296
|
/**
|
|
8765
8297
|
* Adds or updates multiple properties at once with an object.
|
|
@@ -8769,7 +8301,7 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8769
8301
|
this._pool?.assertStorageIsWritable();
|
|
8770
8302
|
if (_LiveObject.detectLargeObjects) {
|
|
8771
8303
|
const data = {};
|
|
8772
|
-
for (const [key, value] of this.#
|
|
8304
|
+
for (const [key, value] of this.#map) {
|
|
8773
8305
|
if (!isLiveNode(value)) {
|
|
8774
8306
|
data[key] = value;
|
|
8775
8307
|
}
|
|
@@ -8798,15 +8330,14 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8798
8330
|
if (newValue === void 0) {
|
|
8799
8331
|
continue;
|
|
8800
8332
|
}
|
|
8801
|
-
const oldValue = this.#
|
|
8333
|
+
const oldValue = this.#map.get(key);
|
|
8802
8334
|
if (isLiveNode(oldValue)) {
|
|
8803
8335
|
oldValue._detach();
|
|
8804
8336
|
}
|
|
8805
8337
|
if (isLiveNode(newValue)) {
|
|
8806
8338
|
newValue._setParentLink(this, key);
|
|
8807
8339
|
}
|
|
8808
|
-
this.#
|
|
8809
|
-
this.#synced.set(key, newValue);
|
|
8340
|
+
this.#map.set(key, newValue);
|
|
8810
8341
|
this.invalidate();
|
|
8811
8342
|
}
|
|
8812
8343
|
return;
|
|
@@ -8826,7 +8357,7 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8826
8357
|
if (newValue === void 0) {
|
|
8827
8358
|
continue;
|
|
8828
8359
|
}
|
|
8829
|
-
const oldValue = this.#
|
|
8360
|
+
const oldValue = this.#map.get(key);
|
|
8830
8361
|
if (isLiveNode(oldValue)) {
|
|
8831
8362
|
for (const childOp of oldValue._toOps(this._id, key)) {
|
|
8832
8363
|
reverseOps.push(childOp);
|
|
@@ -8858,8 +8389,7 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8858
8389
|
updatedProps[key] = newValue;
|
|
8859
8390
|
this.#unackedOpsByKey.set(key, opId);
|
|
8860
8391
|
}
|
|
8861
|
-
this.#
|
|
8862
|
-
this.#synced.set(key, newValue);
|
|
8392
|
+
this.#map.set(key, newValue);
|
|
8863
8393
|
this.invalidate();
|
|
8864
8394
|
updateDelta[key] = { type: "update" };
|
|
8865
8395
|
}
|
|
@@ -8882,19 +8412,6 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8882
8412
|
});
|
|
8883
8413
|
this._pool.dispatch(ops, reverseOps, storageUpdates);
|
|
8884
8414
|
}
|
|
8885
|
-
/**
|
|
8886
|
-
* Reconciles a LiveObject tree to match the given JSON object and sync
|
|
8887
|
-
* config. Only mutates keys that actually changed. Recursively reconciles
|
|
8888
|
-
* the entire Live tree below it.
|
|
8889
|
-
*/
|
|
8890
|
-
reconcile(jsonObj, config) {
|
|
8891
|
-
if (this.immutableIs(jsonObj)) return;
|
|
8892
|
-
if (!isPlainObject(jsonObj))
|
|
8893
|
-
throw new Error(
|
|
8894
|
-
"Reconciling the document root expects a plain object value"
|
|
8895
|
-
);
|
|
8896
|
-
reconcileLiveObject(this, jsonObj, config);
|
|
8897
|
-
}
|
|
8898
8415
|
toImmutable() {
|
|
8899
8416
|
return super.toImmutable();
|
|
8900
8417
|
}
|
|
@@ -8909,7 +8426,7 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8909
8426
|
type: "LiveObject",
|
|
8910
8427
|
id: nodeId,
|
|
8911
8428
|
key,
|
|
8912
|
-
payload: Array.from(this.#
|
|
8429
|
+
payload: Array.from(this.#map.entries()).map(
|
|
8913
8430
|
([key2, value]) => isLiveNode(value) ? value.toTreeNode(key2) : { type: "Json", id: `${nodeId}:${key2}`, key: key2, payload: value }
|
|
8914
8431
|
)
|
|
8915
8432
|
};
|
|
@@ -8917,27 +8434,20 @@ var LiveObject = class _LiveObject extends AbstractCrdt {
|
|
|
8917
8434
|
/** @internal */
|
|
8918
8435
|
_toImmutable() {
|
|
8919
8436
|
const result = {};
|
|
8920
|
-
for (const [key, val] of this.#
|
|
8437
|
+
for (const [key, val] of this.#map) {
|
|
8921
8438
|
result[key] = isLiveStructure(val) ? val.toImmutable() : val;
|
|
8922
8439
|
}
|
|
8923
|
-
for (const [key, val] of this.#local) {
|
|
8924
|
-
result[key] = val;
|
|
8925
|
-
}
|
|
8926
8440
|
return process.env.NODE_ENV === "production" ? result : Object.freeze(result);
|
|
8927
8441
|
}
|
|
8928
8442
|
clone() {
|
|
8929
|
-
|
|
8443
|
+
return new _LiveObject(
|
|
8930
8444
|
Object.fromEntries(
|
|
8931
|
-
Array.from(this.#
|
|
8445
|
+
Array.from(this.#map).map(([key, value]) => [
|
|
8932
8446
|
key,
|
|
8933
8447
|
isLiveStructure(value) ? value.clone() : deepClone(value)
|
|
8934
8448
|
])
|
|
8935
8449
|
)
|
|
8936
8450
|
);
|
|
8937
|
-
for (const [key, value] of this.#local) {
|
|
8938
|
-
cloned.#local.set(key, deepClone(value));
|
|
8939
|
-
}
|
|
8940
|
-
return cloned;
|
|
8941
8451
|
}
|
|
8942
8452
|
};
|
|
8943
8453
|
|
|
@@ -9237,6 +8747,17 @@ var Deque = class {
|
|
|
9237
8747
|
}
|
|
9238
8748
|
};
|
|
9239
8749
|
|
|
8750
|
+
// src/lib/Json.ts
|
|
8751
|
+
function isJsonScalar(data) {
|
|
8752
|
+
return data === null || typeof data === "string" || typeof data === "number" || typeof data === "boolean";
|
|
8753
|
+
}
|
|
8754
|
+
function isJsonArray(data) {
|
|
8755
|
+
return Array.isArray(data);
|
|
8756
|
+
}
|
|
8757
|
+
function isJsonObject(data) {
|
|
8758
|
+
return !isJsonScalar(data) && !isJsonArray(data);
|
|
8759
|
+
}
|
|
8760
|
+
|
|
9240
8761
|
// src/lib/stopwatch.ts
|
|
9241
8762
|
function makeStopWatch() {
|
|
9242
8763
|
let startTime = 0;
|
|
@@ -9270,7 +8791,16 @@ var ClientMsgCode = Object.freeze({
|
|
|
9270
8791
|
UPDATE_STORAGE: 201,
|
|
9271
8792
|
// For Yjs support
|
|
9272
8793
|
FETCH_YDOC: 300,
|
|
9273
|
-
UPDATE_YDOC: 301
|
|
8794
|
+
UPDATE_YDOC: 301,
|
|
8795
|
+
// For Feeds
|
|
8796
|
+
FETCH_FEEDS: 510,
|
|
8797
|
+
FETCH_FEED_MESSAGES: 511,
|
|
8798
|
+
ADD_FEED: 512,
|
|
8799
|
+
UPDATE_FEED: 513,
|
|
8800
|
+
DELETE_FEED: 514,
|
|
8801
|
+
ADD_FEED_MESSAGE: 515,
|
|
8802
|
+
UPDATE_FEED_MESSAGE: 516,
|
|
8803
|
+
DELETE_FEED_MESSAGE: 517
|
|
9274
8804
|
});
|
|
9275
8805
|
|
|
9276
8806
|
// src/refs/ManagedOthers.ts
|
|
@@ -9506,12 +9036,15 @@ function defaultMessageFromContext(context) {
|
|
|
9506
9036
|
return "Could not update notification settings";
|
|
9507
9037
|
case "LARGE_MESSAGE_ERROR":
|
|
9508
9038
|
return "Could not send large message";
|
|
9039
|
+
case "FEED_REQUEST_ERROR":
|
|
9040
|
+
return context.reason ?? "Feed request failed";
|
|
9509
9041
|
default:
|
|
9510
9042
|
return assertNever(context, "Unhandled case");
|
|
9511
9043
|
}
|
|
9512
9044
|
}
|
|
9513
9045
|
|
|
9514
9046
|
// src/room.ts
|
|
9047
|
+
var FEEDS_TIMEOUT = 5e3;
|
|
9515
9048
|
function makeIdFactory(connectionId) {
|
|
9516
9049
|
let count = 0;
|
|
9517
9050
|
return () => `${connectionId}:${count++}`;
|
|
@@ -9722,13 +9255,9 @@ function createRoom(options, config) {
|
|
|
9722
9255
|
}
|
|
9723
9256
|
context.activeBatch.reverseOps.pushLeft(reverse);
|
|
9724
9257
|
} else {
|
|
9725
|
-
|
|
9726
|
-
|
|
9727
|
-
|
|
9728
|
-
if (ops.length > 0) {
|
|
9729
|
-
context.redoStack.length = 0;
|
|
9730
|
-
dispatchOps(ops);
|
|
9731
|
-
}
|
|
9258
|
+
addToUndoStack(reverse);
|
|
9259
|
+
context.redoStack.length = 0;
|
|
9260
|
+
dispatchOps(ops);
|
|
9732
9261
|
notify({ storageUpdates });
|
|
9733
9262
|
}
|
|
9734
9263
|
}
|
|
@@ -9750,6 +9279,7 @@ function createRoom(options, config) {
|
|
|
9750
9279
|
storageStatus: makeEventSource(),
|
|
9751
9280
|
ydoc: makeEventSource(),
|
|
9752
9281
|
comments: makeEventSource(),
|
|
9282
|
+
feeds: makeEventSource(),
|
|
9753
9283
|
roomWillDestroy: makeEventSource()
|
|
9754
9284
|
};
|
|
9755
9285
|
async function createTextMention(mentionId, mention) {
|
|
@@ -9837,20 +9367,19 @@ function createRoom(options, config) {
|
|
|
9837
9367
|
);
|
|
9838
9368
|
}
|
|
9839
9369
|
const canWrite = self.get()?.canWrite ?? true;
|
|
9840
|
-
const
|
|
9841
|
-
|
|
9842
|
-
|
|
9843
|
-
if (
|
|
9844
|
-
|
|
9845
|
-
|
|
9846
|
-
|
|
9847
|
-
|
|
9848
|
-
|
|
9849
|
-
);
|
|
9850
|
-
}
|
|
9370
|
+
const stackSizeBefore = context.undoStack.length;
|
|
9371
|
+
for (const key in context.initialStorage) {
|
|
9372
|
+
if (context.root.get(key) === void 0) {
|
|
9373
|
+
if (canWrite) {
|
|
9374
|
+
context.root.set(key, cloneLson(context.initialStorage[key]));
|
|
9375
|
+
} else {
|
|
9376
|
+
warn(
|
|
9377
|
+
`Attempted to populate missing storage key '${key}', but current user has no write access`
|
|
9378
|
+
);
|
|
9851
9379
|
}
|
|
9852
9380
|
}
|
|
9853
|
-
}
|
|
9381
|
+
}
|
|
9382
|
+
context.undoStack.length = stackSizeBefore;
|
|
9854
9383
|
}
|
|
9855
9384
|
function _addToRealUndoStack(frames) {
|
|
9856
9385
|
if (context.undoStack.length >= 50) {
|
|
@@ -10171,6 +9700,9 @@ function createRoom(options, config) {
|
|
|
10171
9700
|
notify(result.updates);
|
|
10172
9701
|
sendMessages(messages);
|
|
10173
9702
|
}
|
|
9703
|
+
function isFeedRequestFailedMsg(msg) {
|
|
9704
|
+
return msg.type === ServerMsgCode.FEED_REQUEST_FAILED;
|
|
9705
|
+
}
|
|
10174
9706
|
function handleServerMessage(event) {
|
|
10175
9707
|
if (typeof event.data !== "string") {
|
|
10176
9708
|
return;
|
|
@@ -10284,6 +9816,94 @@ function createRoom(options, config) {
|
|
|
10284
9816
|
eventHub.comments.notify(message);
|
|
10285
9817
|
break;
|
|
10286
9818
|
}
|
|
9819
|
+
case ServerMsgCode.FEEDS_LIST: {
|
|
9820
|
+
const feedsListMsg = message;
|
|
9821
|
+
const pending = pendingFeedsRequests.get(feedsListMsg.requestId);
|
|
9822
|
+
if (pending) {
|
|
9823
|
+
pending.resolve({
|
|
9824
|
+
feeds: feedsListMsg.feeds,
|
|
9825
|
+
nextCursor: feedsListMsg.nextCursor
|
|
9826
|
+
});
|
|
9827
|
+
pendingFeedsRequests.delete(feedsListMsg.requestId);
|
|
9828
|
+
}
|
|
9829
|
+
eventHub.feeds.notify(feedsListMsg);
|
|
9830
|
+
break;
|
|
9831
|
+
}
|
|
9832
|
+
case ServerMsgCode.FEEDS_ADDED: {
|
|
9833
|
+
const feedsAddedMsg = message;
|
|
9834
|
+
eventHub.feeds.notify(feedsAddedMsg);
|
|
9835
|
+
tryResolvePendingFeedMutationsFromFeedsEvent(feedsAddedMsg);
|
|
9836
|
+
break;
|
|
9837
|
+
}
|
|
9838
|
+
case ServerMsgCode.FEEDS_UPDATED: {
|
|
9839
|
+
const feedsUpdatedMsg = message;
|
|
9840
|
+
eventHub.feeds.notify(feedsUpdatedMsg);
|
|
9841
|
+
tryResolvePendingFeedMutationsFromFeedsEvent(feedsUpdatedMsg);
|
|
9842
|
+
break;
|
|
9843
|
+
}
|
|
9844
|
+
case ServerMsgCode.FEED_DELETED: {
|
|
9845
|
+
eventHub.feeds.notify(message);
|
|
9846
|
+
tryResolvePendingFeedMutationsFromFeedsEvent(message);
|
|
9847
|
+
break;
|
|
9848
|
+
}
|
|
9849
|
+
case ServerMsgCode.FEED_MESSAGES_LIST: {
|
|
9850
|
+
const feedMsgsListMsg = message;
|
|
9851
|
+
const pending = pendingFeedMessagesRequests.get(
|
|
9852
|
+
feedMsgsListMsg.requestId
|
|
9853
|
+
);
|
|
9854
|
+
if (pending) {
|
|
9855
|
+
pending.resolve({
|
|
9856
|
+
messages: feedMsgsListMsg.messages,
|
|
9857
|
+
nextCursor: feedMsgsListMsg.nextCursor
|
|
9858
|
+
});
|
|
9859
|
+
pendingFeedMessagesRequests.delete(feedMsgsListMsg.requestId);
|
|
9860
|
+
}
|
|
9861
|
+
eventHub.feeds.notify(feedMsgsListMsg);
|
|
9862
|
+
break;
|
|
9863
|
+
}
|
|
9864
|
+
case ServerMsgCode.FEED_MESSAGES_ADDED: {
|
|
9865
|
+
const feedMsgsAddedMsg = message;
|
|
9866
|
+
eventHub.feeds.notify(feedMsgsAddedMsg);
|
|
9867
|
+
tryResolvePendingFeedMutationsFromFeedsEvent(feedMsgsAddedMsg);
|
|
9868
|
+
break;
|
|
9869
|
+
}
|
|
9870
|
+
case ServerMsgCode.FEED_MESSAGES_UPDATED: {
|
|
9871
|
+
const feedMsgsUpdatedMsg = message;
|
|
9872
|
+
eventHub.feeds.notify(feedMsgsUpdatedMsg);
|
|
9873
|
+
tryResolvePendingFeedMutationsFromFeedsEvent(feedMsgsUpdatedMsg);
|
|
9874
|
+
break;
|
|
9875
|
+
}
|
|
9876
|
+
case ServerMsgCode.FEED_MESSAGES_DELETED: {
|
|
9877
|
+
eventHub.feeds.notify(message);
|
|
9878
|
+
tryResolvePendingFeedMutationsFromFeedsEvent(message);
|
|
9879
|
+
break;
|
|
9880
|
+
}
|
|
9881
|
+
case ServerMsgCode.FEED_REQUEST_FAILED: {
|
|
9882
|
+
if (!isFeedRequestFailedMsg(message)) {
|
|
9883
|
+
break;
|
|
9884
|
+
}
|
|
9885
|
+
const { requestId, code, reason } = message;
|
|
9886
|
+
const err = new LiveblocksError(reason ?? "Feed request failed", {
|
|
9887
|
+
type: "FEED_REQUEST_ERROR",
|
|
9888
|
+
roomId,
|
|
9889
|
+
requestId,
|
|
9890
|
+
code,
|
|
9891
|
+
reason
|
|
9892
|
+
});
|
|
9893
|
+
if (pendingFeedMutations.has(requestId)) {
|
|
9894
|
+
settleFeedMutation(requestId, "error", err);
|
|
9895
|
+
} else if (pendingFeedsRequests.has(requestId)) {
|
|
9896
|
+
const pending = pendingFeedsRequests.get(requestId);
|
|
9897
|
+
pendingFeedsRequests.delete(requestId);
|
|
9898
|
+
pending?.reject(err);
|
|
9899
|
+
} else if (pendingFeedMessagesRequests.has(requestId)) {
|
|
9900
|
+
const pending = pendingFeedMessagesRequests.get(requestId);
|
|
9901
|
+
pendingFeedMessagesRequests.delete(requestId);
|
|
9902
|
+
pending?.reject(err);
|
|
9903
|
+
}
|
|
9904
|
+
eventHub.feeds.notify(message);
|
|
9905
|
+
break;
|
|
9906
|
+
}
|
|
10287
9907
|
case ServerMsgCode.STORAGE_STATE_V7:
|
|
10288
9908
|
// No longer used in V8
|
|
10289
9909
|
default:
|
|
@@ -10387,6 +10007,141 @@ function createRoom(options, config) {
|
|
|
10387
10007
|
}
|
|
10388
10008
|
let _getStorage$ = null;
|
|
10389
10009
|
let _resolveStoragePromise = null;
|
|
10010
|
+
const pendingFeedsRequests = /* @__PURE__ */ new Map();
|
|
10011
|
+
const pendingFeedMessagesRequests = /* @__PURE__ */ new Map();
|
|
10012
|
+
const pendingFeedMutations = /* @__PURE__ */ new Map();
|
|
10013
|
+
const pendingAddMessageFifoByFeed = /* @__PURE__ */ new Map();
|
|
10014
|
+
function settleFeedMutation(requestId, outcome, error3) {
|
|
10015
|
+
const pending = pendingFeedMutations.get(requestId);
|
|
10016
|
+
if (pending === void 0) {
|
|
10017
|
+
return;
|
|
10018
|
+
}
|
|
10019
|
+
clearTimeout(pending.timeoutId);
|
|
10020
|
+
pendingFeedMutations.delete(requestId);
|
|
10021
|
+
if (pending.kind === "add-message" && !pending.expectedClientMessageId) {
|
|
10022
|
+
const q = pendingAddMessageFifoByFeed.get(pending.feedId);
|
|
10023
|
+
if (q !== void 0) {
|
|
10024
|
+
const idx = q.indexOf(requestId);
|
|
10025
|
+
if (idx >= 0) {
|
|
10026
|
+
q.splice(idx, 1);
|
|
10027
|
+
}
|
|
10028
|
+
if (q.length === 0) {
|
|
10029
|
+
pendingAddMessageFifoByFeed.delete(pending.feedId);
|
|
10030
|
+
}
|
|
10031
|
+
}
|
|
10032
|
+
}
|
|
10033
|
+
if (outcome === "ok") {
|
|
10034
|
+
pending.resolve();
|
|
10035
|
+
} else {
|
|
10036
|
+
pending.reject(error3 ?? new Error("Feed mutation failed"));
|
|
10037
|
+
}
|
|
10038
|
+
}
|
|
10039
|
+
function registerFeedMutation(requestId, kind, feedId, options2) {
|
|
10040
|
+
const { promise, resolve, reject } = Promise_withResolvers();
|
|
10041
|
+
const timeoutId = setTimeout(() => {
|
|
10042
|
+
if (pendingFeedMutations.has(requestId)) {
|
|
10043
|
+
settleFeedMutation(
|
|
10044
|
+
requestId,
|
|
10045
|
+
"error",
|
|
10046
|
+
new Error("Feed mutation timeout")
|
|
10047
|
+
);
|
|
10048
|
+
}
|
|
10049
|
+
}, FEEDS_TIMEOUT);
|
|
10050
|
+
pendingFeedMutations.set(requestId, {
|
|
10051
|
+
resolve,
|
|
10052
|
+
reject,
|
|
10053
|
+
timeoutId,
|
|
10054
|
+
kind,
|
|
10055
|
+
feedId,
|
|
10056
|
+
messageId: options2?.messageId,
|
|
10057
|
+
expectedClientMessageId: options2?.expectedClientMessageId
|
|
10058
|
+
});
|
|
10059
|
+
if (kind === "add-message" && options2?.expectedClientMessageId === void 0) {
|
|
10060
|
+
const q = pendingAddMessageFifoByFeed.get(feedId) ?? [];
|
|
10061
|
+
q.push(requestId);
|
|
10062
|
+
pendingAddMessageFifoByFeed.set(feedId, q);
|
|
10063
|
+
}
|
|
10064
|
+
return promise;
|
|
10065
|
+
}
|
|
10066
|
+
function tryResolvePendingFeedMutationsFromFeedsEvent(message) {
|
|
10067
|
+
switch (message.type) {
|
|
10068
|
+
case ServerMsgCode.FEEDS_ADDED: {
|
|
10069
|
+
for (const feed of message.feeds) {
|
|
10070
|
+
for (const [requestId, pending] of [...pendingFeedMutations]) {
|
|
10071
|
+
if (pending.kind === "add-feed" && pending.feedId === feed.feedId) {
|
|
10072
|
+
settleFeedMutation(requestId, "ok");
|
|
10073
|
+
break;
|
|
10074
|
+
}
|
|
10075
|
+
}
|
|
10076
|
+
}
|
|
10077
|
+
break;
|
|
10078
|
+
}
|
|
10079
|
+
case ServerMsgCode.FEEDS_UPDATED: {
|
|
10080
|
+
for (const feed of message.feeds) {
|
|
10081
|
+
for (const [requestId, pending] of [...pendingFeedMutations]) {
|
|
10082
|
+
if (pending.kind === "update-feed" && pending.feedId === feed.feedId) {
|
|
10083
|
+
settleFeedMutation(requestId, "ok");
|
|
10084
|
+
}
|
|
10085
|
+
}
|
|
10086
|
+
}
|
|
10087
|
+
break;
|
|
10088
|
+
}
|
|
10089
|
+
case ServerMsgCode.FEED_DELETED: {
|
|
10090
|
+
for (const [requestId, pending] of [...pendingFeedMutations]) {
|
|
10091
|
+
if (pending.kind === "delete-feed" && pending.feedId === message.feedId) {
|
|
10092
|
+
settleFeedMutation(requestId, "ok");
|
|
10093
|
+
break;
|
|
10094
|
+
}
|
|
10095
|
+
}
|
|
10096
|
+
break;
|
|
10097
|
+
}
|
|
10098
|
+
case ServerMsgCode.FEED_MESSAGES_ADDED: {
|
|
10099
|
+
for (const m of message.messages) {
|
|
10100
|
+
let matched = false;
|
|
10101
|
+
for (const [requestId, pending] of [...pendingFeedMutations]) {
|
|
10102
|
+
if (pending.kind === "add-message" && pending.feedId === message.feedId && pending.expectedClientMessageId === m.id) {
|
|
10103
|
+
settleFeedMutation(requestId, "ok");
|
|
10104
|
+
matched = true;
|
|
10105
|
+
break;
|
|
10106
|
+
}
|
|
10107
|
+
}
|
|
10108
|
+
if (!matched) {
|
|
10109
|
+
const q = pendingAddMessageFifoByFeed.get(message.feedId);
|
|
10110
|
+
const headId = q?.[0];
|
|
10111
|
+
if (headId !== void 0) {
|
|
10112
|
+
const pending = pendingFeedMutations.get(headId);
|
|
10113
|
+
if (pending?.kind === "add-message" && pending.expectedClientMessageId === void 0) {
|
|
10114
|
+
settleFeedMutation(headId, "ok");
|
|
10115
|
+
}
|
|
10116
|
+
}
|
|
10117
|
+
}
|
|
10118
|
+
}
|
|
10119
|
+
break;
|
|
10120
|
+
}
|
|
10121
|
+
case ServerMsgCode.FEED_MESSAGES_UPDATED: {
|
|
10122
|
+
for (const m of message.messages) {
|
|
10123
|
+
for (const [requestId, pending] of [...pendingFeedMutations]) {
|
|
10124
|
+
if (pending.kind === "update-message" && pending.feedId === message.feedId && pending.messageId === m.id) {
|
|
10125
|
+
settleFeedMutation(requestId, "ok");
|
|
10126
|
+
}
|
|
10127
|
+
}
|
|
10128
|
+
}
|
|
10129
|
+
break;
|
|
10130
|
+
}
|
|
10131
|
+
case ServerMsgCode.FEED_MESSAGES_DELETED: {
|
|
10132
|
+
for (const mid of message.messageIds) {
|
|
10133
|
+
for (const [requestId, pending] of [...pendingFeedMutations]) {
|
|
10134
|
+
if (pending.kind === "delete-message" && pending.feedId === message.feedId && pending.messageId === mid) {
|
|
10135
|
+
settleFeedMutation(requestId, "ok");
|
|
10136
|
+
}
|
|
10137
|
+
}
|
|
10138
|
+
}
|
|
10139
|
+
break;
|
|
10140
|
+
}
|
|
10141
|
+
default:
|
|
10142
|
+
break;
|
|
10143
|
+
}
|
|
10144
|
+
}
|
|
10390
10145
|
function processInitialStorage(nodes) {
|
|
10391
10146
|
const unacknowledgedOps = new Map(context.unacknowledgedOps);
|
|
10392
10147
|
createOrUpdateRootFromMessage(nodes);
|
|
@@ -10458,6 +10213,138 @@ function createRoom(options, config) {
|
|
|
10458
10213
|
}
|
|
10459
10214
|
flushNowOrSoon();
|
|
10460
10215
|
}
|
|
10216
|
+
async function fetchFeeds(options2) {
|
|
10217
|
+
const requestId = nanoid();
|
|
10218
|
+
const { promise, resolve, reject } = Promise_withResolvers();
|
|
10219
|
+
pendingFeedsRequests.set(requestId, { resolve, reject });
|
|
10220
|
+
const message = {
|
|
10221
|
+
type: ClientMsgCode.FETCH_FEEDS,
|
|
10222
|
+
requestId,
|
|
10223
|
+
cursor: options2?.cursor,
|
|
10224
|
+
since: options2?.since,
|
|
10225
|
+
limit: options2?.limit,
|
|
10226
|
+
metadata: options2?.metadata
|
|
10227
|
+
};
|
|
10228
|
+
context.buffer.messages.push(message);
|
|
10229
|
+
flushNowOrSoon();
|
|
10230
|
+
setTimeout(() => {
|
|
10231
|
+
if (pendingFeedsRequests.has(requestId)) {
|
|
10232
|
+
pendingFeedsRequests.delete(requestId);
|
|
10233
|
+
reject(new Error("Feeds fetch timeout"));
|
|
10234
|
+
}
|
|
10235
|
+
}, FEEDS_TIMEOUT);
|
|
10236
|
+
return promise;
|
|
10237
|
+
}
|
|
10238
|
+
async function fetchFeedMessages(feedId, options2) {
|
|
10239
|
+
const requestId = nanoid();
|
|
10240
|
+
const { promise, resolve, reject } = Promise_withResolvers();
|
|
10241
|
+
pendingFeedMessagesRequests.set(requestId, { resolve, reject });
|
|
10242
|
+
const message = {
|
|
10243
|
+
type: ClientMsgCode.FETCH_FEED_MESSAGES,
|
|
10244
|
+
requestId,
|
|
10245
|
+
feedId,
|
|
10246
|
+
cursor: options2?.cursor,
|
|
10247
|
+
since: options2?.since,
|
|
10248
|
+
limit: options2?.limit
|
|
10249
|
+
};
|
|
10250
|
+
context.buffer.messages.push(message);
|
|
10251
|
+
flushNowOrSoon();
|
|
10252
|
+
setTimeout(() => {
|
|
10253
|
+
if (pendingFeedMessagesRequests.has(requestId)) {
|
|
10254
|
+
pendingFeedMessagesRequests.delete(requestId);
|
|
10255
|
+
reject(new Error("Feed messages fetch timeout"));
|
|
10256
|
+
}
|
|
10257
|
+
}, FEEDS_TIMEOUT);
|
|
10258
|
+
return promise;
|
|
10259
|
+
}
|
|
10260
|
+
function addFeed(feedId, options2) {
|
|
10261
|
+
const requestId = nanoid();
|
|
10262
|
+
const promise = registerFeedMutation(requestId, "add-feed", feedId);
|
|
10263
|
+
const message = {
|
|
10264
|
+
type: ClientMsgCode.ADD_FEED,
|
|
10265
|
+
requestId,
|
|
10266
|
+
feedId,
|
|
10267
|
+
metadata: options2?.metadata,
|
|
10268
|
+
createdAt: options2?.createdAt
|
|
10269
|
+
};
|
|
10270
|
+
context.buffer.messages.push(message);
|
|
10271
|
+
flushNowOrSoon();
|
|
10272
|
+
return promise;
|
|
10273
|
+
}
|
|
10274
|
+
function updateFeed(feedId, metadata) {
|
|
10275
|
+
const requestId = nanoid();
|
|
10276
|
+
const promise = registerFeedMutation(requestId, "update-feed", feedId);
|
|
10277
|
+
const message = {
|
|
10278
|
+
type: ClientMsgCode.UPDATE_FEED,
|
|
10279
|
+
requestId,
|
|
10280
|
+
feedId,
|
|
10281
|
+
metadata
|
|
10282
|
+
};
|
|
10283
|
+
context.buffer.messages.push(message);
|
|
10284
|
+
flushNowOrSoon();
|
|
10285
|
+
return promise;
|
|
10286
|
+
}
|
|
10287
|
+
function deleteFeed(feedId) {
|
|
10288
|
+
const requestId = nanoid();
|
|
10289
|
+
const promise = registerFeedMutation(requestId, "delete-feed", feedId);
|
|
10290
|
+
const message = {
|
|
10291
|
+
type: ClientMsgCode.DELETE_FEED,
|
|
10292
|
+
requestId,
|
|
10293
|
+
feedId
|
|
10294
|
+
};
|
|
10295
|
+
context.buffer.messages.push(message);
|
|
10296
|
+
flushNowOrSoon();
|
|
10297
|
+
return promise;
|
|
10298
|
+
}
|
|
10299
|
+
function addFeedMessage(feedId, data, options2) {
|
|
10300
|
+
const requestId = nanoid();
|
|
10301
|
+
const promise = registerFeedMutation(requestId, "add-message", feedId, {
|
|
10302
|
+
expectedClientMessageId: options2?.id
|
|
10303
|
+
});
|
|
10304
|
+
const message = {
|
|
10305
|
+
type: ClientMsgCode.ADD_FEED_MESSAGE,
|
|
10306
|
+
requestId,
|
|
10307
|
+
feedId,
|
|
10308
|
+
data,
|
|
10309
|
+
id: options2?.id,
|
|
10310
|
+
createdAt: options2?.createdAt
|
|
10311
|
+
};
|
|
10312
|
+
context.buffer.messages.push(message);
|
|
10313
|
+
flushNowOrSoon();
|
|
10314
|
+
return promise;
|
|
10315
|
+
}
|
|
10316
|
+
function updateFeedMessage(feedId, messageId, data, options2) {
|
|
10317
|
+
const requestId = nanoid();
|
|
10318
|
+
const promise = registerFeedMutation(requestId, "update-message", feedId, {
|
|
10319
|
+
messageId
|
|
10320
|
+
});
|
|
10321
|
+
const message = {
|
|
10322
|
+
type: ClientMsgCode.UPDATE_FEED_MESSAGE,
|
|
10323
|
+
requestId,
|
|
10324
|
+
feedId,
|
|
10325
|
+
messageId,
|
|
10326
|
+
data,
|
|
10327
|
+
updatedAt: options2?.updatedAt
|
|
10328
|
+
};
|
|
10329
|
+
context.buffer.messages.push(message);
|
|
10330
|
+
flushNowOrSoon();
|
|
10331
|
+
return promise;
|
|
10332
|
+
}
|
|
10333
|
+
function deleteFeedMessage(feedId, messageId) {
|
|
10334
|
+
const requestId = nanoid();
|
|
10335
|
+
const promise = registerFeedMutation(requestId, "delete-message", feedId, {
|
|
10336
|
+
messageId
|
|
10337
|
+
});
|
|
10338
|
+
const message = {
|
|
10339
|
+
type: ClientMsgCode.DELETE_FEED_MESSAGE,
|
|
10340
|
+
requestId,
|
|
10341
|
+
feedId,
|
|
10342
|
+
messageId
|
|
10343
|
+
};
|
|
10344
|
+
context.buffer.messages.push(message);
|
|
10345
|
+
flushNowOrSoon();
|
|
10346
|
+
return promise;
|
|
10347
|
+
}
|
|
10461
10348
|
function undo() {
|
|
10462
10349
|
if (context.activeBatch) {
|
|
10463
10350
|
throw new Error("undo is not allowed during a batch");
|
|
@@ -10543,16 +10430,6 @@ function createRoom(options, config) {
|
|
|
10543
10430
|
_addToRealUndoStack(Array.from(frames));
|
|
10544
10431
|
}
|
|
10545
10432
|
}
|
|
10546
|
-
function withoutHistory(fn) {
|
|
10547
|
-
const undoBefore = context.undoStack.length;
|
|
10548
|
-
const redoBefore = context.redoStack.length;
|
|
10549
|
-
try {
|
|
10550
|
-
return fn();
|
|
10551
|
-
} finally {
|
|
10552
|
-
context.undoStack.length = undoBefore;
|
|
10553
|
-
context.redoStack.length = redoBefore;
|
|
10554
|
-
}
|
|
10555
|
-
}
|
|
10556
10433
|
const syncSourceForStorage = config.createSyncSource();
|
|
10557
10434
|
function getStorageStatus() {
|
|
10558
10435
|
if (context.root === void 0) {
|
|
@@ -10610,6 +10487,7 @@ function createRoom(options, config) {
|
|
|
10610
10487
|
storageStatus: eventHub.storageStatus.observable,
|
|
10611
10488
|
ydoc: eventHub.ydoc.observable,
|
|
10612
10489
|
comments: eventHub.comments.observable,
|
|
10490
|
+
feeds: eventHub.feeds.observable,
|
|
10613
10491
|
roomWillDestroy: eventHub.roomWillDestroy.observable
|
|
10614
10492
|
};
|
|
10615
10493
|
async function getThreadsSince(options2) {
|
|
@@ -10819,13 +10697,19 @@ function createRoom(options, config) {
|
|
|
10819
10697
|
id: roomId,
|
|
10820
10698
|
subscribe: makeClassicSubscribeFn(
|
|
10821
10699
|
roomId,
|
|
10822
|
-
|
|
10700
|
+
eventHub,
|
|
10823
10701
|
config.errorEventSource
|
|
10824
10702
|
),
|
|
10825
10703
|
connect: () => managedSocket.connect(),
|
|
10826
10704
|
reconnect: () => managedSocket.reconnect(),
|
|
10827
10705
|
disconnect: () => managedSocket.disconnect(),
|
|
10828
10706
|
destroy: () => {
|
|
10707
|
+
pendingFeedsRequests.forEach(
|
|
10708
|
+
(request) => request.reject(new Error("Room destroyed"))
|
|
10709
|
+
);
|
|
10710
|
+
pendingFeedMessagesRequests.forEach(
|
|
10711
|
+
(request) => request.reject(new Error("Room destroyed"))
|
|
10712
|
+
);
|
|
10829
10713
|
const { roomWillDestroy, ...eventsExceptDestroy } = eventHub;
|
|
10830
10714
|
for (const source of Object.values(eventsExceptDestroy)) {
|
|
10831
10715
|
source.dispose();
|
|
@@ -10851,12 +10735,17 @@ function createRoom(options, config) {
|
|
|
10851
10735
|
canRedo,
|
|
10852
10736
|
clear,
|
|
10853
10737
|
pause: pauseHistory,
|
|
10854
|
-
resume: resumeHistory
|
|
10855
|
-
[kInternal]: {
|
|
10856
|
-
withoutHistory
|
|
10857
|
-
}
|
|
10738
|
+
resume: resumeHistory
|
|
10858
10739
|
},
|
|
10859
10740
|
fetchYDoc,
|
|
10741
|
+
fetchFeeds,
|
|
10742
|
+
fetchFeedMessages,
|
|
10743
|
+
addFeed,
|
|
10744
|
+
updateFeed,
|
|
10745
|
+
deleteFeed,
|
|
10746
|
+
addFeedMessage,
|
|
10747
|
+
updateFeedMessage,
|
|
10748
|
+
deleteFeedMessage,
|
|
10860
10749
|
getStorage,
|
|
10861
10750
|
getStorageSnapshot,
|
|
10862
10751
|
getStorageStatus,
|
|
@@ -11764,6 +11653,309 @@ function toPlainLson(lson) {
|
|
|
11764
11653
|
}
|
|
11765
11654
|
}
|
|
11766
11655
|
|
|
11656
|
+
// src/immutable.ts
|
|
11657
|
+
function lsonObjectToJson(obj) {
|
|
11658
|
+
const result = {};
|
|
11659
|
+
for (const key in obj) {
|
|
11660
|
+
const val = obj[key];
|
|
11661
|
+
if (val !== void 0) {
|
|
11662
|
+
result[key] = lsonToJson(val);
|
|
11663
|
+
}
|
|
11664
|
+
}
|
|
11665
|
+
return result;
|
|
11666
|
+
}
|
|
11667
|
+
function liveObjectToJson(liveObject) {
|
|
11668
|
+
return lsonObjectToJson(liveObject.toObject());
|
|
11669
|
+
}
|
|
11670
|
+
function liveMapToJson(map) {
|
|
11671
|
+
const result = {};
|
|
11672
|
+
for (const [key, value] of map.entries()) {
|
|
11673
|
+
result[key] = lsonToJson(value);
|
|
11674
|
+
}
|
|
11675
|
+
return result;
|
|
11676
|
+
}
|
|
11677
|
+
function lsonListToJson(value) {
|
|
11678
|
+
return value.map(lsonToJson);
|
|
11679
|
+
}
|
|
11680
|
+
function liveListToJson(value) {
|
|
11681
|
+
return lsonListToJson(value.toArray());
|
|
11682
|
+
}
|
|
11683
|
+
function lsonToJson(value) {
|
|
11684
|
+
if (value instanceof LiveObject) {
|
|
11685
|
+
return liveObjectToJson(value);
|
|
11686
|
+
} else if (value instanceof LiveList) {
|
|
11687
|
+
return liveListToJson(value);
|
|
11688
|
+
} else if (value instanceof LiveMap) {
|
|
11689
|
+
return liveMapToJson(value);
|
|
11690
|
+
} else if (value instanceof LiveRegister) {
|
|
11691
|
+
return value.data;
|
|
11692
|
+
}
|
|
11693
|
+
if (Array.isArray(value)) {
|
|
11694
|
+
return lsonListToJson(value);
|
|
11695
|
+
} else if (isPlainObject(value)) {
|
|
11696
|
+
return lsonObjectToJson(value);
|
|
11697
|
+
}
|
|
11698
|
+
return value;
|
|
11699
|
+
}
|
|
11700
|
+
function deepLiveify(value) {
|
|
11701
|
+
if (Array.isArray(value)) {
|
|
11702
|
+
return new LiveList(value.map(deepLiveify));
|
|
11703
|
+
} else if (isPlainObject(value)) {
|
|
11704
|
+
const init = {};
|
|
11705
|
+
for (const key in value) {
|
|
11706
|
+
const val = value[key];
|
|
11707
|
+
if (val === void 0) {
|
|
11708
|
+
continue;
|
|
11709
|
+
}
|
|
11710
|
+
init[key] = deepLiveify(val);
|
|
11711
|
+
}
|
|
11712
|
+
return new LiveObject(init);
|
|
11713
|
+
} else {
|
|
11714
|
+
return value;
|
|
11715
|
+
}
|
|
11716
|
+
}
|
|
11717
|
+
function patchLiveList(liveList, prev, next) {
|
|
11718
|
+
let i = 0;
|
|
11719
|
+
let prevEnd = prev.length - 1;
|
|
11720
|
+
let nextEnd = next.length - 1;
|
|
11721
|
+
let prevNode = prev[0];
|
|
11722
|
+
let nextNode = next[0];
|
|
11723
|
+
outer: {
|
|
11724
|
+
while (prevNode === nextNode) {
|
|
11725
|
+
++i;
|
|
11726
|
+
if (i > prevEnd || i > nextEnd) {
|
|
11727
|
+
break outer;
|
|
11728
|
+
}
|
|
11729
|
+
prevNode = prev[i];
|
|
11730
|
+
nextNode = next[i];
|
|
11731
|
+
}
|
|
11732
|
+
prevNode = prev[prevEnd];
|
|
11733
|
+
nextNode = next[nextEnd];
|
|
11734
|
+
while (prevNode === nextNode) {
|
|
11735
|
+
prevEnd--;
|
|
11736
|
+
nextEnd--;
|
|
11737
|
+
if (i > prevEnd || i > nextEnd) {
|
|
11738
|
+
break outer;
|
|
11739
|
+
}
|
|
11740
|
+
prevNode = prev[prevEnd];
|
|
11741
|
+
nextNode = next[nextEnd];
|
|
11742
|
+
}
|
|
11743
|
+
}
|
|
11744
|
+
if (i > prevEnd) {
|
|
11745
|
+
if (i <= nextEnd) {
|
|
11746
|
+
while (i <= nextEnd) {
|
|
11747
|
+
liveList.insert(deepLiveify(next[i]), i);
|
|
11748
|
+
i++;
|
|
11749
|
+
}
|
|
11750
|
+
}
|
|
11751
|
+
} else if (i > nextEnd) {
|
|
11752
|
+
let localI = i;
|
|
11753
|
+
while (localI <= prevEnd) {
|
|
11754
|
+
liveList.delete(i);
|
|
11755
|
+
localI++;
|
|
11756
|
+
}
|
|
11757
|
+
} else {
|
|
11758
|
+
while (i <= prevEnd && i <= nextEnd) {
|
|
11759
|
+
prevNode = prev[i];
|
|
11760
|
+
nextNode = next[i];
|
|
11761
|
+
const liveListNode = liveList.get(i);
|
|
11762
|
+
if (isLiveObject(liveListNode) && isPlainObject(prevNode) && isPlainObject(nextNode)) {
|
|
11763
|
+
patchLiveObject(liveListNode, prevNode, nextNode);
|
|
11764
|
+
} else {
|
|
11765
|
+
liveList.set(i, deepLiveify(nextNode));
|
|
11766
|
+
}
|
|
11767
|
+
i++;
|
|
11768
|
+
}
|
|
11769
|
+
while (i <= nextEnd) {
|
|
11770
|
+
liveList.insert(deepLiveify(next[i]), i);
|
|
11771
|
+
i++;
|
|
11772
|
+
}
|
|
11773
|
+
let localI = i;
|
|
11774
|
+
while (localI <= prevEnd) {
|
|
11775
|
+
liveList.delete(i);
|
|
11776
|
+
localI++;
|
|
11777
|
+
}
|
|
11778
|
+
}
|
|
11779
|
+
}
|
|
11780
|
+
function patchLiveObjectKey(liveObject, key, prev, next) {
|
|
11781
|
+
if (process.env.NODE_ENV !== "production") {
|
|
11782
|
+
const nonSerializableValue = findNonSerializableValue(next);
|
|
11783
|
+
if (nonSerializableValue) {
|
|
11784
|
+
error2(
|
|
11785
|
+
`New state path: '${nonSerializableValue.path}' value: '${String(
|
|
11786
|
+
nonSerializableValue.value
|
|
11787
|
+
)}' is not serializable.
|
|
11788
|
+
Only serializable value can be synced with Liveblocks.`
|
|
11789
|
+
);
|
|
11790
|
+
return;
|
|
11791
|
+
}
|
|
11792
|
+
}
|
|
11793
|
+
const value = liveObject.get(key);
|
|
11794
|
+
if (next === void 0) {
|
|
11795
|
+
liveObject.delete(key);
|
|
11796
|
+
} else if (value === void 0) {
|
|
11797
|
+
liveObject.set(key, deepLiveify(next));
|
|
11798
|
+
} else if (prev === next) {
|
|
11799
|
+
return;
|
|
11800
|
+
} else if (isLiveList(value) && Array.isArray(prev) && Array.isArray(next)) {
|
|
11801
|
+
patchLiveList(value, prev, next);
|
|
11802
|
+
} else if (isLiveObject(value) && isPlainObject(prev) && isPlainObject(next)) {
|
|
11803
|
+
patchLiveObject(value, prev, next);
|
|
11804
|
+
} else {
|
|
11805
|
+
liveObject.set(key, deepLiveify(next));
|
|
11806
|
+
}
|
|
11807
|
+
}
|
|
11808
|
+
function patchLiveObject(root, prev, next) {
|
|
11809
|
+
const updates = {};
|
|
11810
|
+
for (const key in next) {
|
|
11811
|
+
patchLiveObjectKey(root, key, prev[key], next[key]);
|
|
11812
|
+
}
|
|
11813
|
+
for (const key in prev) {
|
|
11814
|
+
if (next[key] === void 0) {
|
|
11815
|
+
root.delete(key);
|
|
11816
|
+
}
|
|
11817
|
+
}
|
|
11818
|
+
if (Object.keys(updates).length > 0) {
|
|
11819
|
+
root.update(updates);
|
|
11820
|
+
}
|
|
11821
|
+
}
|
|
11822
|
+
function getParentsPath(node) {
|
|
11823
|
+
const path = [];
|
|
11824
|
+
while (node.parent.type === "HasParent") {
|
|
11825
|
+
if (isLiveList(node.parent.node)) {
|
|
11826
|
+
path.push(node.parent.node._indexOfPosition(node.parent.key));
|
|
11827
|
+
} else {
|
|
11828
|
+
path.push(node.parent.key);
|
|
11829
|
+
}
|
|
11830
|
+
node = node.parent.node;
|
|
11831
|
+
}
|
|
11832
|
+
return path;
|
|
11833
|
+
}
|
|
11834
|
+
function legacy_patchImmutableObject(state, updates) {
|
|
11835
|
+
return updates.reduce(
|
|
11836
|
+
(state2, update) => legacy_patchImmutableObjectWithUpdate(state2, update),
|
|
11837
|
+
state
|
|
11838
|
+
);
|
|
11839
|
+
}
|
|
11840
|
+
function legacy_patchImmutableObjectWithUpdate(state, update) {
|
|
11841
|
+
const path = getParentsPath(update.node);
|
|
11842
|
+
return legacy_patchImmutableNode(state, path, update);
|
|
11843
|
+
}
|
|
11844
|
+
function legacy_patchImmutableNode(state, path, update) {
|
|
11845
|
+
const pathItem = path.pop();
|
|
11846
|
+
if (pathItem === void 0) {
|
|
11847
|
+
switch (update.type) {
|
|
11848
|
+
case "LiveObject": {
|
|
11849
|
+
if (!isJsonObject(state)) {
|
|
11850
|
+
throw new Error(
|
|
11851
|
+
"Internal: received update on LiveObject but state was not an object"
|
|
11852
|
+
);
|
|
11853
|
+
}
|
|
11854
|
+
const newState = Object.assign({}, state);
|
|
11855
|
+
for (const key in update.updates) {
|
|
11856
|
+
if (update.updates[key]?.type === "update") {
|
|
11857
|
+
const val = update.node.get(key);
|
|
11858
|
+
if (val !== void 0) {
|
|
11859
|
+
newState[key] = lsonToJson(val);
|
|
11860
|
+
}
|
|
11861
|
+
} else if (update.updates[key]?.type === "delete") {
|
|
11862
|
+
delete newState[key];
|
|
11863
|
+
}
|
|
11864
|
+
}
|
|
11865
|
+
return newState;
|
|
11866
|
+
}
|
|
11867
|
+
case "LiveList": {
|
|
11868
|
+
if (!Array.isArray(state)) {
|
|
11869
|
+
throw new Error(
|
|
11870
|
+
"Internal: received update on LiveList but state was not an array"
|
|
11871
|
+
);
|
|
11872
|
+
}
|
|
11873
|
+
let newState = state.map((x) => x);
|
|
11874
|
+
for (const listUpdate of update.updates) {
|
|
11875
|
+
if (listUpdate.type === "set") {
|
|
11876
|
+
newState = newState.map(
|
|
11877
|
+
(item, index) => index === listUpdate.index ? lsonToJson(listUpdate.item) : item
|
|
11878
|
+
);
|
|
11879
|
+
} else if (listUpdate.type === "insert") {
|
|
11880
|
+
if (listUpdate.index === newState.length) {
|
|
11881
|
+
newState.push(lsonToJson(listUpdate.item));
|
|
11882
|
+
} else {
|
|
11883
|
+
newState = [
|
|
11884
|
+
...newState.slice(0, listUpdate.index),
|
|
11885
|
+
lsonToJson(listUpdate.item),
|
|
11886
|
+
...newState.slice(listUpdate.index)
|
|
11887
|
+
];
|
|
11888
|
+
}
|
|
11889
|
+
} else if (listUpdate.type === "delete") {
|
|
11890
|
+
newState.splice(listUpdate.index, 1);
|
|
11891
|
+
} else if (listUpdate.type === "move") {
|
|
11892
|
+
if (listUpdate.previousIndex > listUpdate.index) {
|
|
11893
|
+
newState = [
|
|
11894
|
+
...newState.slice(0, listUpdate.index),
|
|
11895
|
+
lsonToJson(listUpdate.item),
|
|
11896
|
+
...newState.slice(listUpdate.index, listUpdate.previousIndex),
|
|
11897
|
+
...newState.slice(listUpdate.previousIndex + 1)
|
|
11898
|
+
];
|
|
11899
|
+
} else {
|
|
11900
|
+
newState = [
|
|
11901
|
+
...newState.slice(0, listUpdate.previousIndex),
|
|
11902
|
+
...newState.slice(
|
|
11903
|
+
listUpdate.previousIndex + 1,
|
|
11904
|
+
listUpdate.index + 1
|
|
11905
|
+
),
|
|
11906
|
+
lsonToJson(listUpdate.item),
|
|
11907
|
+
...newState.slice(listUpdate.index + 1)
|
|
11908
|
+
];
|
|
11909
|
+
}
|
|
11910
|
+
}
|
|
11911
|
+
}
|
|
11912
|
+
return newState;
|
|
11913
|
+
}
|
|
11914
|
+
case "LiveMap": {
|
|
11915
|
+
if (!isJsonObject(state)) {
|
|
11916
|
+
throw new Error(
|
|
11917
|
+
"Internal: received update on LiveMap but state was not an object"
|
|
11918
|
+
);
|
|
11919
|
+
}
|
|
11920
|
+
const newState = Object.assign({}, state);
|
|
11921
|
+
for (const key in update.updates) {
|
|
11922
|
+
if (update.updates[key]?.type === "update") {
|
|
11923
|
+
const value = update.node.get(key);
|
|
11924
|
+
if (value !== void 0) {
|
|
11925
|
+
newState[key] = lsonToJson(value);
|
|
11926
|
+
}
|
|
11927
|
+
} else if (update.updates[key]?.type === "delete") {
|
|
11928
|
+
delete newState[key];
|
|
11929
|
+
}
|
|
11930
|
+
}
|
|
11931
|
+
return newState;
|
|
11932
|
+
}
|
|
11933
|
+
}
|
|
11934
|
+
}
|
|
11935
|
+
if (Array.isArray(state)) {
|
|
11936
|
+
const newArray = [...state];
|
|
11937
|
+
newArray[pathItem] = legacy_patchImmutableNode(
|
|
11938
|
+
state[pathItem],
|
|
11939
|
+
path,
|
|
11940
|
+
update
|
|
11941
|
+
);
|
|
11942
|
+
return newArray;
|
|
11943
|
+
} else if (isJsonObject(state)) {
|
|
11944
|
+
const node = state[pathItem];
|
|
11945
|
+
if (node === void 0) {
|
|
11946
|
+
return state;
|
|
11947
|
+
} else {
|
|
11948
|
+
const stateAsObj = state;
|
|
11949
|
+
return {
|
|
11950
|
+
...stateAsObj,
|
|
11951
|
+
[pathItem]: legacy_patchImmutableNode(node, path, update)
|
|
11952
|
+
};
|
|
11953
|
+
}
|
|
11954
|
+
} else {
|
|
11955
|
+
return state;
|
|
11956
|
+
}
|
|
11957
|
+
}
|
|
11958
|
+
|
|
11767
11959
|
// src/lib/abortController.ts
|
|
11768
11960
|
function makeAbortController(externalSignal) {
|
|
11769
11961
|
const ctl = new AbortController();
|
|
@@ -11935,6 +12127,7 @@ export {
|
|
|
11935
12127
|
DefaultMap,
|
|
11936
12128
|
Deque,
|
|
11937
12129
|
DerivedSignal,
|
|
12130
|
+
FeedRequestErrorCode,
|
|
11938
12131
|
HttpError,
|
|
11939
12132
|
LiveList,
|
|
11940
12133
|
LiveMap,
|
|
@@ -11976,7 +12169,6 @@ export {
|
|
|
11976
12169
|
createManagedPool,
|
|
11977
12170
|
createNotificationSettings,
|
|
11978
12171
|
createThreadId,
|
|
11979
|
-
deepLiveifyObject,
|
|
11980
12172
|
defineAiTool,
|
|
11981
12173
|
deprecate,
|
|
11982
12174
|
deprecateIf,
|
|
@@ -12010,7 +12202,6 @@ export {
|
|
|
12010
12202
|
kInternal,
|
|
12011
12203
|
keys,
|
|
12012
12204
|
legacy_patchImmutableObject,
|
|
12013
|
-
legacy_patchLiveObjectKey,
|
|
12014
12205
|
lsonToJson,
|
|
12015
12206
|
makeAbortController,
|
|
12016
12207
|
makeEventSource,
|
|
@@ -12022,6 +12213,7 @@ export {
|
|
|
12022
12213
|
nn,
|
|
12023
12214
|
nodeStreamToCompactNodes,
|
|
12024
12215
|
objectToQuery,
|
|
12216
|
+
patchLiveObjectKey,
|
|
12025
12217
|
patchNotificationSettings,
|
|
12026
12218
|
raise,
|
|
12027
12219
|
resolveMentionsInCommentBody,
|