2023-01-17 01:55:35 +08:00
|
|
|
import { removeDirectiveBody, SpaceSync, SyncStatusItem } from "./sync.ts";
|
2023-01-13 22:41:29 +08:00
|
|
|
import { DiskSpacePrimitives } from "./disk_space_primitives.ts";
|
|
|
|
import { assertEquals } from "../../test_deps.ts";
|
|
|
|
|
|
|
|
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 statusMap = new Map<string, SyncStatusItem>();
|
2023-02-28 18:13:18 +08:00
|
|
|
const sync = new SpaceSync(primary, secondary, statusMap, {});
|
2023-01-13 22:41:29 +08:00
|
|
|
|
|
|
|
// Write one page to primary
|
|
|
|
await primary.writeFile("index", "utf8", "Hello");
|
|
|
|
assertEquals((await secondary.fetchFileList()).length, 0);
|
|
|
|
console.log("Initial sync ops", await doSync());
|
|
|
|
|
|
|
|
assertEquals((await secondary.fetchFileList()).length, 1);
|
|
|
|
assertEquals((await secondary.readFile("index", "utf8")).data, "Hello");
|
|
|
|
|
|
|
|
// Should be a no-op
|
|
|
|
assertEquals(await doSync(), 0);
|
|
|
|
|
|
|
|
// Now let's make a change on the secondary
|
|
|
|
await secondary.writeFile("index", "utf8", "Hello!!");
|
|
|
|
await secondary.writeFile("test", "utf8", "Test page");
|
|
|
|
|
|
|
|
// And sync it
|
|
|
|
await doSync();
|
|
|
|
|
|
|
|
assertEquals((await primary.fetchFileList()).length, 2);
|
|
|
|
assertEquals((await secondary.fetchFileList()).length, 2);
|
|
|
|
|
|
|
|
assertEquals((await primary.readFile("index", "utf8")).data, "Hello!!");
|
|
|
|
|
|
|
|
// Let's make some random edits on both ends
|
|
|
|
await primary.writeFile("index", "utf8", "1");
|
|
|
|
await primary.writeFile("index2", "utf8", "2");
|
|
|
|
await secondary.writeFile("index3", "utf8", "3");
|
|
|
|
await secondary.writeFile("index4", "utf8", "4");
|
|
|
|
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", "utf8", "I'm back");
|
|
|
|
|
|
|
|
await doSync();
|
|
|
|
|
|
|
|
assertEquals((await primary.readFile("index", "utf8")).data, "I'm back");
|
|
|
|
|
|
|
|
// Cause a conflict
|
|
|
|
console.log("Introducing a conflict now");
|
|
|
|
await primary.writeFile("index", "utf8", "Hello 1");
|
|
|
|
await secondary.writeFile("index", "utf8", "Hello 2");
|
|
|
|
|
|
|
|
await doSync();
|
|
|
|
|
|
|
|
// Sync conflicting copy back
|
|
|
|
await doSync();
|
|
|
|
|
|
|
|
// Verify that primary won
|
|
|
|
assertEquals((await primary.readFile("index", "utf8")).data, "Hello 1");
|
|
|
|
assertEquals((await secondary.readFile("index", "utf8")).data, "Hello 1");
|
|
|
|
|
|
|
|
// 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", "utf8", "Hello 1");
|
|
|
|
await secondary.writeFile("index", "utf8", "Hello 1");
|
|
|
|
|
2023-01-17 01:55:35 +08:00
|
|
|
// And two more files with different bodies, but only within a query directive — shouldn't conflict
|
|
|
|
await primary.writeFile(
|
|
|
|
"index.md",
|
|
|
|
"utf8",
|
|
|
|
"Hello\n<!-- #query page -->\nHello 1\n<!-- /query -->",
|
|
|
|
);
|
|
|
|
await secondary.writeFile(
|
|
|
|
"index.md",
|
|
|
|
"utf8",
|
|
|
|
"Hello\n<!-- #query page -->\nHello 2\n<!-- /query -->",
|
|
|
|
);
|
|
|
|
|
2023-01-13 22:41:29 +08:00
|
|
|
await doSync();
|
|
|
|
await doSync();
|
|
|
|
|
2023-01-17 01:55:35 +08:00
|
|
|
// test + index + index.md + previous index.conflicting copy but nothing more
|
|
|
|
assertEquals((await primary.fetchFileList()).length, 4);
|
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,
|
|
|
|
new Map<string, SyncStatusItem>(),
|
2023-02-28 18:13:18 +08:00
|
|
|
{},
|
2023-01-13 22:41:29 +08:00
|
|
|
);
|
2023-01-20 23:08:01 +08:00
|
|
|
console.log(
|
|
|
|
"N ops",
|
|
|
|
await sync2.syncFiles(SpaceSync.primaryConflictResolver),
|
|
|
|
);
|
2023-01-13 22:41:29 +08:00
|
|
|
await sleep(2);
|
2023-01-20 23:08:01 +08:00
|
|
|
assertEquals(await sync2.syncFiles(SpaceSync.primaryConflictResolver), 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,
|
|
|
|
new Map<string, SyncStatusItem>(),
|
|
|
|
{
|
|
|
|
excludePrefixes: ["index"],
|
|
|
|
},
|
|
|
|
);
|
|
|
|
const selectingOps = await sync3.syncFiles(SpaceSync.primaryConflictResolver);
|
|
|
|
|
|
|
|
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(
|
|
|
|
SpaceSync.primaryConflictResolver,
|
|
|
|
);
|
|
|
|
await sleep();
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
function sleep(ms = 10): Promise<void> {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
setTimeout(resolve, ms);
|
|
|
|
});
|
|
|
|
}
|
2023-01-17 01:55:35 +08:00
|
|
|
|
|
|
|
Deno.test("Remove directive bodies", () => {
|
|
|
|
assertEquals(
|
|
|
|
removeDirectiveBody(`<!-- #query page -->
|
|
|
|
This is a body
|
|
|
|
bla bla bla
|
|
|
|
<!-- /query -->
|
|
|
|
Hello
|
|
|
|
<!-- #include [[test]] -->
|
|
|
|
This is a body
|
|
|
|
<!-- /include -->
|
|
|
|
`),
|
|
|
|
`<!-- #query page -->
|
|
|
|
<!-- /query -->
|
|
|
|
Hello
|
|
|
|
<!-- #include [[test]] -->
|
|
|
|
<!-- /include -->
|
|
|
|
`,
|
|
|
|
);
|
|
|
|
});
|