silverbullet/common/spaces/sync.test.ts

185 lines
5.3 KiB
TypeScript
Raw Permalink Normal View History

2024-07-30 23:33:33 +08:00
import { SpaceSync, type SyncStatusItem } from "./sync.ts";
2023-01-13 22:41:29 +08:00
import { DiskSpacePrimitives } from "./disk_space_primitives.ts";
2024-07-30 23:24:17 +08:00
import { assertEquals } from "@std/assert";
2023-01-13 22:41:29 +08:00
Deno.test("Test store", async () => {
const primaryPath = await Deno.makeTempDir();
const secondaryPath = await Deno.makeTempDir();
console.log("Primary", primaryPath);
console.log("Secondary", secondaryPath);
const primary = new DiskSpacePrimitives(primaryPath);
const secondary = new DiskSpacePrimitives(secondaryPath);
const snapshot = new Map<string, SyncStatusItem>();
const sync = new SpaceSync(primary, secondary, {
conflictResolver: SpaceSync.primaryConflictResolver,
});
2023-01-13 22:41:29 +08:00
// Write one page to primary
await primary.writeFile("index", stringToBytes("Hello"));
2023-01-13 22:41:29 +08:00
assertEquals((await secondary.fetchFileList()).length, 0);
console.log("Initial sync ops", await doSync());
assertEquals((await secondary.fetchFileList()).length, 1);
assertEquals(
(await secondary.readFile("index")).data,
stringToBytes("Hello"),
);
2023-01-13 22:41:29 +08:00
// Should be a no-op
assertEquals(await doSync(), 0);
// Now let's make a change on the secondary
await secondary.writeFile("index", stringToBytes("Hello!!"));
await secondary.writeFile("test", stringToBytes("Test page"));
2023-01-13 22:41:29 +08:00
// And sync it
await doSync();
assertEquals((await primary.fetchFileList()).length, 2);
assertEquals((await secondary.fetchFileList()).length, 2);
assertEquals(
(await primary.readFile("index")).data,
stringToBytes("Hello!!"),
);
2023-01-13 22:41:29 +08:00
// Let's make some random edits on both ends
await primary.writeFile("index", stringToBytes("1"));
await primary.writeFile("index2", stringToBytes("2"));
await secondary.writeFile("index3", stringToBytes("3"));
await secondary.writeFile("index4", stringToBytes("4"));
2023-01-13 22:41:29 +08:00
await doSync();
assertEquals((await primary.fetchFileList()).length, 5);
assertEquals((await secondary.fetchFileList()).length, 5);
assertEquals(await doSync(), 0);
console.log("Deleting pages");
// Delete some pages
await primary.deleteFile("index");
await primary.deleteFile("index3");
await doSync();
assertEquals((await primary.fetchFileList()).length, 3);
assertEquals((await secondary.fetchFileList()).length, 3);
// No-op
assertEquals(await doSync(), 0);
await secondary.deleteFile("index4");
await primary.deleteFile("index2");
await doSync();
// Just "test" left
assertEquals((await primary.fetchFileList()).length, 1);
assertEquals((await secondary.fetchFileList()).length, 1);
// No-op
assertEquals(await doSync(), 0);
await secondary.writeFile("index", stringToBytes("I'm back"));
2023-01-13 22:41:29 +08:00
await doSync();
assertEquals(
(await primary.readFile("index")).data,
stringToBytes("I'm back"),
);
2023-01-13 22:41:29 +08:00
// Cause a conflict
console.log("Introducing a conflict now");
await primary.writeFile("index", stringToBytes("Hello 1"));
await secondary.writeFile("index", stringToBytes("Hello 2"));
2023-01-13 22:41:29 +08:00
await doSync();
// Sync conflicting copy back
await doSync();
// Verify that primary won
assertEquals(
(await primary.readFile("index")).data,
stringToBytes("Hello 1"),
);
assertEquals(
(await secondary.readFile("index")).data,
stringToBytes("Hello 1"),
);
2023-01-13 22:41:29 +08:00
// test + index + index.conflicting copy
assertEquals((await primary.fetchFileList()).length, 3);
assertEquals((await secondary.fetchFileList()).length, 3);
// Introducing a fake conflict (same content, so not really conflicting)
await primary.writeFile("index", stringToBytes("Hello 1"));
await secondary.writeFile("index", stringToBytes("Hello 1"));
2023-01-13 22:41:29 +08:00
await doSync();
await doSync();
// test + index + index.md + previous index.conflicting copy but nothing more
assertEquals((await primary.fetchFileList()).length, 3);
2023-01-13 22:41:29 +08:00
console.log("Bringing a third device in the mix");
const ternaryPath = await Deno.makeTempDir();
console.log("Ternary", ternaryPath);
const ternary = new DiskSpacePrimitives(ternaryPath);
const sync2 = new SpaceSync(
secondary,
ternary,
{
conflictResolver: SpaceSync.primaryConflictResolver,
},
2023-01-13 22:41:29 +08:00
);
const snapshot2 = new Map<string, SyncStatusItem>();
console.log(
"N ops",
await sync2.syncFiles(snapshot2),
);
2023-01-13 22:41:29 +08:00
await sleep(2);
assertEquals(await sync2.syncFiles(snapshot2), 0);
2023-01-13 22:41:29 +08:00
2023-02-28 18:13:18 +08:00
// I had to look up what follows ternary (https://english.stackexchange.com/questions/25116/what-follows-next-in-the-sequence-unary-binary-ternary)
const quaternaryPath = await Deno.makeTempDir();
const quaternary = new DiskSpacePrimitives(quaternaryPath);
const sync3 = new SpaceSync(
secondary,
quaternary,
{
isSyncCandidate: (path) => !path.startsWith("index"),
conflictResolver: SpaceSync.primaryConflictResolver,
2023-02-28 18:13:18 +08:00
},
);
const selectingOps = await sync3.syncFiles(new Map());
2023-02-28 18:13:18 +08:00
assertEquals(selectingOps, 1);
2023-01-13 22:41:29 +08:00
await Deno.remove(primaryPath, { recursive: true });
await Deno.remove(secondaryPath, { recursive: true });
await Deno.remove(ternaryPath, { recursive: true });
2023-02-28 18:13:18 +08:00
await Deno.remove(quaternaryPath, { recursive: true });
2023-01-13 22:41:29 +08:00
async function doSync() {
await sleep();
const r = await sync.syncFiles(snapshot);
2023-01-13 22:41:29 +08:00
await sleep();
return r;
}
});
function sleep(ms = 10): Promise<void> {
return new Promise((resolve) => {
setTimeout(resolve, ms);
});
}
function stringToBytes(s: string): Uint8Array {
return new TextEncoder().encode(s);
}