mirror of
https://github.com/loro-dev/loro.git
synced 2025-01-23 05:24:51 +00:00
9ecc0a90b1
This PR includes a BREAKING CHANGE. It enables you to create containers before attaching them to the document, making the API more intuitive and straightforward. A container can be either attached to a document or detached. When it's detached, its history/state does not persist. You can attach a container to a document by inserting it into an attached container. Once a container is attached, its state, along with all of its descendants's states, will be recreated in the document. After attaching, the container and its descendants, will each have their corresponding "attached" version of themselves? When a detached container x is attached to a document, you can use x.getAttached() to obtain the corresponding attached container.
136 lines
3.9 KiB
TypeScript
136 lines
3.9 KiB
TypeScript
import { describe, expect, it } from "vitest";
|
|
import { Loro, LoroMap, OpId, VersionVector } from "../src";
|
|
|
|
describe("Frontiers", () => {
|
|
it("two clients", () => {
|
|
const doc = new Loro();
|
|
doc.setPeerId(0);
|
|
const text = doc.getText("text");
|
|
text.insert(0, "0");
|
|
doc.commit();
|
|
|
|
const v0 = doc.frontiers();
|
|
const docB = new Loro();
|
|
docB.setPeerId(1);
|
|
docB.import(doc.exportFrom());
|
|
expect(docB.cmpWithFrontiers(v0)).toBe(0);
|
|
text.insert(1, "0");
|
|
doc.commit();
|
|
expect(docB.cmpWithFrontiers(doc.frontiers())).toBe(-1);
|
|
const textB = docB.getText("text");
|
|
textB.insert(0, "0");
|
|
docB.commit();
|
|
expect(docB.cmpWithFrontiers(doc.frontiers())).toBe(-1);
|
|
docB.import(doc.exportFrom());
|
|
expect(docB.cmpWithFrontiers(doc.frontiers())).toBe(1);
|
|
doc.import(docB.exportFrom());
|
|
expect(docB.cmpWithFrontiers(doc.frontiers())).toBe(0);
|
|
});
|
|
|
|
it("cmp frontiers", () => {
|
|
const doc1 = new Loro();
|
|
doc1.setPeerId(1);
|
|
const doc2 = new Loro();
|
|
doc2.setPeerId(2n);
|
|
|
|
doc1.getText("text").insert(0, "01234");
|
|
doc2.import(doc1.exportFrom());
|
|
doc2.getText("text").insert(0, "56789");
|
|
doc1.import(doc2.exportFrom());
|
|
doc1.getText("text").insert(0, "01234");
|
|
doc1.commit();
|
|
|
|
expect(() => {
|
|
doc1.cmpFrontiers([{ peer: "1", counter: 1 }], [{
|
|
peer: "2",
|
|
counter: 10,
|
|
}]);
|
|
}).toThrow();
|
|
expect(doc1.cmpFrontiers([], [{ peer: "1", counter: 1 }])).toBe(-1);
|
|
expect(doc1.cmpFrontiers([], [])).toBe(0);
|
|
expect(
|
|
doc1.cmpFrontiers([{ peer: "1", counter: 4 }], [{
|
|
peer: "2",
|
|
counter: 3,
|
|
}]),
|
|
).toBe(-1);
|
|
expect(
|
|
doc1.cmpFrontiers([{ peer: "1", counter: 5 }], [{
|
|
peer: "2",
|
|
counter: 3,
|
|
}]),
|
|
).toBe(1);
|
|
});
|
|
});
|
|
|
|
it("peer id repr should be consistent", () => {
|
|
const doc = new Loro();
|
|
const id = doc.peerIdStr;
|
|
doc.getText("text").insert(0, "hello");
|
|
doc.commit();
|
|
const f = doc.frontiers();
|
|
expect(f[0].peer).toBe(id);
|
|
const child = new LoroMap();
|
|
console.dir(child);
|
|
const map = doc.getList("list").insertContainer(0, child);
|
|
console.dir(child);
|
|
const mapId = map.id;
|
|
const peerIdInContainerId = mapId.split(":")[1].split("@")[1];
|
|
expect(peerIdInContainerId).toBe(id);
|
|
doc.commit();
|
|
expect(doc.version().get(id)).toBe(6);
|
|
expect(doc.version().toJSON().get(id)).toBe(6);
|
|
const m = doc.getMap(mapId);
|
|
m.set("0", 1);
|
|
expect(map.get("0")).toBe(1);
|
|
});
|
|
|
|
describe("Version", () => {
|
|
const a = new Loro();
|
|
a.setPeerId(0n);
|
|
const b = new Loro();
|
|
b.setPeerId(1n);
|
|
a.getText("text").insert(0, "ha");
|
|
b.getText("text").insert(0, "yo");
|
|
a.import(b.exportFrom());
|
|
a.getText("text").insert(0, "k");
|
|
a.commit();
|
|
|
|
it("version vector to frontiers", () => {
|
|
{
|
|
const vv = new Map();
|
|
vv.set("0", 3);
|
|
vv.set("1", 2);
|
|
expect(a.version().toJSON()).toStrictEqual(vv);
|
|
expect(a.version().toJSON()).toStrictEqual(vv);
|
|
expect(a.vvToFrontiers(new VersionVector(vv))).toStrictEqual(
|
|
a.frontiers(),
|
|
);
|
|
const v = a.version();
|
|
const temp = a.vvToFrontiers(v);
|
|
expect(temp).toStrictEqual(a.frontiers());
|
|
expect(a.frontiers()).toStrictEqual(
|
|
[{ peer: "0", counter: 2 }] as OpId[],
|
|
);
|
|
}
|
|
});
|
|
|
|
it("get changes", () => {
|
|
const changes = a.getAllChanges();
|
|
expect(typeof changes.get("0")?.[0].peer == "string").toBeTruthy();
|
|
expect(changes.size).toBe(2);
|
|
expect(changes.get("0")?.length).toBe(2);
|
|
expect(changes.get("0")?.[0].length).toBe(2);
|
|
expect(changes.get("0")?.[1].lamport).toBe(2);
|
|
expect(changes.get("0")?.[1].deps).toStrictEqual([
|
|
{ peer: "0", counter: 1 },
|
|
{ peer: "1", counter: 1 },
|
|
]);
|
|
expect(changes.get("1")?.length).toBe(1);
|
|
});
|
|
|
|
it("get ops inside changes", () => {
|
|
const change = a.getOpsInChange({ peer: "0", counter: 2 });
|
|
expect(change.length).toBe(1);
|
|
});
|
|
});
|