@graffiti-garden/api 0.2.7 → 0.2.8

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.
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../tests/location.ts", "../tests/utils.ts", "../tests/crud.ts", "../tests/synchronize.ts", "../tests/discover.ts", "../tests/orphans.ts", "../tests/channel-stats.ts"],
4
- "sourcesContent": ["import { it, expect, describe } from \"vitest\";\nimport type { Graffiti } from \"@graffiti-garden/api\";\nimport { GraffitiErrorInvalidUri } from \"@graffiti-garden/api\";\nimport { randomString } from \"./utils\";\n\nexport const graffitiLocationTests = (\n useGraffiti: () => Pick<Graffiti, \"locationToUri\" | \"uriToLocation\">,\n) => {\n describe(\"URI and location conversion\", () => {\n it(\"location to uri and back\", async () => {\n const graffiti = useGraffiti();\n const location = {\n name: randomString(),\n actor: randomString(),\n source: randomString(),\n };\n const uri = graffiti.locationToUri(location);\n const location2 = graffiti.uriToLocation(uri);\n expect(location).toEqual(location2);\n });\n\n it(\"collision resistance\", async () => {\n const graffiti = useGraffiti();\n const location1 = {\n name: randomString(),\n actor: randomString(),\n source: randomString(),\n };\n for (const prop of [\"name\", \"actor\", \"source\"] as const) {\n const location2 = { ...location1, [prop]: randomString() };\n const uri1 = graffiti.locationToUri(location1);\n const uri2 = graffiti.locationToUri(location2);\n expect(uri1).not.toEqual(uri2);\n }\n });\n\n it(\"random URI should not be a valid location\", async () => {\n const graffiti = useGraffiti();\n expect(() => graffiti.uriToLocation(\"\")).toThrow(GraffitiErrorInvalidUri);\n });\n });\n};\n", "import { assert } from \"vitest\";\nimport type { GraffitiPutObject, GraffitiStream } from \"@graffiti-garden/api\";\n\nexport function randomString(): string {\n const array = new Uint8Array(16);\n crypto.getRandomValues(array);\n const str = Array.from(array)\n .map((b) => b.toString(16).padStart(2, \"0\"))\n .join(\"\");\n\n // check for unicode support\n return str + \"\uD83D\uDC69\uD83C\uDFFD\u200D\u2764\uFE0F\u200D\uD83D\uDC8B\u200D\uD83D\uDC69\uD83C\uDFFB\uD83E\uDEF1\uD83C\uDFFC\u200D\uD83E\uDEF2\uD83C\uDFFF\";\n}\n\nexport function randomValue() {\n return {\n [randomString()]: randomString(),\n };\n}\n\nexport function randomPutObject(): GraffitiPutObject<{}> {\n return {\n value: randomValue(),\n channels: [randomString(), randomString()],\n };\n}\n\nexport async function nextStreamValue<S, T>(iterator: GraffitiStream<S, T>) {\n const result = await iterator.next();\n assert(!result.done && !result.value.error, \"result has no value\");\n return result.value.value;\n}\n", "import { it, expect, describe } from \"vitest\";\nimport type {\n Graffiti,\n GraffitiSession,\n GraffitiPatch,\n} from \"@graffiti-garden/api\";\nimport {\n GraffitiErrorNotFound,\n GraffitiErrorSchemaMismatch,\n GraffitiErrorInvalidSchema,\n GraffitiErrorForbidden,\n GraffitiErrorPatchTestFailed,\n GraffitiErrorPatchError,\n} from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiCRUDTests = (\n useGraffiti: () => Pick<Graffiti, \"put\" | \"get\" | \"delete\" | \"patch\">,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe(\n \"CRUD\",\n {\n timeout: 20000,\n },\n () => {\n it(\"put, get, delete\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const value = {\n something: \"hello, world~ c:\",\n };\n const channels = [randomString(), randomString()];\n\n // Put the object\n const previous = await graffiti.put({ value, channels }, session);\n expect(previous.value).toEqual({});\n expect(previous.channels).toEqual([]);\n expect(previous.allowed).toBeUndefined();\n expect(previous.actor).toEqual(session.actor);\n\n // Get it back\n const gotten = await graffiti.get(previous, {});\n expect(gotten.value).toEqual(value);\n expect(gotten.channels).toEqual([]);\n expect(gotten.allowed).toBeUndefined();\n expect(gotten.name).toEqual(previous.name);\n expect(gotten.actor).toEqual(previous.actor);\n expect(gotten.source).toEqual(previous.source);\n expect(gotten.lastModified).toEqual(previous.lastModified);\n\n // Replace it\n const newValue = {\n something: \"goodbye, world~ :c\",\n };\n const beforeReplaced = await graffiti.put(\n { ...previous, value: newValue, channels: [] },\n session,\n );\n expect(beforeReplaced.value).toEqual(value);\n expect(beforeReplaced.tombstone).toEqual(true);\n expect(beforeReplaced.name).toEqual(previous.name);\n expect(beforeReplaced.actor).toEqual(previous.actor);\n expect(beforeReplaced.source).toEqual(previous.source);\n expect(beforeReplaced.lastModified).toBeGreaterThan(\n gotten.lastModified,\n );\n\n // Get it again\n const afterReplaced = await graffiti.get(previous, {});\n expect(afterReplaced.value).toEqual(newValue);\n expect(afterReplaced.lastModified).toEqual(beforeReplaced.lastModified);\n expect(afterReplaced.tombstone).toEqual(false);\n\n // Delete it\n const beforeDeleted = await graffiti.delete(afterReplaced, session);\n expect(beforeDeleted.tombstone).toEqual(true);\n expect(beforeDeleted.value).toEqual(newValue);\n expect(beforeDeleted.lastModified).toBeGreaterThan(\n beforeReplaced.lastModified,\n );\n\n // Get a tombstone\n const final = await graffiti.get(afterReplaced, {});\n expect(final).toEqual(beforeDeleted);\n });\n\n it(\"get non-existant\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(randomPutObject(), session);\n await expect(\n graffiti.get(\n {\n ...putted,\n name: randomString(),\n },\n {},\n ),\n ).rejects.toBeInstanceOf(GraffitiErrorNotFound);\n });\n\n it(\"put, get, delete with wrong actor\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n await expect(\n graffiti.put(\n { value: {}, channels: [], actor: session2.actor },\n session1,\n ),\n ).rejects.toThrow(GraffitiErrorForbidden);\n\n const putted = await graffiti.put(\n { value: {}, channels: [] },\n session2,\n );\n\n await expect(graffiti.delete(putted, session1)).rejects.toThrow(\n GraffitiErrorForbidden,\n );\n\n await expect(graffiti.patch({}, putted, session1)).rejects.toThrow(\n GraffitiErrorForbidden,\n );\n });\n\n it(\"put and get with schema\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const schema = {\n properties: {\n value: {\n properties: {\n something: {\n type: \"string\",\n },\n another: {\n type: \"integer\",\n },\n },\n },\n },\n } as const;\n\n const goodValue = {\n something: \"hello\",\n another: 42,\n } as const;\n\n const putted = await graffiti.put<typeof schema>(\n {\n value: goodValue,\n channels: [],\n },\n session,\n );\n\n const gotten = await graffiti.get(putted, schema);\n expect(gotten.value.something).toEqual(goodValue.something);\n expect(gotten.value.another).toEqual(goodValue.another);\n });\n\n it(\"put and get with invalid schema\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put({ value: {}, channels: [] }, session);\n await expect(\n graffiti.get(putted, {\n properties: {\n value: {\n //@ts-ignore\n type: \"asdf\",\n },\n },\n }),\n ).rejects.toThrow(GraffitiErrorInvalidSchema);\n });\n\n it(\"put and get with wrong schema\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(\n {\n value: {\n hello: \"world\",\n },\n channels: [],\n },\n session,\n );\n\n await expect(\n graffiti.get(putted, {\n properties: {\n value: {\n properties: {\n hello: {\n type: \"number\",\n },\n },\n },\n },\n }),\n ).rejects.toThrow(GraffitiErrorSchemaMismatch);\n });\n\n it(\"put and get with empty access control\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const value = {\n um: \"hi\",\n };\n const allowed = [randomString()];\n const channels = [randomString()];\n const putted = await graffiti.put(\n { value, allowed, channels },\n session1,\n );\n\n // Get it with authenticated session\n const gotten = await graffiti.get(putted, {}, session1);\n expect(gotten.value).toEqual(value);\n expect(gotten.allowed).toEqual(allowed);\n expect(gotten.channels).toEqual(channels);\n\n // But not without session\n await expect(graffiti.get(putted, {})).rejects.toBeInstanceOf(\n GraffitiErrorNotFound,\n );\n\n // Or the wrong session\n await expect(graffiti.get(putted, {}, session2)).rejects.toBeInstanceOf(\n GraffitiErrorNotFound,\n );\n });\n\n it(\"put and get with specific access control\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const value = {\n um: \"hi\",\n };\n const allowed = [randomString(), session2.actor, randomString()];\n const channels = [randomString()];\n const putted = await graffiti.put(\n {\n value,\n allowed,\n channels,\n },\n session1,\n );\n\n // Get it with authenticated session\n const gotten = await graffiti.get(putted, {}, session1);\n expect(gotten.value).toEqual(value);\n expect(gotten.allowed).toEqual(allowed);\n expect(gotten.channels).toEqual(channels);\n\n // But not without session\n await expect(graffiti.get(putted, {})).rejects.toBeInstanceOf(\n GraffitiErrorNotFound,\n );\n\n const gotten2 = await graffiti.get(putted, {}, session2);\n expect(gotten2.value).toEqual(value);\n // They should only see that is is private to them\n expect(gotten2.allowed).toEqual([session2.actor]);\n // And not see any channels\n expect(gotten2.channels).toEqual([]);\n });\n\n it(\"patch value\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const value = {\n something: \"hello, world~ c:\",\n };\n const putted = await graffiti.put({ value, channels: [] }, session);\n\n const patch: GraffitiPatch = {\n value: [\n { op: \"replace\", path: \"/something\", value: \"goodbye, world~ :c\" },\n ],\n };\n const beforePatched = await graffiti.patch(patch, putted, session);\n expect(beforePatched.value).toEqual(value);\n expect(beforePatched.tombstone).toBe(true);\n\n const gotten = await graffiti.get(putted, {});\n expect(gotten.value).toEqual({\n something: \"goodbye, world~ :c\",\n });\n expect(beforePatched.lastModified).toBe(gotten.lastModified);\n\n await graffiti.delete(putted, session);\n });\n\n it(\"patch deleted object\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(randomPutObject(), session);\n const deleted = await graffiti.delete(putted, session);\n await expect(\n graffiti.patch({}, putted, session),\n ).rejects.toBeInstanceOf(GraffitiErrorNotFound);\n });\n\n it(\"deep patch\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const value = {\n something: {\n another: {\n somethingElse: \"hello\",\n },\n },\n };\n const putted = await graffiti.put(\n { value: value, channels: [] },\n session,\n );\n\n const beforePatch = await graffiti.patch(\n {\n value: [\n {\n op: \"replace\",\n path: \"/something/another/somethingElse\",\n value: \"goodbye\",\n },\n ],\n },\n putted,\n session,\n );\n const gotten = await graffiti.get(putted, {});\n\n expect(beforePatch.value).toEqual(value);\n expect(gotten.value).toEqual({\n something: {\n another: {\n somethingElse: \"goodbye\",\n },\n },\n });\n });\n\n it(\"patch channels\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const channelsBefore = [randomString()];\n const channelsAfter = [randomString()];\n\n const putted = await graffiti.put(\n { value: {}, channels: channelsBefore },\n session,\n );\n\n const patch: GraffitiPatch = {\n channels: [{ op: \"replace\", path: \"/0\", value: channelsAfter[0] }],\n };\n const patched = await graffiti.patch(patch, putted, session);\n expect(patched.channels).toEqual(channelsBefore);\n const gotten = await graffiti.get(putted, {}, session);\n expect(gotten.channels).toEqual(channelsAfter);\n await graffiti.delete(putted, session);\n });\n\n it(\"patch 'increment' with test\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(\n {\n value: {\n counter: 1,\n },\n channels: [],\n },\n session,\n );\n\n const previous = await graffiti.patch(\n {\n value: [\n { op: \"test\", path: \"/counter\", value: 1 },\n { op: \"replace\", path: \"/counter\", value: 2 },\n ],\n },\n putted,\n session,\n );\n expect(previous.value).toEqual({ counter: 1 });\n const result = await graffiti.get(previous, {\n properties: {\n value: {\n properties: {\n counter: {\n type: \"integer\",\n },\n },\n },\n },\n });\n expect(result.value.counter).toEqual(2);\n\n await expect(\n graffiti.patch(\n {\n value: [\n { op: \"test\", path: \"/counter\", value: 1 },\n { op: \"replace\", path: \"/counter\", value: 3 },\n ],\n },\n putted,\n session,\n ),\n ).rejects.toThrow(GraffitiErrorPatchTestFailed);\n });\n\n it(\"invalid patch\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n\n await expect(\n graffiti.patch(\n {\n value: [\n { op: \"add\", path: \"/root\", value: [] },\n { op: \"add\", path: \"/root/2\", value: 2 }, // out of bounds\n ],\n },\n putted,\n session,\n ),\n ).rejects.toThrow(GraffitiErrorPatchError);\n });\n\n it(\"patch channels to be wrong\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n object.allowed = [randomString()];\n const putted = await graffiti.put(object, session);\n\n const patches: GraffitiPatch[] = [\n {\n channels: [{ op: \"replace\", path: \"\", value: null }],\n },\n {\n channels: [{ op: \"replace\", path: \"\", value: {} }],\n },\n {\n channels: [{ op: \"replace\", path: \"\", value: [\"hello\", [\"hi\"]] }],\n },\n {\n channels: [{ op: \"add\", path: \"/0\", value: 1 }],\n },\n {\n value: [{ op: \"replace\", path: \"\", value: \"not an object\" }],\n },\n {\n value: [{ op: \"replace\", path: \"\", value: null }],\n },\n {\n value: [{ op: \"replace\", path: \"\", value: [] }],\n },\n {\n allowed: [{ op: \"replace\", path: \"\", value: {} }],\n },\n {\n allowed: [{ op: \"replace\", path: \"\", value: [\"hello\", [\"hi\"]] }],\n },\n ];\n\n for (const patch of patches) {\n await expect(graffiti.patch(patch, putted, session)).rejects.toThrow(\n GraffitiErrorPatchError,\n );\n }\n\n const gotten = await graffiti.get(putted, {}, session);\n expect(gotten.value).toEqual(object.value);\n expect(gotten.channels).toEqual(object.channels);\n expect(gotten.allowed).toEqual(object.allowed);\n expect(gotten.lastModified).toEqual(putted.lastModified);\n });\n },\n );\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type { GraffitiFactory, GraffitiSession } from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiSynchronizeTests = (\n useGraffiti: GraffitiFactory,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe(\"synchronizeDiscover\", () => {\n it(\"get\", async () => {\n const graffiti1 = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const channels = object.channels.slice(1);\n const putted = await graffiti1.put(object, session);\n\n const graffiti2 = useGraffiti();\n const next = graffiti2.synchronizeDiscover(channels, {}).next();\n const gotten = await graffiti2.get(putted, {}, session);\n\n const result = (await next).value;\n if (!result || result.error) {\n throw new Error(\"Error in synchronize\");\n }\n expect(result.value.value).toEqual(object.value);\n expect(result.value.channels).toEqual(channels);\n expect(result.value.tombstone).toBe(false);\n expect(result.value.lastModified).toEqual(gotten.lastModified);\n });\n\n it(\"put\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const beforeChannel = randomString();\n const afterChannel = randomString();\n const sharedChannel = randomString();\n\n const oldValue = { hello: \"world\" };\n const oldChannels = [beforeChannel, sharedChannel];\n const putted = await graffiti.put(\n {\n value: oldValue,\n channels: oldChannels,\n },\n session,\n );\n\n // Start listening for changes...\n const before = graffiti.synchronizeDiscover([beforeChannel], {}).next();\n const after = graffiti.synchronizeDiscover([afterChannel], {}).next();\n const shared = graffiti.synchronizeDiscover([sharedChannel], {}).next();\n\n // Replace the object\n const newValue = { goodbye: \"world\" };\n const newChannels = [afterChannel, sharedChannel];\n await graffiti.put(\n {\n ...putted,\n value: newValue,\n channels: newChannels,\n },\n session,\n );\n\n const beforeResult = (await before).value;\n const afterResult = (await after).value;\n const sharedResult = (await shared).value;\n if (\n !beforeResult ||\n beforeResult.error ||\n !afterResult ||\n afterResult.error ||\n !sharedResult ||\n sharedResult.error\n ) {\n throw new Error(\"Error in synchronize\");\n }\n\n expect(beforeResult.value.value).toEqual(oldValue);\n expect(beforeResult.value.channels).toEqual([beforeChannel]);\n expect(beforeResult.value.tombstone).toBe(true);\n expect(afterResult.value.value).toEqual(newValue);\n expect(afterResult.value.channels).toEqual([afterChannel]);\n expect(afterResult.value.tombstone).toBe(false);\n expect(sharedResult.value.value).toEqual(newValue);\n expect(sharedResult.value.channels).toEqual([sharedChannel]);\n expect(sharedResult.value.tombstone).toBe(false);\n expect(beforeResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n expect(sharedResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n });\n\n it(\"patch\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const beforeChannel = randomString();\n const afterChannel = randomString();\n const sharedChannel = randomString();\n\n const oldValue = { hello: \"world\" };\n const oldChannels = [beforeChannel, sharedChannel];\n const putted = await graffiti.put(\n {\n value: oldValue,\n channels: oldChannels,\n },\n session,\n );\n\n // Start listening for changes...\n const before = graffiti.synchronizeDiscover([beforeChannel], {}).next();\n const after = graffiti.synchronizeDiscover([afterChannel], {}).next();\n const shared = graffiti.synchronizeDiscover([sharedChannel], {}).next();\n\n await graffiti.patch(\n {\n value: [\n {\n op: \"add\",\n path: \"/something\",\n value: \"new value\",\n },\n ],\n channels: [\n {\n op: \"add\",\n path: \"/-\",\n value: afterChannel,\n },\n {\n op: \"remove\",\n path: `/${oldChannels.indexOf(beforeChannel)}`,\n },\n ],\n },\n putted,\n session,\n );\n\n const beforeResult = (await before).value;\n const afterResult = (await after).value;\n const sharedResult = (await shared).value;\n if (\n !beforeResult ||\n beforeResult.error ||\n !afterResult ||\n afterResult.error ||\n !sharedResult ||\n sharedResult.error\n ) {\n throw new Error(\"Error in synchronize\");\n }\n\n const newValue = { ...oldValue, something: \"new value\" };\n const newChannels = [sharedChannel, afterChannel];\n expect(beforeResult.value.value).toEqual(oldValue);\n expect(beforeResult.value.channels).toEqual([beforeChannel]);\n expect(beforeResult.value.tombstone).toBe(true);\n expect(afterResult.value.value).toEqual(newValue);\n expect(afterResult.value.channels).toEqual([afterChannel]);\n expect(afterResult.value.tombstone).toBe(false);\n expect(sharedResult.value.value).toEqual(newValue);\n expect(sharedResult.value.channels).toEqual([sharedChannel]);\n expect(sharedResult.value.tombstone).toBe(false);\n expect(beforeResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n expect(sharedResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n });\n\n it(\"delete\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const channels = [randomString(), randomString(), randomString()];\n\n const oldValue = { hello: \"world\" };\n const oldChannels = [randomString(), ...channels.slice(1)];\n const putted = await graffiti.put(\n {\n value: oldValue,\n channels: oldChannels,\n },\n session,\n );\n\n const next = graffiti.synchronizeDiscover(channels, {}).next();\n\n graffiti.delete(putted, session);\n\n const result = (await next).value;\n if (!result || result.error) {\n throw new Error(\"Error in synchronize\");\n }\n expect(result.value.tombstone).toBe(true);\n expect(result.value.value).toEqual(oldValue);\n expect(result.value.channels).toEqual(\n channels.filter((c) => oldChannels.includes(c)),\n );\n });\n\n it(\"not allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const allChannels = [randomString(), randomString(), randomString()];\n const channels = allChannels.slice(1);\n\n const creatorNext = graffiti\n .synchronizeDiscover(channels, {}, session1)\n .next();\n const allowedNext = graffiti\n .synchronizeDiscover(channels, {}, session2)\n .next();\n const noSession = graffiti.synchronizeDiscover(channels, {}).next();\n\n const value = {\n hello: \"world\",\n };\n const allowed = [randomString(), session2.actor];\n await graffiti.put({ value, channels: allChannels, allowed }, session1);\n\n // Expect no session to time out!\n await expect(\n Promise.race([\n noSession,\n new Promise((resolve, rejects) =>\n setTimeout(rejects, 100, \"Timeout\"),\n ),\n ]),\n ).rejects.toThrow(\"Timeout\");\n\n const creatorResult = (await creatorNext).value;\n const allowedResult = (await allowedNext).value;\n\n if (\n !creatorResult ||\n creatorResult.error ||\n !allowedResult ||\n allowedResult.error\n ) {\n throw new Error(\"Error in synchronize\");\n }\n\n expect(creatorResult.value.value).toEqual(value);\n expect(creatorResult.value.allowed).toEqual(allowed);\n expect(creatorResult.value.channels).toEqual(allChannels);\n expect(allowedResult.value.value).toEqual(value);\n expect(allowedResult.value.allowed).toEqual([session2.actor]);\n expect(allowedResult.value.channels).toEqual(channels);\n });\n });\n\n describe(\"synchronizeGet\", () => {\n it(\"replace, delete\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n\n const iterator = graffiti.synchronizeGet(putted, {});\n const next = iterator.next();\n\n // Change the object\n const newValue = { goodbye: \"world\" };\n const putted2 = await graffiti.put(\n {\n ...putted,\n value: newValue,\n },\n session,\n );\n\n const result = (await next).value;\n assert(result && !result.error);\n\n expect(result.value.value).toEqual(newValue);\n expect(result.value.actor).toEqual(session.actor);\n expect(result.value.channels).toEqual([]);\n expect(result.value.tombstone).toBe(false);\n expect(result.value.lastModified).toEqual(putted2.lastModified);\n expect(result.value.allowed).toBeUndefined();\n\n // Delete the object\n const deleted = await graffiti.delete(putted2, session);\n const result2 = (await iterator.next()).value;\n assert(result2 && !result2.error);\n expect(result2.value.tombstone).toBe(true);\n expect(result2.value.lastModified).toEqual(deleted.lastModified);\n\n // Put something else\n await graffiti.put(randomPutObject(), session);\n await expect(\n Promise.race([\n iterator.next(),\n new Promise((resolve, reject) => setTimeout(reject, 100, \"Timeout\")),\n ]),\n ).rejects.toThrow(\"Timeout\");\n });\n\n it(\"not allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n const putted = await graffiti.put(object, session1);\n\n const iterator1 = graffiti.synchronizeGet(putted, {}, session1);\n const iterator2 = graffiti.synchronizeGet(putted, {}, session2);\n\n const next1 = iterator1.next();\n const next2 = iterator2.next();\n\n const newValue = { goodbye: \"world\" };\n const putted2 = await graffiti.put(\n {\n ...putted,\n ...object,\n allowed: [],\n value: newValue,\n },\n session1,\n );\n const result1 = (await next1).value;\n const result2 = (await next2).value;\n assert(result1 && !result1.error);\n assert(result2 && !result2.error);\n\n expect(result1.value.value).toEqual(newValue);\n expect(result2.value.value).toEqual(object.value);\n expect(result1.value.actor).toEqual(session1.actor);\n expect(result2.value.actor).toEqual(session1.actor);\n expect(result1.value.channels).toEqual(object.channels);\n expect(result2.value.channels).toEqual([]);\n expect(result1.value.tombstone).toBe(false);\n expect(result2.value.tombstone).toBe(true);\n expect(result1.value.lastModified).toEqual(putted2.lastModified);\n expect(result2.value.lastModified).toEqual(putted2.lastModified);\n });\n });\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type {\n Graffiti,\n GraffitiSession,\n JSONSchema4,\n} from \"@graffiti-garden/api\";\nimport { randomString, nextStreamValue, randomPutObject } from \"./utils\";\n\nexport const graffitiDiscoverTests = (\n useGraffiti: () => Pick<Graffiti, \"discover\" | \"put\" | \"delete\" | \"patch\">,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe(\"discover\", { timeout: 20000 }, () => {\n it(\"discover nothing\", async () => {\n const graffiti = useGraffiti();\n const iterator = graffiti.discover([], {});\n expect(await iterator.next()).toHaveProperty(\"done\", true);\n });\n\n it(\"discover single\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n\n const putted = await graffiti.put(object, session);\n\n const queryChannels = [randomString(), object.channels[0]];\n const iterator = graffiti.discover(queryChannels, {});\n const value = await nextStreamValue(iterator);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual([object.channels[0]]);\n expect(value.allowed).toBeUndefined();\n expect(value.actor).toEqual(session.actor);\n expect(value.tombstone).toBe(false);\n expect(value.lastModified).toEqual(putted.lastModified);\n const result2 = await iterator.next();\n expect(result2.done).toBe(true);\n });\n\n it(\"discover wrong channel\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n await graffiti.put(object, session);\n const iterator = graffiti.discover([randomString()], {});\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover not allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.allowed = [randomString(), randomString()];\n const putted = await graffiti.put(object, session1);\n\n const iteratorSession1 = graffiti.discover(object.channels, {}, session1);\n const value = await nextStreamValue(iteratorSession1);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual(object.channels);\n expect(value.allowed).toEqual(object.allowed);\n expect(value.actor).toEqual(session1.actor);\n expect(value.tombstone).toBe(false);\n expect(value.lastModified).toEqual(putted.lastModified);\n\n const iteratorSession2 = graffiti.discover(object.channels, {}, session2);\n expect(await iteratorSession2.next()).toHaveProperty(\"done\", true);\n\n const iteratorNoSession = graffiti.discover(object.channels, {});\n expect(await iteratorNoSession.next()).toHaveProperty(\"done\", true);\n });\n\n it(\"discover allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.allowed = [randomString(), session2.actor, randomString()];\n const putted = await graffiti.put(object, session1);\n\n const iteratorSession2 = graffiti.discover(object.channels, {}, session2);\n const value = await nextStreamValue(iteratorSession2);\n expect(value.value).toEqual(object.value);\n expect(value.allowed).toEqual([session2.actor]);\n expect(value.channels).toEqual(object.channels);\n expect(value.actor).toEqual(session1.actor);\n expect(value.tombstone).toBe(false);\n expect(value.lastModified).toEqual(putted.lastModified);\n });\n\n for (const prop of [\"name\", \"actor\", \"lastModified\"] as const) {\n it(`discover for ${prop}`, async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object1 = randomPutObject();\n const putted1 = await graffiti.put(object1, session1);\n\n const object2 = randomPutObject();\n object2.channels = object1.channels;\n // Make sure the lastModified is different for the query\n await new Promise((r) => setTimeout(r, 20));\n const putted2 = await graffiti.put(object2, session2);\n\n const iterator = graffiti.discover(object1.channels, {\n properties: {\n [prop]: {\n enum: [putted1[prop]],\n },\n },\n });\n\n const value = await nextStreamValue(iterator);\n expect(value.name).toEqual(putted1.name);\n expect(value.name).not.toEqual(putted2.name);\n expect(value.value).toEqual(object1.value);\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n }\n\n it(\"discover with lastModified range\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const putted1 = await graffiti.put(object, session);\n // Make sure the lastModified is different\n await new Promise((r) => setTimeout(r, 20));\n const putted2 = await graffiti.put(object, session);\n\n expect(putted1.name).not.toEqual(putted2.name);\n expect(putted1.lastModified).toBeLessThan(putted2.lastModified);\n\n const gtIterator = graffiti.discover([object.channels[0]], {\n properties: {\n lastModified: {\n minimum: putted2.lastModified,\n exclusiveMinimum: true,\n },\n },\n });\n expect(await gtIterator.next()).toHaveProperty(\"done\", true);\n const gtIteratorEpsilon = graffiti.discover([object.channels[0]], {\n properties: {\n lastModified: {\n minimum: putted2.lastModified - 0.1,\n exclusiveMinimum: true,\n },\n },\n });\n const value1 = await nextStreamValue(gtIteratorEpsilon);\n expect(value1.name).toEqual(putted2.name);\n expect(await gtIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n const gteIterator = graffiti.discover(object.channels, {\n properties: {\n value: {},\n lastModified: {\n minimum: putted2.lastModified,\n },\n },\n });\n const value = await nextStreamValue(gteIterator);\n expect(value.name).toEqual(putted2.name);\n expect(await gteIterator.next()).toHaveProperty(\"done\", true);\n const gteIteratorEpsilon = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n minimum: putted2.lastModified + 0.1,\n },\n },\n });\n expect(await gteIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n\n const ltIterator = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified,\n exclusiveMaximum: true,\n },\n },\n });\n expect(await ltIterator.next()).toHaveProperty(\"done\", true);\n\n const ltIteratorEpsilon = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified + 0.1,\n exclusiveMaximum: true,\n },\n },\n });\n const value3 = await nextStreamValue(ltIteratorEpsilon);\n expect(value3.name).toEqual(putted1.name);\n expect(await ltIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n\n const lteIterator = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified,\n },\n },\n });\n const value2 = await nextStreamValue(lteIterator);\n expect(value2.name).toEqual(putted1.name);\n expect(await lteIterator.next()).toHaveProperty(\"done\", true);\n\n const lteIteratorEpsilon = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified - 0.1,\n },\n },\n });\n expect(await lteIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n });\n\n it(\"discover schema allowed, as and not as owner\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.allowed = [randomString(), session2.actor, randomString()];\n await graffiti.put(object, session1);\n\n const iteratorSession1 = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n minItems: 3,\n // Make sure session2.actor is in the allow list\n not: {\n items: {\n not: {\n enum: [session2.actor],\n },\n },\n },\n },\n },\n },\n session1,\n );\n const value = await nextStreamValue(iteratorSession1);\n expect(value.value).toEqual(object.value);\n await expect(iteratorSession1.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n\n const iteratorSession2BigAllow = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n minItems: 3,\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2BigAllow.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2PeekOther = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n not: {\n items: {\n not: {\n enum: [object.channels[0]],\n },\n },\n },\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2PeekOther.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2SmallAllowPeekSelf = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n maxItems: 1,\n not: {\n items: {\n not: {\n enum: [session2.actor],\n },\n },\n },\n },\n },\n },\n session2,\n );\n const value2 = await nextStreamValue(iteratorSession2SmallAllowPeekSelf);\n expect(value2.value).toEqual(object.value);\n await expect(\n iteratorSession2SmallAllowPeekSelf.next(),\n ).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover schema channels, as and not as owner\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.channels = [randomString(), randomString(), randomString()];\n await graffiti.put(object, session1);\n\n const iteratorSession1 = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n channels: {\n minItems: 3,\n // Make sure session2.actor is in the allow list\n not: {\n items: {\n not: {\n enum: [object.channels[1]],\n },\n },\n },\n },\n },\n },\n session1,\n );\n const value = await nextStreamValue(iteratorSession1);\n expect(value.value).toEqual(object.value);\n await expect(iteratorSession1.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n\n const iteratorSession2BigAllow = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n channels: {\n minItems: 3,\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2BigAllow.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2PeekOther = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n channels: {\n not: {\n items: {\n not: {\n enum: [object.channels[1]],\n },\n },\n },\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2PeekOther.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2SmallAllowPeekSelf = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n allowed: {\n maxItems: 2,\n not: {\n items: {\n not: {\n enum: [object.channels[2]],\n },\n },\n },\n },\n },\n },\n session2,\n );\n const value2 = await nextStreamValue(iteratorSession2SmallAllowPeekSelf);\n expect(value2.value).toEqual(object.value);\n await expect(\n iteratorSession2SmallAllowPeekSelf.next(),\n ).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover query for empty allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n\n const publicO = randomPutObject();\n\n const publicSchema = {\n not: {\n required: [\"allowed\"],\n },\n } satisfies JSONSchema4;\n\n await graffiti.put(publicO, session1);\n const iterator = graffiti.discover(\n publicO.channels,\n publicSchema,\n session1,\n );\n const value = await nextStreamValue(iterator);\n expect(value.value).toEqual(publicO.value);\n expect(value.allowed).toBeUndefined();\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n\n const restricted = randomPutObject();\n restricted.allowed = [];\n await graffiti.put(restricted, session1);\n const iterator2 = graffiti.discover(\n restricted.channels,\n publicSchema,\n session1,\n );\n await expect(iterator2.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover query for values\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object1 = randomPutObject();\n object1.value = { test: randomString() };\n await graffiti.put(object1, session);\n\n const object2 = randomPutObject();\n object2.channels = object1.channels;\n object2.value = { test: randomString(), something: randomString() };\n await graffiti.put(object2, session);\n\n const object3 = randomPutObject();\n object3.channels = object1.channels;\n object3.value = { other: randomString(), something: randomString() };\n await graffiti.put(object3, session);\n\n const counts = new Map<string, number>();\n for (const property of [\"test\", \"something\", \"other\"] as const) {\n let count = 0;\n for await (const result of graffiti.discover(object1.channels, {\n properties: {\n value: {\n required: [property],\n },\n },\n })) {\n assert(!result.error, \"result has error\");\n if (property in result.value.value) {\n count++;\n }\n }\n counts.set(property, count);\n }\n\n expect(counts.get(\"test\")).toBe(2);\n expect(counts.get(\"something\")).toBe(2);\n expect(counts.get(\"other\")).toBe(1);\n });\n\n it(\"discover for deleted content\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n const deleted = await graffiti.delete(putted, session);\n\n const iterator = graffiti.discover(object.channels, {});\n const value = await nextStreamValue(iterator);\n expect(value.tombstone).toBe(true);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual(object.channels);\n expect(value.actor).toEqual(session.actor);\n expect(value.lastModified).toEqual(deleted.lastModified);\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover for replaced channels\", async () => {\n // Do this a bunch to check for concurrency issues\n for (let i = 0; i < 10; i++) {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object1 = randomPutObject();\n const putted = await graffiti.put(object1, session);\n const object2 = randomPutObject();\n const replaced = await graffiti.put(\n {\n ...putted,\n ...object2,\n },\n session,\n );\n\n const iterator1 = graffiti.discover(object1.channels, {});\n const value1 = await nextStreamValue(iterator1);\n await expect(iterator1.next()).resolves.toHaveProperty(\"done\", true);\n\n const iterator2 = graffiti.discover(object2.channels, {});\n const value2 = await nextStreamValue(iterator2);\n await expect(iterator2.next()).resolves.toHaveProperty(\"done\", true);\n\n // If they have the same timestamp, except\n // only one to have a tombstone\n if (putted.lastModified === replaced.lastModified) {\n expect(value1.tombstone || value2.tombstone).toBe(true);\n expect(value1.tombstone && value2.tombstone).toBe(false);\n } else {\n expect(value1.tombstone).toBe(true);\n expect(value1.value).toEqual(object1.value);\n expect(value1.channels).toEqual(object1.channels);\n expect(value1.lastModified).toEqual(replaced.lastModified);\n\n expect(value2.tombstone).toBe(false);\n expect(value2.value).toEqual(object2.value);\n expect(value2.channels).toEqual(object2.channels);\n expect(value2.lastModified).toEqual(replaced.lastModified);\n }\n }\n });\n\n it(\"discover for patched allowed\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n await graffiti.patch(\n {\n allowed: [{ op: \"add\", path: \"\", value: [] }],\n },\n putted,\n session,\n );\n const iterator = graffiti.discover(object.channels, {});\n const value = await nextStreamValue(iterator);\n expect(value.tombstone).toBe(true);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual(object.channels);\n expect(value.allowed).toBeUndefined();\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"put concurrently and discover one\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n object.name = randomString();\n\n const putPromises = Array(100)\n .fill(0)\n .map(() => graffiti.put(object, session));\n await Promise.all(putPromises);\n\n const iterator = graffiti.discover(object.channels, {});\n let tombstoneCount = 0;\n let valueCount = 0;\n for await (const result of iterator) {\n assert(!result.error, \"result has error\");\n if (result.value.tombstone) {\n tombstoneCount++;\n } else {\n valueCount++;\n }\n }\n expect(tombstoneCount).toBe(99);\n expect(valueCount).toBe(1);\n });\n });\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type { Graffiti, GraffitiSession } from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiOrphanTests = (\n useGraffiti: () => Pick<\n Graffiti,\n \"recoverOrphans\" | \"put\" | \"delete\" | \"patch\"\n >,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe(\"recoverOrphans\", () => {\n it(\"list orphans\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const existingOrphans: string[] = [];\n const orphanIterator1 = graffiti.recoverOrphans({}, session);\n for await (const orphan of orphanIterator1) {\n if (orphan.error) continue;\n existingOrphans.push(orphan.value.name);\n }\n\n const object = randomPutObject();\n object.channels = [];\n const putted = await graffiti.put(object, session);\n const orphanIterator2 = graffiti.recoverOrphans({}, session);\n let numResults = 0;\n for await (const orphan of orphanIterator2) {\n if (orphan.error) continue;\n if (orphan.value.name === putted.name) {\n numResults++;\n expect(orphan.value.source).toBe(putted.source);\n expect(orphan.value.lastModified).toBe(putted.lastModified);\n }\n }\n expect(numResults).toBe(1);\n });\n\n it(\"replaced orphan, no longer\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n object.channels = [];\n const putOrphan = await graffiti.put(object, session);\n\n const putNotOrphan = await graffiti.put(\n {\n ...putOrphan,\n ...object,\n channels: [randomString()],\n },\n session,\n );\n expect(putNotOrphan.name).toBe(putOrphan.name);\n\n const orphanIterator = graffiti.recoverOrphans({}, session);\n let numResults = 0;\n for await (const orphan of orphanIterator) {\n if (orphan.error) continue;\n if (orphan.value.name === putOrphan.name) {\n numResults++;\n expect(orphan.value.tombstone).toBe(true);\n expect(orphan.value.lastModified).toBe(putNotOrphan.lastModified);\n expect(orphan.value.channels).toEqual([]);\n }\n }\n expect(numResults).toBe(1);\n });\n });\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type { Graffiti, GraffitiSession } from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiChannelStatsTests = (\n useGraffiti: () => Pick<\n Graffiti,\n \"channelStats\" | \"put\" | \"delete\" | \"patch\"\n >,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe(\"channel stats\", () => {\n it(\"list channels\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const existingChannels: Map<string, number> = new Map();\n const channelIterator1 = graffiti.channelStats(session);\n for await (const channel of channelIterator1) {\n if (channel.error) continue;\n existingChannels.set(channel.value.channel, channel.value.count);\n }\n\n const channels = [randomString(), randomString(), randomString()];\n\n // Add one value to channels[0],\n // two values to both channels[0] and channels[1],\n // three values to all channels\n // one value to channels[2]\n for (let i = 0; i < 3; i++) {\n for (let j = 0; j < i + 1; j++) {\n await graffiti.put(\n {\n value: {\n index: j,\n },\n channels: channels.slice(0, i + 1),\n },\n session,\n );\n }\n }\n await graffiti.put(\n { value: { index: 3 }, channels: [channels[2]] },\n session,\n );\n\n const channelIterator2 = graffiti.channelStats(session);\n let newChannels: Map<string, number> = new Map();\n for await (const channel of channelIterator2) {\n if (channel.error) continue;\n newChannels.set(channel.value.channel, channel.value.count);\n }\n // Filter out existing channels\n newChannels = new Map(\n Array.from(newChannels).filter(\n ([channel, count]) => !existingChannels.has(channel),\n ),\n );\n expect(newChannels.size).toBe(3);\n expect(newChannels.get(channels[0])).toBe(6);\n expect(newChannels.get(channels[1])).toBe(5);\n expect(newChannels.get(channels[2])).toBe(4);\n });\n\n it(\"list channels with deleted channel\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const channels = [randomString(), randomString(), randomString()];\n\n // Add an item with two channels\n const before = await graffiti.put(\n {\n value: { index: 2 },\n channels: channels.slice(1),\n },\n session,\n );\n\n // Add an item with all channels\n const first = await graffiti.put(\n { value: { index: 0 }, channels },\n session,\n );\n // But then delete it\n await graffiti.delete(first, session);\n\n // Create a new object with only one channel\n const second = await graffiti.put(\n {\n value: { index: 1 },\n channels: channels.slice(2),\n },\n session,\n );\n\n const channelIterator = graffiti.channelStats(session);\n\n let got1 = 0;\n let got2 = 0;\n for await (const result of channelIterator) {\n if (result.error) continue;\n const { channel, count, lastModified } = result.value;\n assert(\n channel !== channels[0],\n \"There should not be an object in channel[0]\",\n );\n if (channel === channels[1]) {\n expect(count).toBe(1);\n expect(lastModified).toBe(before.lastModified);\n got1++;\n } else if (channel === channels[2]) {\n expect(count).toBe(2);\n expect(lastModified).toBe(second.lastModified);\n got2++;\n }\n }\n expect(got1).toBe(1);\n expect(got2).toBe(1);\n });\n });\n};\n"],
5
- "mappings": "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",
6
- "names": ["it", "expect", "describe", "GraffitiErrorInvalidUri", "assert", "randomString", "array", "b", "randomValue", "randomPutObject", "nextStreamValue", "iterator", "result", "graffitiLocationTests", "useGraffiti", "describe", "it", "graffiti", "location", "randomString", "uri", "location2", "expect", "location1", "prop", "uri1", "uri2", "GraffitiErrorInvalidUri", "it", "expect", "describe", "GraffitiErrorNotFound", "GraffitiErrorSchemaMismatch", "GraffitiErrorInvalidSchema", "GraffitiErrorForbidden", "GraffitiErrorPatchTestFailed", "GraffitiErrorPatchError", "graffitiCRUDTests", "useGraffiti", "useSession1", "useSession2", "describe", "it", "graffiti", "session", "value", "channels", "randomString", "previous", "expect", "gotten", "newValue", "beforeReplaced", "afterReplaced", "beforeDeleted", "final", "putted", "randomPutObject", "GraffitiErrorNotFound", "session1", "session2", "GraffitiErrorForbidden", "schema", "goodValue", "GraffitiErrorInvalidSchema", "GraffitiErrorSchemaMismatch", "allowed", "gotten2", "patch", "beforePatched", "deleted", "beforePatch", "channelsBefore", "channelsAfter", "patched", "result", "GraffitiErrorPatchTestFailed", "object", "GraffitiErrorPatchError", "patches", "it", "expect", "describe", "assert", "graffitiSynchronizeTests", "useGraffiti", "useSession1", "useSession2", "describe", "it", "graffiti1", "session", "object", "randomPutObject", "channels", "putted", "graffiti2", "next", "gotten", "result", "expect", "graffiti", "beforeChannel", "randomString", "afterChannel", "sharedChannel", "oldValue", "oldChannels", "before", "after", "shared", "newValue", "newChannels", "beforeResult", "afterResult", "sharedResult", "c", "session1", "session2", "allChannels", "creatorNext", "allowedNext", "noSession", "value", "allowed", "resolve", "rejects", "creatorResult", "allowedResult", "iterator", "putted2", "assert", "deleted", "result2", "reject", "iterator1", "iterator2", "next1", "next2", "result1", "it", "expect", "describe", "assert", "graffitiDiscoverTests", "useGraffiti", "useSession1", "useSession2", "describe", "it", "iterator", "expect", "graffiti", "session", "object", "randomPutObject", "putted", "queryChannels", "randomString", "value", "nextStreamValue", "result2", "session1", "session2", "iteratorSession1", "iteratorSession2", "iteratorNoSession", "prop", "object1", "putted1", "object2", "r", "putted2", "gtIterator", "gtIteratorEpsilon", "value1", "gteIterator", "gteIteratorEpsilon", "ltIterator", "ltIteratorEpsilon", "value3", "lteIterator", "value2", "lteIteratorEpsilon", "iteratorSession2BigAllow", "iteratorSession2PeekOther", "iteratorSession2SmallAllowPeekSelf", "publicO", "publicSchema", "restricted", "iterator2", "object3", "counts", "property", "count", "result", "assert", "deleted", "i", "replaced", "iterator1", "putPromises", "tombstoneCount", "valueCount", "it", "expect", "describe", "graffitiOrphanTests", "useGraffiti", "useSession1", "useSession2", "describe", "it", "graffiti", "session", "existingOrphans", "orphanIterator1", "orphan", "object", "randomPutObject", "putted", "orphanIterator2", "numResults", "expect", "putOrphan", "putNotOrphan", "randomString", "orphanIterator", "it", "expect", "describe", "assert", "graffitiChannelStatsTests", "useGraffiti", "useSession1", "useSession2", "describe", "it", "graffiti", "session", "existingChannels", "channelIterator1", "channel", "channels", "randomString", "i", "j", "channelIterator2", "newChannels", "count", "expect", "before", "first", "second", "channelIterator", "got1", "got2", "result", "lastModified", "assert"]
4
+ "sourcesContent": ["import { it, expect, describe } from \"vitest\";\nimport type { Graffiti } from \"@graffiti-garden/api\";\nimport { GraffitiErrorInvalidUri } from \"@graffiti-garden/api\";\nimport { randomString } from \"./utils\";\n\nexport const graffitiLocationTests = (\n useGraffiti: () => Pick<Graffiti, \"locationToUri\" | \"uriToLocation\">,\n) => {\n describe.concurrent(\"URI and location conversion\", () => {\n it(\"location to uri and back\", async () => {\n const graffiti = useGraffiti();\n const location = {\n name: randomString(),\n actor: randomString(),\n source: randomString(),\n };\n const uri = graffiti.locationToUri(location);\n const location2 = graffiti.uriToLocation(uri);\n expect(location).toEqual(location2);\n });\n\n it(\"collision resistance\", async () => {\n const graffiti = useGraffiti();\n const location1 = {\n name: randomString(),\n actor: randomString(),\n source: randomString(),\n };\n for (const prop of [\"name\", \"actor\", \"source\"] as const) {\n const location2 = { ...location1, [prop]: randomString() };\n const uri1 = graffiti.locationToUri(location1);\n const uri2 = graffiti.locationToUri(location2);\n expect(uri1).not.toEqual(uri2);\n }\n });\n\n it(\"random URI should not be a valid location\", async () => {\n const graffiti = useGraffiti();\n expect(() => graffiti.uriToLocation(\"\")).toThrow(GraffitiErrorInvalidUri);\n });\n });\n};\n", "import { assert } from \"vitest\";\nimport type { GraffitiPutObject, GraffitiStream } from \"@graffiti-garden/api\";\n\nexport function randomString(): string {\n const array = new Uint8Array(16);\n crypto.getRandomValues(array);\n const str = Array.from(array)\n .map((b) => b.toString(16).padStart(2, \"0\"))\n .join(\"\");\n\n // check for unicode support\n return str + \"\uD83D\uDC69\uD83C\uDFFD\u200D\u2764\uFE0F\u200D\uD83D\uDC8B\u200D\uD83D\uDC69\uD83C\uDFFB\uD83E\uDEF1\uD83C\uDFFC\u200D\uD83E\uDEF2\uD83C\uDFFF\";\n}\n\nexport function randomValue() {\n return {\n [randomString()]: randomString(),\n };\n}\n\nexport function randomPutObject(): GraffitiPutObject<{}> {\n return {\n value: randomValue(),\n channels: [randomString(), randomString()],\n };\n}\n\nexport async function nextStreamValue<S, T>(iterator: GraffitiStream<S, T>) {\n const result = await iterator.next();\n assert(!result.done && !result.value.error, \"result has no value\");\n return result.value.value;\n}\n", "import { it, expect, describe } from \"vitest\";\nimport type {\n Graffiti,\n GraffitiSession,\n GraffitiPatch,\n} from \"@graffiti-garden/api\";\nimport {\n GraffitiErrorNotFound,\n GraffitiErrorSchemaMismatch,\n GraffitiErrorInvalidSchema,\n GraffitiErrorForbidden,\n GraffitiErrorPatchTestFailed,\n GraffitiErrorPatchError,\n} from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiCRUDTests = (\n useGraffiti: () => Pick<Graffiti, \"put\" | \"get\" | \"delete\" | \"patch\">,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe.concurrent(\n \"CRUD\",\n {\n timeout: 20000,\n },\n () => {\n it(\"put, get, delete\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const value = {\n something: \"hello, world~ c:\",\n };\n const channels = [randomString(), randomString()];\n\n // Put the object\n const previous = await graffiti.put({ value, channels }, session);\n expect(previous.value).toEqual({});\n expect(previous.channels).toEqual([]);\n expect(previous.allowed).toBeUndefined();\n expect(previous.actor).toEqual(session.actor);\n\n // Get it back\n const gotten = await graffiti.get(previous, {});\n expect(gotten.value).toEqual(value);\n expect(gotten.channels).toEqual([]);\n expect(gotten.allowed).toBeUndefined();\n expect(gotten.name).toEqual(previous.name);\n expect(gotten.actor).toEqual(previous.actor);\n expect(gotten.source).toEqual(previous.source);\n expect(gotten.lastModified).toEqual(previous.lastModified);\n\n // Replace it\n const newValue = {\n something: \"goodbye, world~ :c\",\n };\n const beforeReplaced = await graffiti.put(\n { ...previous, value: newValue, channels: [] },\n session,\n );\n expect(beforeReplaced.value).toEqual(value);\n expect(beforeReplaced.tombstone).toEqual(true);\n expect(beforeReplaced.name).toEqual(previous.name);\n expect(beforeReplaced.actor).toEqual(previous.actor);\n expect(beforeReplaced.source).toEqual(previous.source);\n expect(beforeReplaced.lastModified).toBeGreaterThanOrEqual(\n gotten.lastModified,\n );\n\n // Get it again\n const afterReplaced = await graffiti.get(previous, {});\n expect(afterReplaced.value).toEqual(newValue);\n expect(afterReplaced.lastModified).toEqual(beforeReplaced.lastModified);\n expect(afterReplaced.tombstone).toEqual(false);\n\n // Delete it\n const beforeDeleted = await graffiti.delete(afterReplaced, session);\n expect(beforeDeleted.tombstone).toEqual(true);\n expect(beforeDeleted.value).toEqual(newValue);\n expect(beforeDeleted.lastModified).toBeGreaterThanOrEqual(\n beforeReplaced.lastModified,\n );\n\n // Get a tombstone\n const final = await graffiti.get(afterReplaced, {});\n expect(final).toEqual(beforeDeleted);\n });\n\n it(\"get non-existant\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(randomPutObject(), session);\n await expect(\n graffiti.get(\n {\n ...putted,\n name: randomString(),\n },\n {},\n ),\n ).rejects.toBeInstanceOf(GraffitiErrorNotFound);\n });\n\n it(\"put, get, delete with wrong actor\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n await expect(\n graffiti.put(\n { value: {}, channels: [], actor: session2.actor },\n session1,\n ),\n ).rejects.toThrow(GraffitiErrorForbidden);\n\n const putted = await graffiti.put(\n { value: {}, channels: [] },\n session2,\n );\n\n await expect(graffiti.delete(putted, session1)).rejects.toThrow(\n GraffitiErrorForbidden,\n );\n\n await expect(graffiti.patch({}, putted, session1)).rejects.toThrow(\n GraffitiErrorForbidden,\n );\n });\n\n it(\"put and get with schema\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const schema = {\n properties: {\n value: {\n properties: {\n something: {\n type: \"string\",\n },\n another: {\n type: \"integer\",\n },\n },\n },\n },\n } as const;\n\n const goodValue = {\n something: \"hello\",\n another: 42,\n } as const;\n\n const putted = await graffiti.put<typeof schema>(\n {\n value: goodValue,\n channels: [],\n },\n session,\n );\n\n const gotten = await graffiti.get(putted, schema);\n expect(gotten.value.something).toEqual(goodValue.something);\n expect(gotten.value.another).toEqual(goodValue.another);\n });\n\n it(\"put and get with invalid schema\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put({ value: {}, channels: [] }, session);\n await expect(\n graffiti.get(putted, {\n properties: {\n value: {\n //@ts-ignore\n type: \"asdf\",\n },\n },\n }),\n ).rejects.toThrow(GraffitiErrorInvalidSchema);\n });\n\n it(\"put and get with wrong schema\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(\n {\n value: {\n hello: \"world\",\n },\n channels: [],\n },\n session,\n );\n\n await expect(\n graffiti.get(putted, {\n properties: {\n value: {\n properties: {\n hello: {\n type: \"number\",\n },\n },\n },\n },\n }),\n ).rejects.toThrow(GraffitiErrorSchemaMismatch);\n });\n\n it(\"put and get with empty access control\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const value = {\n um: \"hi\",\n };\n const allowed = [randomString()];\n const channels = [randomString()];\n const putted = await graffiti.put(\n { value, allowed, channels },\n session1,\n );\n\n // Get it with authenticated session\n const gotten = await graffiti.get(putted, {}, session1);\n expect(gotten.value).toEqual(value);\n expect(gotten.allowed).toEqual(allowed);\n expect(gotten.channels).toEqual(channels);\n\n // But not without session\n await expect(graffiti.get(putted, {})).rejects.toBeInstanceOf(\n GraffitiErrorNotFound,\n );\n\n // Or the wrong session\n await expect(graffiti.get(putted, {}, session2)).rejects.toBeInstanceOf(\n GraffitiErrorNotFound,\n );\n });\n\n it(\"put and get with specific access control\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const value = {\n um: \"hi\",\n };\n const allowed = [randomString(), session2.actor, randomString()];\n const channels = [randomString()];\n const putted = await graffiti.put(\n {\n value,\n allowed,\n channels,\n },\n session1,\n );\n\n // Get it with authenticated session\n const gotten = await graffiti.get(putted, {}, session1);\n expect(gotten.value).toEqual(value);\n expect(gotten.allowed).toEqual(allowed);\n expect(gotten.channels).toEqual(channels);\n\n // But not without session\n await expect(graffiti.get(putted, {})).rejects.toBeInstanceOf(\n GraffitiErrorNotFound,\n );\n\n const gotten2 = await graffiti.get(putted, {}, session2);\n expect(gotten2.value).toEqual(value);\n // They should only see that is is private to them\n expect(gotten2.allowed).toEqual([session2.actor]);\n // And not see any channels\n expect(gotten2.channels).toEqual([]);\n });\n\n it(\"patch value\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const value = {\n something: \"hello, world~ c:\",\n };\n const putted = await graffiti.put({ value, channels: [] }, session);\n\n // Wait just a bit to make sure the lastModified is different\n await new Promise((resolve) => setTimeout(resolve, 10));\n\n const patch: GraffitiPatch = {\n value: [\n { op: \"replace\", path: \"/something\", value: \"goodbye, world~ :c\" },\n ],\n };\n const beforePatched = await graffiti.patch(patch, putted, session);\n expect(beforePatched.value).toEqual(value);\n expect(beforePatched.tombstone).toBe(true);\n expect(beforePatched.lastModified).toBeGreaterThan(putted.lastModified);\n\n const gotten = await graffiti.get(putted, {});\n expect(gotten.value).toEqual({\n something: \"goodbye, world~ :c\",\n });\n expect(beforePatched.lastModified).toBe(gotten.lastModified);\n\n await graffiti.delete(putted, session);\n });\n\n it(\"patch deleted object\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(randomPutObject(), session);\n const deleted = await graffiti.delete(putted, session);\n await expect(\n graffiti.patch({}, putted, session),\n ).rejects.toBeInstanceOf(GraffitiErrorNotFound);\n });\n\n it(\"deep patch\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const value = {\n something: {\n another: {\n somethingElse: \"hello\",\n },\n },\n };\n const putted = await graffiti.put(\n { value: value, channels: [] },\n session,\n );\n\n const beforePatch = await graffiti.patch(\n {\n value: [\n {\n op: \"replace\",\n path: \"/something/another/somethingElse\",\n value: \"goodbye\",\n },\n ],\n },\n putted,\n session,\n );\n const gotten = await graffiti.get(putted, {});\n\n expect(beforePatch.value).toEqual(value);\n expect(gotten.value).toEqual({\n something: {\n another: {\n somethingElse: \"goodbye\",\n },\n },\n });\n });\n\n it(\"patch channels\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const channelsBefore = [randomString()];\n const channelsAfter = [randomString()];\n\n const putted = await graffiti.put(\n { value: {}, channels: channelsBefore },\n session,\n );\n\n const patch: GraffitiPatch = {\n channels: [{ op: \"replace\", path: \"/0\", value: channelsAfter[0] }],\n };\n const patched = await graffiti.patch(patch, putted, session);\n expect(patched.channels).toEqual(channelsBefore);\n const gotten = await graffiti.get(putted, {}, session);\n expect(gotten.channels).toEqual(channelsAfter);\n await graffiti.delete(putted, session);\n });\n\n it(\"patch 'increment' with test\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const putted = await graffiti.put(\n {\n value: {\n counter: 1,\n },\n channels: [],\n },\n session,\n );\n\n const previous = await graffiti.patch(\n {\n value: [\n { op: \"test\", path: \"/counter\", value: 1 },\n { op: \"replace\", path: \"/counter\", value: 2 },\n ],\n },\n putted,\n session,\n );\n expect(previous.value).toEqual({ counter: 1 });\n const result = await graffiti.get(previous, {\n properties: {\n value: {\n properties: {\n counter: {\n type: \"integer\",\n },\n },\n },\n },\n });\n expect(result.value.counter).toEqual(2);\n\n await expect(\n graffiti.patch(\n {\n value: [\n { op: \"test\", path: \"/counter\", value: 1 },\n { op: \"replace\", path: \"/counter\", value: 3 },\n ],\n },\n putted,\n session,\n ),\n ).rejects.toThrow(GraffitiErrorPatchTestFailed);\n });\n\n it(\"invalid patch\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n\n await expect(\n graffiti.patch(\n {\n value: [\n { op: \"add\", path: \"/root\", value: [] },\n { op: \"add\", path: \"/root/2\", value: 2 }, // out of bounds\n ],\n },\n putted,\n session,\n ),\n ).rejects.toThrow(GraffitiErrorPatchError);\n });\n\n it(\"patch channels to be wrong\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n object.allowed = [randomString()];\n const putted = await graffiti.put(object, session);\n\n const patches: GraffitiPatch[] = [\n {\n channels: [{ op: \"replace\", path: \"\", value: null }],\n },\n {\n channels: [{ op: \"replace\", path: \"\", value: {} }],\n },\n {\n channels: [{ op: \"replace\", path: \"\", value: [\"hello\", [\"hi\"]] }],\n },\n {\n channels: [{ op: \"add\", path: \"/0\", value: 1 }],\n },\n {\n value: [{ op: \"replace\", path: \"\", value: \"not an object\" }],\n },\n {\n value: [{ op: \"replace\", path: \"\", value: null }],\n },\n {\n value: [{ op: \"replace\", path: \"\", value: [] }],\n },\n {\n allowed: [{ op: \"replace\", path: \"\", value: {} }],\n },\n {\n allowed: [{ op: \"replace\", path: \"\", value: [\"hello\", [\"hi\"]] }],\n },\n ];\n\n for (const patch of patches) {\n await expect(graffiti.patch(patch, putted, session)).rejects.toThrow(\n GraffitiErrorPatchError,\n );\n }\n\n const gotten = await graffiti.get(putted, {}, session);\n expect(gotten.value).toEqual(object.value);\n expect(gotten.channels).toEqual(object.channels);\n expect(gotten.allowed).toEqual(object.allowed);\n expect(gotten.lastModified).toEqual(putted.lastModified);\n });\n },\n );\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type { GraffitiFactory, GraffitiSession } from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiSynchronizeTests = (\n useGraffiti: GraffitiFactory,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe.concurrent(\"synchronizeDiscover\", () => {\n it(\"get\", async () => {\n const graffiti1 = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const channels = object.channels.slice(1);\n const putted = await graffiti1.put(object, session);\n\n const graffiti2 = useGraffiti();\n const next = graffiti2.synchronizeDiscover(channels, {}).next();\n const gotten = await graffiti2.get(putted, {}, session);\n\n const result = (await next).value;\n if (!result || result.error) {\n throw new Error(\"Error in synchronize\");\n }\n expect(result.value.value).toEqual(object.value);\n expect(result.value.channels).toEqual(channels);\n expect(result.value.tombstone).toBe(false);\n expect(result.value.lastModified).toEqual(gotten.lastModified);\n });\n\n it(\"put\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const beforeChannel = randomString();\n const afterChannel = randomString();\n const sharedChannel = randomString();\n\n const oldValue = { hello: \"world\" };\n const oldChannels = [beforeChannel, sharedChannel];\n const putted = await graffiti.put(\n {\n value: oldValue,\n channels: oldChannels,\n },\n session,\n );\n\n // Start listening for changes...\n const before = graffiti.synchronizeDiscover([beforeChannel], {}).next();\n const after = graffiti.synchronizeDiscover([afterChannel], {}).next();\n const shared = graffiti.synchronizeDiscover([sharedChannel], {}).next();\n\n // Replace the object\n const newValue = { goodbye: \"world\" };\n const newChannels = [afterChannel, sharedChannel];\n await graffiti.put(\n {\n ...putted,\n value: newValue,\n channels: newChannels,\n },\n session,\n );\n\n const beforeResult = (await before).value;\n const afterResult = (await after).value;\n const sharedResult = (await shared).value;\n if (\n !beforeResult ||\n beforeResult.error ||\n !afterResult ||\n afterResult.error ||\n !sharedResult ||\n sharedResult.error\n ) {\n throw new Error(\"Error in synchronize\");\n }\n\n expect(beforeResult.value.value).toEqual(oldValue);\n expect(beforeResult.value.channels).toEqual([beforeChannel]);\n expect(beforeResult.value.tombstone).toBe(true);\n expect(afterResult.value.value).toEqual(newValue);\n expect(afterResult.value.channels).toEqual([afterChannel]);\n expect(afterResult.value.tombstone).toBe(false);\n expect(sharedResult.value.value).toEqual(newValue);\n expect(sharedResult.value.channels).toEqual([sharedChannel]);\n expect(sharedResult.value.tombstone).toBe(false);\n expect(beforeResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n expect(sharedResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n });\n\n it(\"patch\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const beforeChannel = randomString();\n const afterChannel = randomString();\n const sharedChannel = randomString();\n\n const oldValue = { hello: \"world\" };\n const oldChannels = [beforeChannel, sharedChannel];\n const putted = await graffiti.put(\n {\n value: oldValue,\n channels: oldChannels,\n },\n session,\n );\n\n // Start listening for changes...\n const before = graffiti.synchronizeDiscover([beforeChannel], {}).next();\n const after = graffiti.synchronizeDiscover([afterChannel], {}).next();\n const shared = graffiti.synchronizeDiscover([sharedChannel], {}).next();\n\n await graffiti.patch(\n {\n value: [\n {\n op: \"add\",\n path: \"/something\",\n value: \"new value\",\n },\n ],\n channels: [\n {\n op: \"add\",\n path: \"/-\",\n value: afterChannel,\n },\n {\n op: \"remove\",\n path: `/${oldChannels.indexOf(beforeChannel)}`,\n },\n ],\n },\n putted,\n session,\n );\n\n const beforeResult = (await before).value;\n const afterResult = (await after).value;\n const sharedResult = (await shared).value;\n if (\n !beforeResult ||\n beforeResult.error ||\n !afterResult ||\n afterResult.error ||\n !sharedResult ||\n sharedResult.error\n ) {\n throw new Error(\"Error in synchronize\");\n }\n\n const newValue = { ...oldValue, something: \"new value\" };\n const newChannels = [sharedChannel, afterChannel];\n expect(beforeResult.value.value).toEqual(oldValue);\n expect(beforeResult.value.channels).toEqual([beforeChannel]);\n expect(beforeResult.value.tombstone).toBe(true);\n expect(afterResult.value.value).toEqual(newValue);\n expect(afterResult.value.channels).toEqual([afterChannel]);\n expect(afterResult.value.tombstone).toBe(false);\n expect(sharedResult.value.value).toEqual(newValue);\n expect(sharedResult.value.channels).toEqual([sharedChannel]);\n expect(sharedResult.value.tombstone).toBe(false);\n expect(beforeResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n expect(sharedResult.value.lastModified).toEqual(\n afterResult.value.lastModified,\n );\n });\n\n it(\"delete\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const channels = [randomString(), randomString(), randomString()];\n\n const oldValue = { hello: \"world\" };\n const oldChannels = [randomString(), ...channels.slice(1)];\n const putted = await graffiti.put(\n {\n value: oldValue,\n channels: oldChannels,\n },\n session,\n );\n\n const next = graffiti.synchronizeDiscover(channels, {}).next();\n\n graffiti.delete(putted, session);\n\n const result = (await next).value;\n if (!result || result.error) {\n throw new Error(\"Error in synchronize\");\n }\n expect(result.value.tombstone).toBe(true);\n expect(result.value.value).toEqual(oldValue);\n expect(result.value.channels).toEqual(\n channels.filter((c) => oldChannels.includes(c)),\n );\n });\n\n it(\"not allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const allChannels = [randomString(), randomString(), randomString()];\n const channels = allChannels.slice(1);\n\n const creatorNext = graffiti\n .synchronizeDiscover(channels, {}, session1)\n .next();\n const allowedNext = graffiti\n .synchronizeDiscover(channels, {}, session2)\n .next();\n const noSession = graffiti.synchronizeDiscover(channels, {}).next();\n\n const value = {\n hello: \"world\",\n };\n const allowed = [randomString(), session2.actor];\n await graffiti.put({ value, channels: allChannels, allowed }, session1);\n\n // Expect no session to time out!\n await expect(\n Promise.race([\n noSession,\n new Promise((resolve, rejects) =>\n setTimeout(rejects, 100, \"Timeout\"),\n ),\n ]),\n ).rejects.toThrow(\"Timeout\");\n\n const creatorResult = (await creatorNext).value;\n const allowedResult = (await allowedNext).value;\n\n if (\n !creatorResult ||\n creatorResult.error ||\n !allowedResult ||\n allowedResult.error\n ) {\n throw new Error(\"Error in synchronize\");\n }\n\n expect(creatorResult.value.value).toEqual(value);\n expect(creatorResult.value.allowed).toEqual(allowed);\n expect(creatorResult.value.channels).toEqual(allChannels);\n expect(allowedResult.value.value).toEqual(value);\n expect(allowedResult.value.allowed).toEqual([session2.actor]);\n expect(allowedResult.value.channels).toEqual(channels);\n });\n });\n\n describe(\"synchronizeGet\", () => {\n it(\"replace, delete\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n\n const iterator = graffiti.synchronizeGet(putted, {});\n const next = iterator.next();\n\n // Change the object\n const newValue = { goodbye: \"world\" };\n const putted2 = await graffiti.put(\n {\n ...putted,\n value: newValue,\n },\n session,\n );\n\n const result = (await next).value;\n assert(result && !result.error);\n\n expect(result.value.value).toEqual(newValue);\n expect(result.value.actor).toEqual(session.actor);\n expect(result.value.channels).toEqual([]);\n expect(result.value.tombstone).toBe(false);\n expect(result.value.lastModified).toEqual(putted2.lastModified);\n expect(result.value.allowed).toBeUndefined();\n\n // Delete the object\n const deleted = await graffiti.delete(putted2, session);\n const result2 = (await iterator.next()).value;\n assert(result2 && !result2.error);\n expect(result2.value.tombstone).toBe(true);\n expect(result2.value.lastModified).toEqual(deleted.lastModified);\n\n // Put something else\n await graffiti.put(randomPutObject(), session);\n await expect(\n Promise.race([\n iterator.next(),\n new Promise((resolve, reject) => setTimeout(reject, 100, \"Timeout\")),\n ]),\n ).rejects.toThrow(\"Timeout\");\n });\n\n it(\"not allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n const putted = await graffiti.put(object, session1);\n\n const iterator1 = graffiti.synchronizeGet(putted, {}, session1);\n const iterator2 = graffiti.synchronizeGet(putted, {}, session2);\n\n const next1 = iterator1.next();\n const next2 = iterator2.next();\n\n const newValue = { goodbye: \"world\" };\n const putted2 = await graffiti.put(\n {\n ...putted,\n ...object,\n allowed: [],\n value: newValue,\n },\n session1,\n );\n const result1 = (await next1).value;\n const result2 = (await next2).value;\n assert(result1 && !result1.error);\n assert(result2 && !result2.error);\n\n expect(result1.value.value).toEqual(newValue);\n expect(result2.value.value).toEqual(object.value);\n expect(result1.value.actor).toEqual(session1.actor);\n expect(result2.value.actor).toEqual(session1.actor);\n expect(result1.value.channels).toEqual(object.channels);\n expect(result2.value.channels).toEqual([]);\n expect(result1.value.tombstone).toBe(false);\n expect(result2.value.tombstone).toBe(true);\n expect(result1.value.lastModified).toEqual(putted2.lastModified);\n expect(result2.value.lastModified).toEqual(putted2.lastModified);\n });\n });\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type {\n Graffiti,\n GraffitiSession,\n JSONSchema4,\n} from \"@graffiti-garden/api\";\nimport { randomString, nextStreamValue, randomPutObject } from \"./utils\";\n\nexport const graffitiDiscoverTests = (\n useGraffiti: () => Pick<Graffiti, \"discover\" | \"put\" | \"delete\" | \"patch\">,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe.concurrent(\"discover\", { timeout: 20000 }, () => {\n it(\"discover nothing\", async () => {\n const graffiti = useGraffiti();\n const iterator = graffiti.discover([], {});\n expect(await iterator.next()).toHaveProperty(\"done\", true);\n });\n\n it(\"discover single\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n\n const putted = await graffiti.put(object, session);\n\n const queryChannels = [randomString(), object.channels[0]];\n const iterator = graffiti.discover(queryChannels, {});\n const value = await nextStreamValue(iterator);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual([object.channels[0]]);\n expect(value.allowed).toBeUndefined();\n expect(value.actor).toEqual(session.actor);\n expect(value.tombstone).toBe(false);\n expect(value.lastModified).toEqual(putted.lastModified);\n const result2 = await iterator.next();\n expect(result2.done).toBe(true);\n });\n\n it(\"discover wrong channel\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n await graffiti.put(object, session);\n const iterator = graffiti.discover([randomString()], {});\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover not allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.allowed = [randomString(), randomString()];\n const putted = await graffiti.put(object, session1);\n\n const iteratorSession1 = graffiti.discover(object.channels, {}, session1);\n const value = await nextStreamValue(iteratorSession1);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual(object.channels);\n expect(value.allowed).toEqual(object.allowed);\n expect(value.actor).toEqual(session1.actor);\n expect(value.tombstone).toBe(false);\n expect(value.lastModified).toEqual(putted.lastModified);\n\n const iteratorSession2 = graffiti.discover(object.channels, {}, session2);\n expect(await iteratorSession2.next()).toHaveProperty(\"done\", true);\n\n const iteratorNoSession = graffiti.discover(object.channels, {});\n expect(await iteratorNoSession.next()).toHaveProperty(\"done\", true);\n });\n\n it(\"discover allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.allowed = [randomString(), session2.actor, randomString()];\n const putted = await graffiti.put(object, session1);\n\n const iteratorSession2 = graffiti.discover(object.channels, {}, session2);\n const value = await nextStreamValue(iteratorSession2);\n expect(value.value).toEqual(object.value);\n expect(value.allowed).toEqual([session2.actor]);\n expect(value.channels).toEqual(object.channels);\n expect(value.actor).toEqual(session1.actor);\n expect(value.tombstone).toBe(false);\n expect(value.lastModified).toEqual(putted.lastModified);\n });\n\n for (const prop of [\"name\", \"actor\", \"lastModified\"] as const) {\n it(`discover for ${prop}`, async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object1 = randomPutObject();\n const putted1 = await graffiti.put(object1, session1);\n\n const object2 = randomPutObject();\n object2.channels = object1.channels;\n // Make sure the lastModified is different for the query\n await new Promise((r) => setTimeout(r, 20));\n const putted2 = await graffiti.put(object2, session2);\n\n const iterator = graffiti.discover(object1.channels, {\n properties: {\n [prop]: {\n enum: [putted1[prop]],\n },\n },\n });\n\n const value = await nextStreamValue(iterator);\n expect(value.name).toEqual(putted1.name);\n expect(value.name).not.toEqual(putted2.name);\n expect(value.value).toEqual(object1.value);\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n }\n\n it(\"discover with lastModified range\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const putted1 = await graffiti.put(object, session);\n // Make sure the lastModified is different\n await new Promise((r) => setTimeout(r, 20));\n const putted2 = await graffiti.put(object, session);\n\n expect(putted1.name).not.toEqual(putted2.name);\n expect(putted1.lastModified).toBeLessThan(putted2.lastModified);\n\n const gtIterator = graffiti.discover([object.channels[0]], {\n properties: {\n lastModified: {\n minimum: putted2.lastModified,\n exclusiveMinimum: true,\n },\n },\n });\n expect(await gtIterator.next()).toHaveProperty(\"done\", true);\n const gtIteratorEpsilon = graffiti.discover([object.channels[0]], {\n properties: {\n lastModified: {\n minimum: putted2.lastModified - 0.1,\n exclusiveMinimum: true,\n },\n },\n });\n const value1 = await nextStreamValue(gtIteratorEpsilon);\n expect(value1.name).toEqual(putted2.name);\n expect(await gtIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n const gteIterator = graffiti.discover(object.channels, {\n properties: {\n value: {},\n lastModified: {\n minimum: putted2.lastModified,\n },\n },\n });\n const value = await nextStreamValue(gteIterator);\n expect(value.name).toEqual(putted2.name);\n expect(await gteIterator.next()).toHaveProperty(\"done\", true);\n const gteIteratorEpsilon = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n minimum: putted2.lastModified + 0.1,\n },\n },\n });\n expect(await gteIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n\n const ltIterator = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified,\n exclusiveMaximum: true,\n },\n },\n });\n expect(await ltIterator.next()).toHaveProperty(\"done\", true);\n\n const ltIteratorEpsilon = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified + 0.1,\n exclusiveMaximum: true,\n },\n },\n });\n const value3 = await nextStreamValue(ltIteratorEpsilon);\n expect(value3.name).toEqual(putted1.name);\n expect(await ltIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n\n const lteIterator = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified,\n },\n },\n });\n const value2 = await nextStreamValue(lteIterator);\n expect(value2.name).toEqual(putted1.name);\n expect(await lteIterator.next()).toHaveProperty(\"done\", true);\n\n const lteIteratorEpsilon = graffiti.discover(object.channels, {\n properties: {\n lastModified: {\n maximum: putted1.lastModified - 0.1,\n },\n },\n });\n expect(await lteIteratorEpsilon.next()).toHaveProperty(\"done\", true);\n });\n\n it(\"discover schema allowed, as and not as owner\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.allowed = [randomString(), session2.actor, randomString()];\n await graffiti.put(object, session1);\n\n const iteratorSession1 = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n minItems: 3,\n // Make sure session2.actor is in the allow list\n not: {\n items: {\n not: {\n enum: [session2.actor],\n },\n },\n },\n },\n },\n },\n session1,\n );\n const value = await nextStreamValue(iteratorSession1);\n expect(value.value).toEqual(object.value);\n await expect(iteratorSession1.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n\n const iteratorSession2BigAllow = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n minItems: 3,\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2BigAllow.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2PeekOther = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n not: {\n items: {\n not: {\n enum: [object.channels[0]],\n },\n },\n },\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2PeekOther.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2SmallAllowPeekSelf = graffiti.discover(\n object.channels,\n {\n properties: {\n allowed: {\n maxItems: 1,\n not: {\n items: {\n not: {\n enum: [session2.actor],\n },\n },\n },\n },\n },\n },\n session2,\n );\n const value2 = await nextStreamValue(iteratorSession2SmallAllowPeekSelf);\n expect(value2.value).toEqual(object.value);\n await expect(\n iteratorSession2SmallAllowPeekSelf.next(),\n ).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover schema channels, as and not as owner\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n const session2 = useSession2();\n\n const object = randomPutObject();\n object.channels = [randomString(), randomString(), randomString()];\n await graffiti.put(object, session1);\n\n const iteratorSession1 = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n channels: {\n minItems: 3,\n // Make sure session2.actor is in the allow list\n not: {\n items: {\n not: {\n enum: [object.channels[1]],\n },\n },\n },\n },\n },\n },\n session1,\n );\n const value = await nextStreamValue(iteratorSession1);\n expect(value.value).toEqual(object.value);\n await expect(iteratorSession1.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n\n const iteratorSession2BigAllow = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n channels: {\n minItems: 3,\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2BigAllow.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2PeekOther = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n channels: {\n not: {\n items: {\n not: {\n enum: [object.channels[1]],\n },\n },\n },\n },\n },\n },\n session2,\n );\n await expect(iteratorSession2PeekOther.next()).resolves.toHaveProperty(\n \"done\",\n true,\n );\n const iteratorSession2SmallAllowPeekSelf = graffiti.discover(\n [object.channels[0], object.channels[2]],\n {\n properties: {\n allowed: {\n maxItems: 2,\n not: {\n items: {\n not: {\n enum: [object.channels[2]],\n },\n },\n },\n },\n },\n },\n session2,\n );\n const value2 = await nextStreamValue(iteratorSession2SmallAllowPeekSelf);\n expect(value2.value).toEqual(object.value);\n await expect(\n iteratorSession2SmallAllowPeekSelf.next(),\n ).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover query for empty allowed\", async () => {\n const graffiti = useGraffiti();\n const session1 = useSession1();\n\n const publicO = randomPutObject();\n\n const publicSchema = {\n not: {\n required: [\"allowed\"],\n },\n } satisfies JSONSchema4;\n\n await graffiti.put(publicO, session1);\n const iterator = graffiti.discover(\n publicO.channels,\n publicSchema,\n session1,\n );\n const value = await nextStreamValue(iterator);\n expect(value.value).toEqual(publicO.value);\n expect(value.allowed).toBeUndefined();\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n\n const restricted = randomPutObject();\n restricted.allowed = [];\n await graffiti.put(restricted, session1);\n const iterator2 = graffiti.discover(\n restricted.channels,\n publicSchema,\n session1,\n );\n await expect(iterator2.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover query for values\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object1 = randomPutObject();\n object1.value = { test: randomString() };\n await graffiti.put(object1, session);\n\n const object2 = randomPutObject();\n object2.channels = object1.channels;\n object2.value = { test: randomString(), something: randomString() };\n await graffiti.put(object2, session);\n\n const object3 = randomPutObject();\n object3.channels = object1.channels;\n object3.value = { other: randomString(), something: randomString() };\n await graffiti.put(object3, session);\n\n const counts = new Map<string, number>();\n for (const property of [\"test\", \"something\", \"other\"] as const) {\n let count = 0;\n for await (const result of graffiti.discover(object1.channels, {\n properties: {\n value: {\n required: [property],\n },\n },\n })) {\n assert(!result.error, \"result has error\");\n if (property in result.value.value) {\n count++;\n }\n }\n counts.set(property, count);\n }\n\n expect(counts.get(\"test\")).toBe(2);\n expect(counts.get(\"something\")).toBe(2);\n expect(counts.get(\"other\")).toBe(1);\n });\n\n it(\"discover for deleted content\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n const deleted = await graffiti.delete(putted, session);\n\n const iterator = graffiti.discover(object.channels, {});\n const value = await nextStreamValue(iterator);\n expect(value.tombstone).toBe(true);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual(object.channels);\n expect(value.actor).toEqual(session.actor);\n expect(value.lastModified).toEqual(deleted.lastModified);\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"discover for replaced channels\", async () => {\n // Do this a bunch to check for concurrency issues\n for (let i = 0; i < 10; i++) {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object1 = randomPutObject();\n const putted = await graffiti.put(object1, session);\n const object2 = randomPutObject();\n const replaced = await graffiti.put(\n {\n ...putted,\n ...object2,\n },\n session,\n );\n\n const iterator1 = graffiti.discover(object1.channels, {});\n const value1 = await nextStreamValue(iterator1);\n await expect(iterator1.next()).resolves.toHaveProperty(\"done\", true);\n\n const iterator2 = graffiti.discover(object2.channels, {});\n const value2 = await nextStreamValue(iterator2);\n await expect(iterator2.next()).resolves.toHaveProperty(\"done\", true);\n\n // If they have the same timestamp, except\n // only one to have a tombstone\n if (putted.lastModified === replaced.lastModified) {\n expect(value1.tombstone || value2.tombstone).toBe(true);\n expect(value1.tombstone && value2.tombstone).toBe(false);\n } else {\n expect(value1.tombstone).toBe(true);\n expect(value1.value).toEqual(object1.value);\n expect(value1.channels).toEqual(object1.channels);\n expect(value1.lastModified).toEqual(replaced.lastModified);\n\n expect(value2.tombstone).toBe(false);\n expect(value2.value).toEqual(object2.value);\n expect(value2.channels).toEqual(object2.channels);\n expect(value2.lastModified).toEqual(replaced.lastModified);\n }\n }\n });\n\n it(\"discover for patched allowed\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n const object = randomPutObject();\n const putted = await graffiti.put(object, session);\n await graffiti.patch(\n {\n allowed: [{ op: \"add\", path: \"\", value: [] }],\n },\n putted,\n session,\n );\n const iterator = graffiti.discover(object.channels, {});\n const value = await nextStreamValue(iterator);\n expect(value.tombstone).toBe(true);\n expect(value.value).toEqual(object.value);\n expect(value.channels).toEqual(object.channels);\n expect(value.allowed).toBeUndefined();\n await expect(iterator.next()).resolves.toHaveProperty(\"done\", true);\n });\n\n it(\"put concurrently and discover one\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n object.name = randomString();\n\n const putPromises = Array(100)\n .fill(0)\n .map(() => graffiti.put(object, session));\n await Promise.all(putPromises);\n\n const iterator = graffiti.discover(object.channels, {});\n let tombstoneCount = 0;\n let valueCount = 0;\n for await (const result of iterator) {\n assert(!result.error, \"result has error\");\n if (result.value.tombstone) {\n tombstoneCount++;\n } else {\n valueCount++;\n }\n }\n expect(tombstoneCount).toBe(99);\n expect(valueCount).toBe(1);\n });\n });\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type { Graffiti, GraffitiSession } from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiOrphanTests = (\n useGraffiti: () => Pick<\n Graffiti,\n \"recoverOrphans\" | \"put\" | \"delete\" | \"patch\"\n >,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe(\"recoverOrphans\", () => {\n it(\"list orphans\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const existingOrphans: string[] = [];\n const orphanIterator1 = graffiti.recoverOrphans({}, session);\n for await (const orphan of orphanIterator1) {\n if (orphan.error) continue;\n existingOrphans.push(orphan.value.name);\n }\n\n const object = randomPutObject();\n object.channels = [];\n const putted = await graffiti.put(object, session);\n const orphanIterator2 = graffiti.recoverOrphans({}, session);\n let numResults = 0;\n for await (const orphan of orphanIterator2) {\n if (orphan.error) continue;\n if (orphan.value.name === putted.name) {\n numResults++;\n expect(orphan.value.source).toBe(putted.source);\n expect(orphan.value.lastModified).toBe(putted.lastModified);\n }\n }\n expect(numResults).toBe(1);\n });\n\n it(\"replaced orphan, no longer\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const object = randomPutObject();\n object.channels = [];\n const putOrphan = await graffiti.put(object, session);\n\n // Wait for the put to be processed\n await new Promise((resolve) => setTimeout(resolve, 10));\n\n const putNotOrphan = await graffiti.put(\n {\n ...putOrphan,\n ...object,\n channels: [randomString()],\n },\n session,\n );\n expect(putNotOrphan.name).toBe(putOrphan.name);\n expect(putNotOrphan.lastModified).toBeGreaterThan(putOrphan.lastModified);\n\n const orphanIterator = graffiti.recoverOrphans({}, session);\n let numResults = 0;\n for await (const orphan of orphanIterator) {\n if (orphan.error) continue;\n if (orphan.value.name === putOrphan.name) {\n numResults++;\n expect(orphan.value.tombstone).toBe(true);\n expect(orphan.value.lastModified).toBe(putNotOrphan.lastModified);\n expect(orphan.value.channels).toEqual([]);\n }\n }\n expect(numResults).toBe(1);\n });\n });\n};\n", "import { it, expect, describe, assert } from \"vitest\";\nimport type { Graffiti, GraffitiSession } from \"@graffiti-garden/api\";\nimport { randomPutObject, randomString } from \"./utils\";\n\nexport const graffitiChannelStatsTests = (\n useGraffiti: () => Pick<\n Graffiti,\n \"channelStats\" | \"put\" | \"delete\" | \"patch\"\n >,\n useSession1: () => GraffitiSession,\n useSession2: () => GraffitiSession,\n) => {\n describe(\"channel stats\", () => {\n it(\"list channels\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const existingChannels: Map<string, number> = new Map();\n const channelIterator1 = graffiti.channelStats(session);\n for await (const channel of channelIterator1) {\n if (channel.error) continue;\n existingChannels.set(channel.value.channel, channel.value.count);\n }\n\n const channels = [randomString(), randomString(), randomString()];\n\n // Add one value to channels[0],\n // two values to both channels[0] and channels[1],\n // three values to all channels\n // one value to channels[2]\n for (let i = 0; i < 3; i++) {\n for (let j = 0; j < i + 1; j++) {\n await graffiti.put(\n {\n value: {\n index: j,\n },\n channels: channels.slice(0, i + 1),\n },\n session,\n );\n }\n }\n await graffiti.put(\n { value: { index: 3 }, channels: [channels[2]] },\n session,\n );\n\n const channelIterator2 = graffiti.channelStats(session);\n let newChannels: Map<string, number> = new Map();\n for await (const channel of channelIterator2) {\n if (channel.error) continue;\n newChannels.set(channel.value.channel, channel.value.count);\n }\n // Filter out existing channels\n newChannels = new Map(\n Array.from(newChannels).filter(\n ([channel, count]) => !existingChannels.has(channel),\n ),\n );\n expect(newChannels.size).toBe(3);\n expect(newChannels.get(channels[0])).toBe(6);\n expect(newChannels.get(channels[1])).toBe(5);\n expect(newChannels.get(channels[2])).toBe(4);\n });\n\n it(\"list channels with deleted channel\", async () => {\n const graffiti = useGraffiti();\n const session = useSession1();\n\n const channels = [randomString(), randomString(), randomString()];\n\n // Add an item with two channels\n const before = await graffiti.put(\n {\n value: { index: 2 },\n channels: channels.slice(1),\n },\n session,\n );\n\n // Add an item with all channels\n const first = await graffiti.put(\n { value: { index: 0 }, channels },\n session,\n );\n // But then delete it\n await graffiti.delete(first, session);\n\n // Create a new object with only one channel\n const second = await graffiti.put(\n {\n value: { index: 1 },\n channels: channels.slice(2),\n },\n session,\n );\n\n const channelIterator = graffiti.channelStats(session);\n\n let got1 = 0;\n let got2 = 0;\n for await (const result of channelIterator) {\n if (result.error) continue;\n const { channel, count, lastModified } = result.value;\n assert(\n channel !== channels[0],\n \"There should not be an object in channel[0]\",\n );\n if (channel === channels[1]) {\n expect(count).toBe(1);\n expect(lastModified).toBe(before.lastModified);\n got1++;\n } else if (channel === channels[2]) {\n expect(count).toBe(2);\n expect(lastModified).toBe(second.lastModified);\n got2++;\n }\n }\n expect(got1).toBe(1);\n expect(got2).toBe(1);\n });\n });\n};\n"],
5
+ "mappings": "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",
6
+ "names": ["it", "expect", "describe", "describe", "it", "expect", "it", "expect", "describe", "assert", "describe", "it", "expect", "assert", "it", "expect", "describe", "assert", "describe", "it", "expect", "assert", "it", "expect", "describe", "describe", "it", "expect", "it", "expect", "describe", "assert", "describe", "it", "expect", "assert"]
7
7
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@graffiti-garden/api",
3
- "version": "0.2.7",
3
+ "version": "0.2.8",
4
4
  "description": "The heart of Graffiti",
5
5
  "types": "./dist/src/index.d.ts",
6
6
  "module": "./dist/index.mjs",
package/tests/crud.ts CHANGED
@@ -19,7 +19,7 @@ export const graffitiCRUDTests = (
19
19
  useSession1: () => GraffitiSession,
20
20
  useSession2: () => GraffitiSession,
21
21
  ) => {
22
- describe(
22
+ describe.concurrent(
23
23
  "CRUD",
24
24
  {
25
25
  timeout: 20000,
@@ -63,7 +63,7 @@ export const graffitiCRUDTests = (
63
63
  expect(beforeReplaced.name).toEqual(previous.name);
64
64
  expect(beforeReplaced.actor).toEqual(previous.actor);
65
65
  expect(beforeReplaced.source).toEqual(previous.source);
66
- expect(beforeReplaced.lastModified).toBeGreaterThan(
66
+ expect(beforeReplaced.lastModified).toBeGreaterThanOrEqual(
67
67
  gotten.lastModified,
68
68
  );
69
69
 
@@ -77,7 +77,7 @@ export const graffitiCRUDTests = (
77
77
  const beforeDeleted = await graffiti.delete(afterReplaced, session);
78
78
  expect(beforeDeleted.tombstone).toEqual(true);
79
79
  expect(beforeDeleted.value).toEqual(newValue);
80
- expect(beforeDeleted.lastModified).toBeGreaterThan(
80
+ expect(beforeDeleted.lastModified).toBeGreaterThanOrEqual(
81
81
  beforeReplaced.lastModified,
82
82
  );
83
83
 
@@ -290,6 +290,9 @@ export const graffitiCRUDTests = (
290
290
  };
291
291
  const putted = await graffiti.put({ value, channels: [] }, session);
292
292
 
293
+ // Wait just a bit to make sure the lastModified is different
294
+ await new Promise((resolve) => setTimeout(resolve, 10));
295
+
293
296
  const patch: GraffitiPatch = {
294
297
  value: [
295
298
  { op: "replace", path: "/something", value: "goodbye, world~ :c" },
@@ -298,6 +301,7 @@ export const graffitiCRUDTests = (
298
301
  const beforePatched = await graffiti.patch(patch, putted, session);
299
302
  expect(beforePatched.value).toEqual(value);
300
303
  expect(beforePatched.tombstone).toBe(true);
304
+ expect(beforePatched.lastModified).toBeGreaterThan(putted.lastModified);
301
305
 
302
306
  const gotten = await graffiti.get(putted, {});
303
307
  expect(gotten.value).toEqual({
package/tests/discover.ts CHANGED
@@ -11,7 +11,7 @@ export const graffitiDiscoverTests = (
11
11
  useSession1: () => GraffitiSession,
12
12
  useSession2: () => GraffitiSession,
13
13
  ) => {
14
- describe("discover", { timeout: 20000 }, () => {
14
+ describe.concurrent("discover", { timeout: 20000 }, () => {
15
15
  it("discover nothing", async () => {
16
16
  const graffiti = useGraffiti();
17
17
  const iterator = graffiti.discover([], {});
package/tests/location.ts CHANGED
@@ -6,7 +6,7 @@ import { randomString } from "./utils";
6
6
  export const graffitiLocationTests = (
7
7
  useGraffiti: () => Pick<Graffiti, "locationToUri" | "uriToLocation">,
8
8
  ) => {
9
- describe("URI and location conversion", () => {
9
+ describe.concurrent("URI and location conversion", () => {
10
10
  it("location to uri and back", async () => {
11
11
  const graffiti = useGraffiti();
12
12
  const location = {
package/tests/orphans.ts CHANGED
@@ -46,6 +46,9 @@ export const graffitiOrphanTests = (
46
46
  object.channels = [];
47
47
  const putOrphan = await graffiti.put(object, session);
48
48
 
49
+ // Wait for the put to be processed
50
+ await new Promise((resolve) => setTimeout(resolve, 10));
51
+
49
52
  const putNotOrphan = await graffiti.put(
50
53
  {
51
54
  ...putOrphan,
@@ -55,6 +58,7 @@ export const graffitiOrphanTests = (
55
58
  session,
56
59
  );
57
60
  expect(putNotOrphan.name).toBe(putOrphan.name);
61
+ expect(putNotOrphan.lastModified).toBeGreaterThan(putOrphan.lastModified);
58
62
 
59
63
  const orphanIterator = graffiti.recoverOrphans({}, session);
60
64
  let numResults = 0;
@@ -7,7 +7,7 @@ export const graffitiSynchronizeTests = (
7
7
  useSession1: () => GraffitiSession,
8
8
  useSession2: () => GraffitiSession,
9
9
  ) => {
10
- describe("synchronizeDiscover", () => {
10
+ describe.concurrent("synchronizeDiscover", () => {
11
11
  it("get", async () => {
12
12
  const graffiti1 = useGraffiti();
13
13
  const session = useSession1();