@graffiti-garden/implementation-local 0.6.2 → 0.6.4

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": ["../../src/database.ts"],
4
- "sourcesContent": ["import type {\n Graffiti,\n GraffitiObjectBase,\n GraffitiObjectUrl,\n JSONSchema,\n GraffitiSession,\n GraffitiObjectStreamContinue,\n GraffitiObjectStreamContinueEntry,\n} from \"@graffiti-garden/api\";\nimport {\n GraffitiErrorNotFound,\n GraffitiErrorSchemaMismatch,\n GraffitiErrorForbidden,\n GraffitiErrorPatchError,\n} from \"@graffiti-garden/api\";\nimport {\n randomBase64,\n applyGraffitiPatch,\n maskGraffitiObject,\n isActorAllowedGraffitiObject,\n compileGraffitiObjectSchema,\n unpackObjectUrl,\n} from \"./utilities.js\";\nimport type Ajv from \"ajv\";\nimport type { applyPatch } from \"fast-json-patch\";\n\n/**\n * Constructor options for the GraffitiPoubchDB class.\n */\nexport interface GraffitiLocalOptions {\n /**\n * Options to pass to the PouchDB constructor.\n * Defaults to `{ name: \"graffitiDb\" }`.\n *\n * See the [PouchDB documentation](https://pouchdb.com/api.html#create_database)\n * for available options.\n */\n pouchDBOptions?: PouchDB.Configuration.DatabaseConfiguration;\n /**\n * Includes the scheme and other information (possibly domain name)\n * to prefix prefixes all URLs put in the system. Defaults to `graffiti:local`.\n */\n origin?: string;\n /**\n * Whether to allow putting objects at arbtirary URLs, i.e.\n * URLs that are *not* prefixed with the origin or not generated\n * by the system. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettingArbitraryUrls?: boolean;\n /**\n * Whether to allow the user to set the lastModified field\n * when putting objects. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettinngLastModified?: boolean;\n /**\n * An optional Ajv instance to use for schema validation.\n * If not provided, an internal instance will be created.\n */\n ajv?: Ajv;\n}\n\nconst DEFAULT_ORIGIN = \"graffiti:local:\";\nconst LAST_MODIFIED_BUFFER = 60000;\n\ntype GraffitiObjectWithTombstone = GraffitiObjectBase & { tombstone: boolean };\n\n/**\n * An implementation of only the database operations of the\n * GraffitiAPI without synchronization or session management.\n */\nexport class GraffitiLocalDatabase\n implements Omit<Graffiti, \"login\" | \"logout\" | \"sessionEvents\">\n{\n protected db_:\n | Promise<PouchDB.Database<GraffitiObjectWithTombstone>>\n | undefined;\n protected applyPatch_: Promise<typeof applyPatch> | undefined;\n protected ajv_: Promise<Ajv> | undefined;\n protected readonly options: GraffitiLocalOptions;\n protected readonly origin: string;\n\n get db() {\n if (!this.db_) {\n this.db_ = (async () => {\n const { default: PouchDB } = await import(\"pouchdb\");\n const pouchDbOptions = {\n name: \"graffitiDb\",\n ...this.options.pouchDBOptions,\n };\n const db = new PouchDB<GraffitiObjectWithTombstone>(\n pouchDbOptions.name,\n pouchDbOptions,\n );\n await db\n //@ts-ignore\n .put({\n _id: \"_design/indexes\",\n views: {\n objectsPerChannelAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(channel) + \"/\" + paddedLastModified;\n //@ts-ignore\n emit(id);\n });\n }.toString(),\n },\n orphansPerActorAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.channels.length === 0) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n paddedLastModified;\n //@ts-ignore\n emit(id);\n }\n }.toString(),\n },\n channelStatsPerActor: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.tombstone) return;\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n encodeURIComponent(channel);\n //@ts-ignore\n emit(id, object.lastModified);\n });\n }.toString(),\n reduce: \"_stats\",\n },\n },\n })\n //@ts-ignore\n .catch((error) => {\n if (\n error &&\n typeof error === \"object\" &&\n \"name\" in error &&\n error.name === \"conflict\"\n ) {\n // Design document already exists\n return;\n } else {\n throw error;\n }\n });\n return db;\n })();\n }\n return this.db_;\n }\n\n protected get applyPatch() {\n if (!this.applyPatch_) {\n this.applyPatch_ = (async () => {\n const imported = await import(\"fast-json-patch\");\n return imported.applyPatch || imported.default.applyPatch;\n })();\n }\n return this.applyPatch_;\n }\n\n protected get ajv() {\n if (!this.ajv_) {\n this.ajv_ = this.options.ajv\n ? Promise.resolve(this.options.ajv)\n : (async () => {\n const { default: Ajv } = await import(\"ajv\");\n return new Ajv({ strict: false });\n })();\n }\n return this.ajv_;\n }\n\n protected extractGraffitiObject(\n object: GraffitiObjectWithTombstone,\n ): GraffitiObjectBase {\n const { value, channels, allowed, url, actor, lastModified } = object;\n return {\n value,\n channels,\n allowed,\n url,\n actor,\n lastModified,\n };\n }\n\n constructor(options?: GraffitiLocalOptions) {\n this.options = options ?? {};\n this.origin = this.options.origin ?? DEFAULT_ORIGIN;\n if (!this.origin.endsWith(\":\") && !this.origin.endsWith(\"/\")) {\n this.origin += \"/\";\n }\n }\n\n protected async allDocsAtLocation(objectUrl: string | GraffitiObjectUrl) {\n const url = unpackObjectUrl(objectUrl) + \"/\";\n const results = await (\n await this.db\n ).allDocs({\n startkey: url,\n endkey: url + \"\\uffff\", // \\uffff is the last unicode character\n include_docs: true,\n });\n const docs = results.rows\n .map((row) => row.doc)\n // Remove undefined docs\n .reduce<\n PouchDB.Core.ExistingDocument<\n GraffitiObjectWithTombstone & PouchDB.Core.AllDocsMeta\n >[]\n >((acc, doc) => {\n if (doc) acc.push(doc);\n return acc;\n }, []);\n return docs;\n }\n\n protected docId(objectUrl: GraffitiObjectUrl) {\n return objectUrl.url + \"/\" + randomBase64();\n }\n\n get: Graffiti[\"get\"] = async (...args) => {\n const [urlObject, schema, session] = args;\n\n const docsAll = await this.allDocsAtLocation(urlObject);\n\n // Filter out ones not allowed\n const docs = docsAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, session),\n );\n if (!docs.length)\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n\n // Get the most recent document\n const doc = docs.reduce((a, b) =>\n a.lastModified > b.lastModified ||\n (a.lastModified === b.lastModified && !a.tombstone && b.tombstone)\n ? a\n : b,\n );\n\n if (doc.tombstone) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n }\n\n const object = this.extractGraffitiObject(doc);\n\n // Mask out the allowed list and channels\n // if the user is not the owner\n maskGraffitiObject(object, [], session);\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n if (!validate(object)) {\n throw new GraffitiErrorSchemaMismatch();\n }\n return object;\n };\n\n /**\n * Deletes all docs at a particular location.\n * If the `keepLatest` flag is set to true,\n * the doc with the most recent timestamp will be\n * spared. If there are multiple docs with the same\n * timestamp, the one with the highest `_id` will be\n * spared.\n */\n protected async deleteAtLocation(\n url: GraffitiObjectUrl | string,\n options: {\n keepLatest?: boolean;\n session?: GraffitiSession;\n } = {\n keepLatest: false,\n },\n ) {\n const docsAtLocationAll = await this.allDocsAtLocation(url);\n const docsAtLocationAllowed = options.session\n ? docsAtLocationAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, options.session),\n )\n : docsAtLocationAll;\n if (!docsAtLocationAllowed.length) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to delete either does not exist or you are not allowed to see it\",\n );\n } else if (\n options.session &&\n docsAtLocationAllowed.some((doc) => doc.actor !== options.session?.actor)\n ) {\n throw new GraffitiErrorForbidden(\n \"You cannot delete an object owned by another actor\",\n );\n }\n const docsAtLocation = docsAtLocationAllowed.filter(\n (doc) => !doc.tombstone,\n );\n if (!docsAtLocation.length) return undefined;\n\n // Get the most recent lastModified timestamp.\n const latestModified = docsAtLocation\n .map((doc) => doc.lastModified)\n .reduce((a, b) => (a > b ? a : b));\n\n // Delete all old docs\n const docsToDelete = docsAtLocation.filter(\n (doc) => !options.keepLatest || doc.lastModified < latestModified,\n );\n\n // For docs with the same timestamp,\n // keep the one with the highest _id\n // to break concurrency ties\n const concurrentDocsAll = docsAtLocation.filter(\n (doc) => options.keepLatest && doc.lastModified === latestModified,\n );\n if (concurrentDocsAll.length) {\n const keepDocId = concurrentDocsAll\n .map((doc) => doc._id)\n .reduce((a, b) => (a > b ? a : b));\n const concurrentDocsToDelete = concurrentDocsAll.filter(\n (doc) => doc._id !== keepDocId,\n );\n docsToDelete.push(...concurrentDocsToDelete);\n }\n\n const lastModified = options.keepLatest\n ? latestModified\n : new Date().getTime();\n\n const deleteResults = await (\n await this.db\n ).bulkDocs<GraffitiObjectBase>(\n docsToDelete.map((doc) => ({\n ...doc,\n tombstone: true,\n lastModified,\n })),\n );\n\n // Get one of the docs that was deleted\n let deletedObject: GraffitiObjectBase | undefined = undefined;\n for (const resultOrError of deleteResults) {\n if (\"ok\" in resultOrError) {\n const { id } = resultOrError;\n const deletedDoc = docsToDelete.find((doc) => doc._id === id);\n if (deletedDoc) {\n deletedObject = {\n ...this.extractGraffitiObject(deletedDoc),\n lastModified,\n };\n break;\n }\n }\n }\n\n return deletedObject;\n }\n\n delete: Graffiti[\"delete\"] = async (...args) => {\n const [url, session] = args;\n const deletedObject = await this.deleteAtLocation(url, {\n session,\n });\n if (!deletedObject) {\n throw new GraffitiErrorNotFound(\"The object has already been deleted\");\n }\n return deletedObject;\n };\n\n put: Graffiti[\"put\"] = async (...args) => {\n const [objectPartial, session] = args;\n if (objectPartial.actor && objectPartial.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot put an object with a different actor than the session actor\",\n );\n }\n\n if (objectPartial.url) {\n let oldObject: GraffitiObjectBase | undefined;\n try {\n oldObject = await this.get(objectPartial.url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n if (!this.options.allowSettingArbitraryUrls) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to replace does not exist or you are not allowed to see it\",\n );\n }\n } else {\n throw e;\n }\n }\n if (oldObject?.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to replace is owned by another actor\",\n );\n }\n }\n\n const lastModified =\n ((this.options.allowSettinngLastModified ?? false) &&\n objectPartial.lastModified) ||\n new Date().getTime();\n\n const object: GraffitiObjectWithTombstone = {\n value: objectPartial.value,\n channels: objectPartial.channels,\n allowed: objectPartial.allowed,\n url: objectPartial.url ?? this.origin + randomBase64(),\n actor: session.actor,\n tombstone: false,\n lastModified,\n };\n\n await (\n await this.db\n ).put({\n _id: this.docId(object),\n ...object,\n });\n\n // Delete the old object\n const previousObject = await this.deleteAtLocation(object, {\n keepLatest: true,\n });\n if (previousObject) {\n return previousObject;\n } else {\n return {\n ...object,\n value: {},\n channels: [],\n allowed: [],\n tombstone: true,\n };\n }\n };\n\n patch: Graffiti[\"patch\"] = async (...args) => {\n const [patch, url, session] = args;\n let originalObject: GraffitiObjectBase;\n try {\n originalObject = await this.get(url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to patch does not exist or you are not allowed to see it\",\n );\n } else {\n throw e;\n }\n }\n if (originalObject.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to patch is owned by another actor\",\n );\n }\n\n // Patch it outside of the database\n const patchObject: GraffitiObjectBase = { ...originalObject };\n for (const prop of [\"value\", \"channels\", \"allowed\"] as const) {\n applyGraffitiPatch(await this.applyPatch, prop, patch, patchObject);\n }\n\n // Make sure the value is an object\n if (\n typeof patchObject.value !== \"object\" ||\n Array.isArray(patchObject.value) ||\n !patchObject.value\n ) {\n throw new GraffitiErrorPatchError(\"value is no longer an object\");\n }\n\n // Make sure the channels are an array of strings\n if (\n !Array.isArray(patchObject.channels) ||\n !patchObject.channels.every((channel) => typeof channel === \"string\")\n ) {\n throw new GraffitiErrorPatchError(\n \"channels are no longer an array of strings\",\n );\n }\n\n // Make sure the allowed list is an array of strings or undefined\n if (\n patchObject.allowed &&\n (!Array.isArray(patchObject.allowed) ||\n !patchObject.allowed.every((allowed) => typeof allowed === \"string\"))\n ) {\n throw new GraffitiErrorPatchError(\n \"allowed list is not an array of strings\",\n );\n }\n\n patchObject.lastModified = new Date().getTime();\n await (\n await this.db\n ).put({\n ...patchObject,\n tombstone: false,\n _id: this.docId(patchObject),\n });\n\n // Delete the old object\n await this.deleteAtLocation(patchObject, {\n keepLatest: true,\n });\n\n return {\n ...originalObject,\n lastModified: patchObject.lastModified,\n };\n };\n\n protected queryLastModifiedSuffixes(\n schema: JSONSchema,\n lastModified?: number,\n ) {\n // Use the index for queries over ranges of lastModified\n let startKeySuffix = \"\";\n let endKeySuffix = \"\\uffff\";\n if (\n typeof schema === \"object\" &&\n schema.properties?.lastModified &&\n typeof schema.properties.lastModified === \"object\"\n ) {\n const lastModifiedSchema = schema.properties.lastModified;\n\n const minimum =\n lastModified && lastModifiedSchema.minimum\n ? Math.max(lastModified, lastModifiedSchema.minimum)\n : (lastModified ?? lastModifiedSchema.minimum);\n const exclusiveMinimum = lastModifiedSchema.exclusiveMinimum;\n\n let intMinimum: number | undefined;\n if (exclusiveMinimum !== undefined) {\n intMinimum = Math.ceil(exclusiveMinimum);\n intMinimum === exclusiveMinimum && intMinimum++;\n } else if (minimum !== undefined) {\n intMinimum = Math.ceil(minimum);\n }\n\n if (intMinimum !== undefined) {\n startKeySuffix = intMinimum.toString().padStart(15, \"0\");\n }\n\n const maximum = lastModifiedSchema.maximum;\n const exclusiveMaximum = lastModifiedSchema.exclusiveMaximum;\n\n let intMaximum: number | undefined;\n if (exclusiveMaximum !== undefined) {\n intMaximum = Math.floor(exclusiveMaximum);\n intMaximum === exclusiveMaximum && intMaximum--;\n } else if (maximum !== undefined) {\n intMaximum = Math.floor(maximum);\n }\n\n if (intMaximum !== undefined) {\n endKeySuffix = intMaximum.toString().padStart(15, \"0\");\n }\n }\n return {\n startKeySuffix,\n endKeySuffix,\n };\n }\n\n protected async *streamObjects<Schema extends JSONSchema>(\n index: string,\n startkey: string,\n endkey: string,\n validate: ReturnType<typeof compileGraffitiObjectSchema<Schema>>,\n session: GraffitiSession | undefined | null,\n ifModifiedSince: number | undefined,\n channels?: string[],\n processedIds?: Set<string>,\n ): AsyncGenerator<GraffitiObjectStreamContinueEntry<Schema>> {\n const showTombstones = ifModifiedSince !== undefined;\n\n const result = await (\n await this.db\n ).query<GraffitiObjectWithTombstone>(index, {\n startkey,\n endkey,\n include_docs: true,\n });\n\n for (const row of result.rows) {\n const doc = row.doc;\n if (!doc) continue;\n\n if (processedIds?.has(doc._id)) continue;\n processedIds?.add(doc._id);\n\n if (!showTombstones && doc.tombstone) continue;\n\n const object = this.extractGraffitiObject(doc);\n\n if (channels) {\n if (!isActorAllowedGraffitiObject(object, session)) continue;\n maskGraffitiObject(object, channels, session);\n }\n\n if (!validate(object)) continue;\n\n yield doc.tombstone\n ? {\n tombstone: true,\n object: {\n url: object.url,\n lastModified: object.lastModified,\n },\n }\n : { object };\n }\n }\n\n protected async *discoverMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n const [channels, schema, session] = args;\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n\n const processedIds = new Set<string>();\n\n const startTime = new Date().getTime();\n\n for (const channel of channels) {\n const keyPrefix = encodeURIComponent(channel) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/objectsPerChannelAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n channels,\n processedIds,\n );\n\n for await (const result of iterator) yield result;\n }\n\n // Subtract a minute to make sure we don't miss any objects\n return startTime - LAST_MODIFIED_BUFFER;\n }\n\n protected async *recoverOrphansMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n const [schema, session] = args;\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n\n const startTime = new Date().getTime();\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/orphansPerActorAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n );\n\n for await (const result of iterator) yield result;\n\n return startTime - LAST_MODIFIED_BUFFER;\n }\n\n protected discoverCursor(\n args: Parameters<typeof Graffiti.prototype.discover<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"discover:\" +\n JSON.stringify({\n channels: args[0],\n schema: args[1],\n actor: args[2]?.actor,\n ifModifiedSince: ifModifiedSince,\n })\n );\n }\n\n protected async *discoverContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.discoverMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () => this.discoverContinue<Schema>(args, ifModifiedSince),\n cursor: this.discoverCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n discover: Graffiti[\"discover\"] = (...args) => {\n const iterator = this.discoverMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.discoverContinue<(typeof args)[1]>(args, result.value),\n cursor: this_.discoverCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n protected recoverOrphansCursor(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"orphans:\" +\n JSON.stringify({\n schema: args[0],\n actor: args[1]?.actor,\n ifModifiedSince,\n })\n );\n }\n\n protected async *recoverOrphansContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.recoverOrphansMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () =>\n this.recoverOrphansContinue<Schema>(args, ifModifiedSince),\n cursor: this.recoverOrphansCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n recoverOrphans: Graffiti[\"recoverOrphans\"] = (...args) => {\n const iterator = this.recoverOrphansMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.recoverOrphansContinue<(typeof args)[0]>(\n args,\n result.value,\n ),\n cursor: this_.recoverOrphansCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n channelStats: Graffiti[\"channelStats\"] = (session) => {\n const this_ = this;\n return (async function* () {\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const result = await (\n await this_.db\n ).query(\"indexes/channelStatsPerActor\", {\n startkey: keyPrefix,\n endkey: keyPrefix + \"\\uffff\",\n reduce: true,\n group: true,\n });\n for (const row of result.rows) {\n const channelEncoded = row.key.split(\"/\")[1];\n if (typeof channelEncoded !== \"string\") continue;\n const { count, max: lastModified } = row.value;\n if (typeof count !== \"number\" || typeof lastModified !== \"number\")\n continue;\n yield {\n value: {\n channel: decodeURIComponent(channelEncoded),\n count,\n lastModified,\n },\n };\n }\n })();\n };\n\n continueObjectStream: Graffiti[\"continueObjectStream\"] = (\n cursor,\n session,\n ) => {\n if (cursor.startsWith(\"discover:\")) {\n const { channels, schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"discover:\".length),\n );\n if (actor && actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.discoverContinue<{}>(\n [channels, schema, session],\n ifModifiedSince,\n );\n } else if (cursor.startsWith(\"orphans:\")) {\n const { schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"orphans:\".length),\n );\n if (!session || actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.recoverOrphansContinue<{}>(\n [schema, session],\n ifModifiedSince,\n );\n } else {\n throw new GraffitiErrorNotFound(\"Cursor not found\");\n }\n };\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import type {\n Graffiti,\n GraffitiObjectBase,\n GraffitiObjectUrl,\n JSONSchema,\n GraffitiSession,\n GraffitiObjectStreamContinue,\n GraffitiObjectStreamContinueEntry,\n} from \"@graffiti-garden/api\";\nimport {\n GraffitiErrorNotFound,\n GraffitiErrorSchemaMismatch,\n GraffitiErrorForbidden,\n GraffitiErrorPatchError,\n} from \"@graffiti-garden/api\";\nimport {\n randomBase64,\n applyGraffitiPatch,\n maskGraffitiObject,\n isActorAllowedGraffitiObject,\n compileGraffitiObjectSchema,\n unpackObjectUrl,\n} from \"./utilities.js\";\nimport type Ajv from \"ajv\";\nimport type { applyPatch } from \"fast-json-patch\";\n\n/**\n * Constructor options for the GraffitiPoubchDB class.\n */\nexport interface GraffitiLocalOptions {\n /**\n * Options to pass to the PouchDB constructor.\n * Defaults to `{ name: \"graffitiDb\" }`.\n *\n * See the [PouchDB documentation](https://pouchdb.com/api.html#create_database)\n * for available options.\n */\n pouchDBOptions?: PouchDB.Configuration.DatabaseConfiguration;\n /**\n * Includes the scheme and other information (possibly domain name)\n * to prefix prefixes all URLs put in the system. Defaults to `graffiti:local`.\n */\n origin?: string;\n /**\n * Whether to allow putting objects at arbtirary URLs, i.e.\n * URLs that are *not* prefixed with the origin or not generated\n * by the system. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettingArbitraryUrls?: boolean;\n /**\n * Whether to allow the user to set the lastModified field\n * when putting objects. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettinngLastModified?: boolean;\n /**\n * An optional Ajv instance to use for schema validation.\n * If not provided, an internal instance will be created.\n */\n ajv?: Ajv;\n /**\n * Wait at least this long (in milliseconds) before continuing a stream.\n * A basic form of rate limiting. Defaults to 1 seconds.\n */\n continueBuffer?: number;\n}\n\nconst DEFAULT_ORIGIN = \"graffiti:local:\";\n\n// During stream continuations, return objects\n// that have possibly already been seen over this window\nconst LAST_MODIFIED_BUFFER = 60000;\n\ntype GraffitiObjectWithTombstone = GraffitiObjectBase & { tombstone: boolean };\n\n/**\n * An implementation of only the database operations of the\n * GraffitiAPI without synchronization or session management.\n */\nexport class GraffitiLocalDatabase\n implements Omit<Graffiti, \"login\" | \"logout\" | \"sessionEvents\">\n{\n protected db_:\n | Promise<PouchDB.Database<GraffitiObjectWithTombstone>>\n | undefined;\n protected applyPatch_: Promise<typeof applyPatch> | undefined;\n protected ajv_: Promise<Ajv> | undefined;\n protected readonly options: GraffitiLocalOptions;\n protected readonly origin: string;\n\n get db() {\n if (!this.db_) {\n this.db_ = (async () => {\n const { default: PouchDB } = await import(\"pouchdb\");\n const pouchDbOptions = {\n name: \"graffitiDb\",\n ...this.options.pouchDBOptions,\n };\n const db = new PouchDB<GraffitiObjectWithTombstone>(\n pouchDbOptions.name,\n pouchDbOptions,\n );\n await db\n //@ts-ignore\n .put({\n _id: \"_design/indexes\",\n views: {\n objectsPerChannelAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(channel) + \"/\" + paddedLastModified;\n //@ts-ignore\n emit(id);\n });\n }.toString(),\n },\n orphansPerActorAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.channels.length === 0) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n paddedLastModified;\n //@ts-ignore\n emit(id);\n }\n }.toString(),\n },\n channelStatsPerActor: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.tombstone) return;\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n encodeURIComponent(channel);\n //@ts-ignore\n emit(id, object.lastModified);\n });\n }.toString(),\n reduce: \"_stats\",\n },\n },\n })\n //@ts-ignore\n .catch((error) => {\n if (\n error &&\n typeof error === \"object\" &&\n \"name\" in error &&\n error.name === \"conflict\"\n ) {\n // Design document already exists\n return;\n } else {\n throw error;\n }\n });\n return db;\n })();\n }\n return this.db_;\n }\n\n protected get applyPatch() {\n if (!this.applyPatch_) {\n this.applyPatch_ = (async () => {\n const imported = await import(\"fast-json-patch\");\n return imported.applyPatch || imported.default.applyPatch;\n })();\n }\n return this.applyPatch_;\n }\n\n protected get ajv() {\n if (!this.ajv_) {\n this.ajv_ = this.options.ajv\n ? Promise.resolve(this.options.ajv)\n : (async () => {\n const { default: Ajv } = await import(\"ajv\");\n return new Ajv({ strict: false });\n })();\n }\n return this.ajv_;\n }\n\n protected extractGraffitiObject(\n object: GraffitiObjectWithTombstone,\n ): GraffitiObjectBase {\n const { value, channels, allowed, url, actor, lastModified } = object;\n return {\n value,\n channels,\n allowed,\n url,\n actor,\n lastModified,\n };\n }\n\n constructor(options?: GraffitiLocalOptions) {\n this.options = options ?? {};\n this.origin = this.options.origin ?? DEFAULT_ORIGIN;\n if (!this.origin.endsWith(\":\") && !this.origin.endsWith(\"/\")) {\n this.origin += \"/\";\n }\n }\n\n protected async allDocsAtLocation(objectUrl: string | GraffitiObjectUrl) {\n const url = unpackObjectUrl(objectUrl) + \"/\";\n const results = await (\n await this.db\n ).allDocs({\n startkey: url,\n endkey: url + \"\\uffff\", // \\uffff is the last unicode character\n include_docs: true,\n });\n const docs = results.rows\n .map((row) => row.doc)\n // Remove undefined docs\n .reduce<\n PouchDB.Core.ExistingDocument<\n GraffitiObjectWithTombstone & PouchDB.Core.AllDocsMeta\n >[]\n >((acc, doc) => {\n if (doc) acc.push(doc);\n return acc;\n }, []);\n return docs;\n }\n\n protected docId(objectUrl: GraffitiObjectUrl) {\n return objectUrl.url + \"/\" + randomBase64();\n }\n\n get: Graffiti[\"get\"] = async (...args) => {\n const [urlObject, schema, session] = args;\n\n // TODO: Rate limit getting the same object\n // over and over\n\n const docsAll = await this.allDocsAtLocation(urlObject);\n\n // Filter out ones not allowed\n const docs = docsAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, session),\n );\n if (!docs.length)\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n\n // Get the most recent document\n const doc = docs.reduce((a, b) =>\n a.lastModified > b.lastModified ||\n (a.lastModified === b.lastModified && !a.tombstone && b.tombstone)\n ? a\n : b,\n );\n\n if (doc.tombstone) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n }\n\n const object = this.extractGraffitiObject(doc);\n\n // Mask out the allowed list and channels\n // if the user is not the owner\n maskGraffitiObject(object, [], session);\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n if (!validate(object)) {\n throw new GraffitiErrorSchemaMismatch();\n }\n return object;\n };\n\n /**\n * Deletes all docs at a particular location.\n * If the `keepLatest` flag is set to true,\n * the doc with the most recent timestamp will be\n * spared. If there are multiple docs with the same\n * timestamp, the one with the highest `_id` will be\n * spared.\n */\n protected async deleteAtLocation(\n url: GraffitiObjectUrl | string,\n options: {\n keepLatest?: boolean;\n session?: GraffitiSession;\n } = {\n keepLatest: false,\n },\n ) {\n const docsAtLocationAll = await this.allDocsAtLocation(url);\n const docsAtLocationAllowed = options.session\n ? docsAtLocationAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, options.session),\n )\n : docsAtLocationAll;\n if (!docsAtLocationAllowed.length) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to delete either does not exist or you are not allowed to see it\",\n );\n } else if (\n options.session &&\n docsAtLocationAllowed.some((doc) => doc.actor !== options.session?.actor)\n ) {\n throw new GraffitiErrorForbidden(\n \"You cannot delete an object owned by another actor\",\n );\n }\n const docsAtLocation = docsAtLocationAllowed.filter(\n (doc) => !doc.tombstone,\n );\n if (!docsAtLocation.length) return undefined;\n\n // Get the most recent lastModified timestamp.\n const latestModified = docsAtLocation\n .map((doc) => doc.lastModified)\n .reduce((a, b) => (a > b ? a : b));\n\n // Delete all old docs\n const docsToDelete = docsAtLocation.filter(\n (doc) => !options.keepLatest || doc.lastModified < latestModified,\n );\n\n // For docs with the same timestamp,\n // keep the one with the highest _id\n // to break concurrency ties\n const concurrentDocsAll = docsAtLocation.filter(\n (doc) => options.keepLatest && doc.lastModified === latestModified,\n );\n if (concurrentDocsAll.length) {\n const keepDocId = concurrentDocsAll\n .map((doc) => doc._id)\n .reduce((a, b) => (a > b ? a : b));\n const concurrentDocsToDelete = concurrentDocsAll.filter(\n (doc) => doc._id !== keepDocId,\n );\n docsToDelete.push(...concurrentDocsToDelete);\n }\n\n const lastModified = options.keepLatest\n ? latestModified\n : new Date().getTime();\n\n const deleteResults = await (\n await this.db\n ).bulkDocs<GraffitiObjectBase>(\n docsToDelete.map((doc) => ({\n ...doc,\n tombstone: true,\n lastModified,\n })),\n );\n\n // Get one of the docs that was deleted\n let deletedObject: GraffitiObjectBase | undefined = undefined;\n for (const resultOrError of deleteResults) {\n if (\"ok\" in resultOrError) {\n const { id } = resultOrError;\n const deletedDoc = docsToDelete.find((doc) => doc._id === id);\n if (deletedDoc) {\n deletedObject = {\n ...this.extractGraffitiObject(deletedDoc),\n lastModified,\n };\n break;\n }\n }\n }\n\n return deletedObject;\n }\n\n delete: Graffiti[\"delete\"] = async (...args) => {\n const [url, session] = args;\n const deletedObject = await this.deleteAtLocation(url, {\n session,\n });\n if (!deletedObject) {\n throw new GraffitiErrorNotFound(\"The object has already been deleted\");\n }\n return deletedObject;\n };\n\n put: Graffiti[\"put\"] = async (...args) => {\n const [objectPartial, session] = args;\n if (objectPartial.actor && objectPartial.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot put an object with a different actor than the session actor\",\n );\n }\n\n if (objectPartial.url) {\n let oldObject: GraffitiObjectBase | undefined;\n try {\n oldObject = await this.get(objectPartial.url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n if (!this.options.allowSettingArbitraryUrls) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to replace does not exist or you are not allowed to see it\",\n );\n }\n } else {\n throw e;\n }\n }\n if (oldObject?.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to replace is owned by another actor\",\n );\n }\n }\n\n const lastModified =\n ((this.options.allowSettinngLastModified ?? false) &&\n objectPartial.lastModified) ||\n new Date().getTime();\n\n const object: GraffitiObjectWithTombstone = {\n value: objectPartial.value,\n channels: objectPartial.channels,\n allowed: objectPartial.allowed,\n url: objectPartial.url ?? this.origin + randomBase64(),\n actor: session.actor,\n tombstone: false,\n lastModified,\n };\n\n await (\n await this.db\n ).put({\n _id: this.docId(object),\n ...object,\n });\n\n // Delete the old object\n const previousObject = await this.deleteAtLocation(object, {\n keepLatest: true,\n });\n if (previousObject) {\n return previousObject;\n } else {\n return {\n ...object,\n value: {},\n channels: [],\n allowed: [],\n tombstone: true,\n };\n }\n };\n\n patch: Graffiti[\"patch\"] = async (...args) => {\n const [patch, url, session] = args;\n let originalObject: GraffitiObjectBase;\n try {\n originalObject = await this.get(url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to patch does not exist or you are not allowed to see it\",\n );\n } else {\n throw e;\n }\n }\n if (originalObject.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to patch is owned by another actor\",\n );\n }\n\n // Patch it outside of the database\n const patchObject: GraffitiObjectBase = { ...originalObject };\n for (const prop of [\"value\", \"channels\", \"allowed\"] as const) {\n applyGraffitiPatch(await this.applyPatch, prop, patch, patchObject);\n }\n\n // Make sure the value is an object\n if (\n typeof patchObject.value !== \"object\" ||\n Array.isArray(patchObject.value) ||\n !patchObject.value\n ) {\n throw new GraffitiErrorPatchError(\"value is no longer an object\");\n }\n\n // Make sure the channels are an array of strings\n if (\n !Array.isArray(patchObject.channels) ||\n !patchObject.channels.every((channel) => typeof channel === \"string\")\n ) {\n throw new GraffitiErrorPatchError(\n \"channels are no longer an array of strings\",\n );\n }\n\n // Make sure the allowed list is an array of strings or undefined\n if (\n patchObject.allowed &&\n (!Array.isArray(patchObject.allowed) ||\n !patchObject.allowed.every((allowed) => typeof allowed === \"string\"))\n ) {\n throw new GraffitiErrorPatchError(\n \"allowed list is not an array of strings\",\n );\n }\n\n patchObject.lastModified = new Date().getTime();\n await (\n await this.db\n ).put({\n ...patchObject,\n tombstone: false,\n _id: this.docId(patchObject),\n });\n\n // Delete the old object\n await this.deleteAtLocation(patchObject, {\n keepLatest: true,\n });\n\n return {\n ...originalObject,\n lastModified: patchObject.lastModified,\n };\n };\n\n protected queryLastModifiedSuffixes(\n schema: JSONSchema,\n lastModified?: number,\n ) {\n // Use the index for queries over ranges of lastModified\n let startKeySuffix = \"\";\n let endKeySuffix = \"\\uffff\";\n if (\n typeof schema === \"object\" &&\n schema.properties?.lastModified &&\n typeof schema.properties.lastModified === \"object\"\n ) {\n const lastModifiedSchema = schema.properties.lastModified;\n\n const minimum =\n lastModified && lastModifiedSchema.minimum\n ? Math.max(lastModified, lastModifiedSchema.minimum)\n : (lastModified ?? lastModifiedSchema.minimum);\n const exclusiveMinimum = lastModifiedSchema.exclusiveMinimum;\n\n let intMinimum: number | undefined;\n if (exclusiveMinimum !== undefined) {\n intMinimum = Math.ceil(exclusiveMinimum);\n intMinimum === exclusiveMinimum && intMinimum++;\n } else if (minimum !== undefined) {\n intMinimum = Math.ceil(minimum);\n }\n\n if (intMinimum !== undefined) {\n startKeySuffix = intMinimum.toString().padStart(15, \"0\");\n }\n\n const maximum = lastModifiedSchema.maximum;\n const exclusiveMaximum = lastModifiedSchema.exclusiveMaximum;\n\n let intMaximum: number | undefined;\n if (exclusiveMaximum !== undefined) {\n intMaximum = Math.floor(exclusiveMaximum);\n intMaximum === exclusiveMaximum && intMaximum--;\n } else if (maximum !== undefined) {\n intMaximum = Math.floor(maximum);\n }\n\n if (intMaximum !== undefined) {\n endKeySuffix = intMaximum.toString().padStart(15, \"0\");\n }\n }\n return {\n startKeySuffix,\n endKeySuffix,\n };\n }\n\n protected async *streamObjects<Schema extends JSONSchema>(\n index: string,\n startkey: string,\n endkey: string,\n validate: ReturnType<typeof compileGraffitiObjectSchema<Schema>>,\n session: GraffitiSession | undefined | null,\n ifModifiedSince: number | undefined,\n channels?: string[],\n processedIds?: Set<string>,\n ): AsyncGenerator<GraffitiObjectStreamContinueEntry<Schema>> {\n if (ifModifiedSince !== undefined) {\n // Subtract a minute to make sure we don't miss any objects\n ifModifiedSince -= LAST_MODIFIED_BUFFER;\n }\n\n const result = await (\n await this.db\n ).query<GraffitiObjectWithTombstone>(index, {\n startkey,\n endkey,\n include_docs: true,\n });\n\n for (const row of result.rows) {\n const doc = row.doc;\n if (!doc) continue;\n\n if (processedIds?.has(doc._id)) continue;\n processedIds?.add(doc._id);\n\n // If this is not a continuation, skip tombstones\n if (ifModifiedSince === undefined && doc.tombstone) continue;\n\n const object = this.extractGraffitiObject(doc);\n\n if (channels) {\n if (!isActorAllowedGraffitiObject(object, session)) continue;\n maskGraffitiObject(object, channels, session);\n }\n\n if (!validate(object)) continue;\n\n yield doc.tombstone\n ? {\n tombstone: true,\n object: {\n url: object.url,\n lastModified: object.lastModified,\n },\n }\n : { object };\n }\n }\n\n protected async waitToContinue(ifModifiedSince: number | undefined) {\n if (ifModifiedSince === undefined) return;\n const continueBuffer = this.options.continueBuffer ?? 1000;\n const timeElapsedSinceContinue = Date.now() - ifModifiedSince;\n if (timeElapsedSinceContinue < continueBuffer) {\n // Continue was called too soon,\n // wait a bit before continuing\n await new Promise((resolve) =>\n setTimeout(resolve, continueBuffer - timeElapsedSinceContinue),\n );\n }\n }\n\n protected async *discoverMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n await this.waitToContinue(ifModifiedSince);\n\n const [channels, schema, session] = args;\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n\n const processedIds = new Set<string>();\n\n const startTime = new Date().getTime();\n\n for (const channel of channels) {\n const keyPrefix = encodeURIComponent(channel) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/objectsPerChannelAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n channels,\n processedIds,\n );\n\n for await (const result of iterator) yield result;\n }\n\n return startTime;\n }\n\n protected async *recoverOrphansMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n await this.waitToContinue(ifModifiedSince);\n\n const [schema, session] = args;\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n\n const startTime = new Date().getTime();\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/orphansPerActorAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n );\n\n for await (const result of iterator) yield result;\n\n return startTime;\n }\n\n protected discoverCursor(\n args: Parameters<typeof Graffiti.prototype.discover<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"discover:\" +\n JSON.stringify({\n channels: args[0],\n schema: args[1],\n actor: args[2]?.actor,\n ifModifiedSince: ifModifiedSince,\n })\n );\n }\n\n protected async *discoverContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.discoverMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () => this.discoverContinue<Schema>(args, ifModifiedSince),\n cursor: this.discoverCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n discover: Graffiti[\"discover\"] = (...args) => {\n const iterator = this.discoverMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.discoverContinue<(typeof args)[1]>(args, result.value),\n cursor: this_.discoverCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n protected recoverOrphansCursor(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"orphans:\" +\n JSON.stringify({\n schema: args[0],\n actor: args[1]?.actor,\n ifModifiedSince,\n })\n );\n }\n\n protected async *recoverOrphansContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.recoverOrphansMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () =>\n this.recoverOrphansContinue<Schema>(args, ifModifiedSince),\n cursor: this.recoverOrphansCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n recoverOrphans: Graffiti[\"recoverOrphans\"] = (...args) => {\n const iterator = this.recoverOrphansMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.recoverOrphansContinue<(typeof args)[0]>(\n args,\n result.value,\n ),\n cursor: this_.recoverOrphansCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n channelStats: Graffiti[\"channelStats\"] = (session) => {\n const this_ = this;\n return (async function* () {\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const result = await (\n await this_.db\n ).query(\"indexes/channelStatsPerActor\", {\n startkey: keyPrefix,\n endkey: keyPrefix + \"\\uffff\",\n reduce: true,\n group: true,\n });\n for (const row of result.rows) {\n const channelEncoded = row.key.split(\"/\")[1];\n if (typeof channelEncoded !== \"string\") continue;\n const { count, max: lastModified } = row.value;\n if (typeof count !== \"number\" || typeof lastModified !== \"number\")\n continue;\n yield {\n value: {\n channel: decodeURIComponent(channelEncoded),\n count,\n lastModified,\n },\n };\n }\n })();\n };\n\n continueObjectStream: Graffiti[\"continueObjectStream\"] = (\n cursor,\n session,\n ) => {\n if (cursor.startsWith(\"discover:\")) {\n const { channels, schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"discover:\".length),\n );\n if (actor && actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.discoverContinue<{}>(\n [channels, schema, session],\n ifModifiedSince,\n );\n } else if (cursor.startsWith(\"orphans:\")) {\n const { schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"orphans:\".length),\n );\n if (!session || actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.recoverOrphansContinue<{}>(\n [schema, session],\n ifModifiedSince,\n );\n } else {\n throw new GraffitiErrorNotFound(\"Cursor not found\");\n }\n };\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["doc", "ifModifiedSince"]
7
7
  }
@@ -41,6 +41,11 @@ export interface GraffitiLocalOptions {
41
41
  * If not provided, an internal instance will be created.
42
42
  */
43
43
  ajv?: Ajv;
44
+ /**
45
+ * Wait at least this long (in milliseconds) before continuing a stream.
46
+ * A basic form of rate limiting. Defaults to 1 seconds.
47
+ */
48
+ continueBuffer?: number;
44
49
  }
45
50
  type GraffitiObjectWithTombstone = GraffitiObjectBase & {
46
51
  tombstone: boolean;
@@ -85,6 +90,7 @@ export declare class GraffitiLocalDatabase implements Omit<Graffiti, "login" | "
85
90
  endKeySuffix: string;
86
91
  };
87
92
  protected streamObjects<Schema extends JSONSchema>(index: string, startkey: string, endkey: string, validate: ReturnType<typeof compileGraffitiObjectSchema<Schema>>, session: GraffitiSession | undefined | null, ifModifiedSince: number | undefined, channels?: string[], processedIds?: Set<string>): AsyncGenerator<GraffitiObjectStreamContinueEntry<Schema>>;
93
+ protected waitToContinue(ifModifiedSince: number | undefined): Promise<void>;
88
94
  protected discoverMeta<Schema extends JSONSchema>(args: Parameters<typeof Graffiti.prototype.discover<Schema>>, ifModifiedSince?: number): AsyncGenerator<GraffitiObjectStreamContinueEntry<Schema>, number | undefined>;
89
95
  protected recoverOrphansMeta<Schema extends JSONSchema>(args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>, ifModifiedSince?: number): AsyncGenerator<GraffitiObjectStreamContinueEntry<Schema>, number | undefined>;
90
96
  protected discoverCursor(args: Parameters<typeof Graffiti.prototype.discover<{}>>, ifModifiedSince?: number): string;
@@ -1 +1 @@
1
- {"version":3,"file":"database.d.ts","sourceRoot":"","sources":["../src/database.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,QAAQ,EACR,kBAAkB,EAClB,iBAAiB,EACjB,UAAU,EACV,eAAe,EACf,4BAA4B,EAC5B,iCAAiC,EAClC,MAAM,sBAAsB,CAAC;AAO9B,OAAO,EAKL,2BAA2B,EAE5B,MAAM,gBAAgB,CAAC;AACxB,OAAO,KAAK,GAAG,MAAM,KAAK,CAAC;AAC3B,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAElD;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC;;;;;;OAMG;IACH,cAAc,CAAC,EAAE,OAAO,CAAC,aAAa,CAAC,qBAAqB,CAAC;IAC7D;;;OAGG;IACH,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB;;;;;;;OAOG;IACH,yBAAyB,CAAC,EAAE,OAAO,CAAC;IACpC;;;;;;OAMG;IACH,yBAAyB,CAAC,EAAE,OAAO,CAAC;IACpC;;;OAGG;IACH,GAAG,CAAC,EAAE,GAAG,CAAC;CACX;AAKD,KAAK,2BAA2B,GAAG,kBAAkB,GAAG;IAAE,SAAS,EAAE,OAAO,CAAA;CAAE,CAAC;AAE/E;;;GAGG;AACH,qBAAa,qBACX,YAAW,IAAI,CAAC,QAAQ,EAAE,OAAO,GAAG,QAAQ,GAAG,eAAe,CAAC;IAE/D,SAAS,CAAC,GAAG,EACT,OAAO,CAAC,OAAO,CAAC,QAAQ,CAAC,2BAA2B,CAAC,CAAC,GACtD,SAAS,CAAC;IACd,SAAS,CAAC,WAAW,EAAE,OAAO,CAAC,OAAO,UAAU,CAAC,GAAG,SAAS,CAAC;IAC9D,SAAS,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,CAAC,GAAG,SAAS,CAAC;IACzC,SAAS,CAAC,QAAQ,CAAC,OAAO,EAAE,oBAAoB,CAAC;IACjD,SAAS,CAAC,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAC;IAElC,IAAI,EAAE,2DA+EL;IAED,SAAS,KAAK,UAAU,+BAQvB;IAED,SAAS,KAAK,GAAG,iBAUhB;IAED,SAAS,CAAC,qBAAqB,CAC7B,MAAM,EAAE,2BAA2B,GAClC,kBAAkB;gBAYT,OAAO,CAAC,EAAE,oBAAoB;cAQ1B,iBAAiB,CAAC,SAAS,EAAE,MAAM,GAAG,iBAAiB;mBA9IJ,OAAO;;IAqK1E,SAAS,CAAC,KAAK,CAAC,SAAS,EAAE,iBAAiB;IAI5C,GAAG,EAAE,QAAQ,CAAC,KAAK,CAAC,CAuClB;IAEF;;;;;;;OAOG;cACa,gBAAgB,CAC9B,GAAG,EAAE,iBAAiB,GAAG,MAAM,EAC/B,OAAO,GAAE;QACP,UAAU,CAAC,EAAE,OAAO,CAAC;QACrB,OAAO,CAAC,EAAE,eAAe,CAAC;KAG3B;IAoFH,MAAM,EAAE,QAAQ,CAAC,QAAQ,CAAC,CASxB;IAEF,GAAG,EAAE,QAAQ,CAAC,KAAK,CAAC,CAmElB;IAEF,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,CA0EtB;IAEF,SAAS,CAAC,yBAAyB,CACjC,MAAM,EAAE,UAAU,EAClB,YAAY,CAAC,EAAE,MAAM;;;;cAmDN,aAAa,CAAC,MAAM,SAAS,UAAU,EACtD,KAAK,EAAE,MAAM,EACb,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,EACd,QAAQ,EAAE,UAAU,CAAC,OAAO,2BAA2B,CAAC,MAAM,CAAC,CAAC,EAChE,OAAO,EAAE,eAAe,GAAG,SAAS,GAAG,IAAI,EAC3C,eAAe,EAAE,MAAM,GAAG,SAAS,EACnC,QAAQ,CAAC,EAAE,MAAM,EAAE,EACnB,YAAY,CAAC,EAAE,GAAG,CAAC,MAAM,CAAC,GACzB,cAAc,CAAC,iCAAiC,CAAC,MAAM,CAAC,CAAC;cAyC3C,YAAY,CAAC,MAAM,SAAS,UAAU,EACrD,IAAI,EAAE,UAAU,CAAC,OAAO,QAAQ,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,EAC5D,eAAe,CAAC,EAAE,MAAM,GACvB,cAAc,CACf,iCAAiC,CAAC,MAAM,CAAC,EACzC,MAAM,GAAG,SAAS,CACnB;cAmCgB,kBAAkB,CAAC,MAAM,SAAS,UAAU,EAC3D,IAAI,EAAE,UAAU,CAAC,OAAO,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,EAClE,eAAe,CAAC,EAAE,MAAM,GACvB,cAAc,CACf,iCAAiC,CAAC,MAAM,CAAC,EACzC,MAAM,GAAG,SAAS,CACnB;IA4BD,SAAS,CAAC,cAAc,CACtB,IAAI,EAAE,UAAU,CAAC,OAAO,QAAQ,CAAC,SAAS,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,EACxD,eAAe,CAAC,EAAE,MAAM,GACvB,MAAM;cAYQ,gBAAgB,CAAC,MAAM,SAAS,UAAU,EACzD,IAAI,EAAE,UAAU,CAAC,OAAO,QAAQ,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,EAC5D,eAAe,CAAC,EAAE,MAAM,GACvB,4BAA4B,CAAC,MAAM,CAAC;IAgBvC,QAAQ,EAAE,QAAQ,CAAC,UAAU,CAAC,CAmB5B;IAEF,SAAS,CAAC,oBAAoB,CAC5B,IAAI,EAAE,UAAU,CAAC,OAAO,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC,EAC9D,eAAe,CAAC,EAAE,MAAM,GACvB,MAAM;cAWQ,sBAAsB,CAAC,MAAM,SAAS,UAAU,EAC/D,IAAI,EAAE,UAAU,CAAC,OAAO,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,EAClE,eAAe,CAAC,EAAE,MAAM,GACvB,4BAA4B,CAAC,MAAM,CAAC;IAiBvC,cAAc,EAAE,QAAQ,CAAC,gBAAgB,CAAC,CAsBxC;IAEF,YAAY,EAAE,QAAQ,CAAC,cAAc,CAAC,CA2BpC;IAEF,oBAAoB,EAAE,QAAQ,CAAC,sBAAsB,CAAC,CAiCpD;CACH"}
1
+ {"version":3,"file":"database.d.ts","sourceRoot":"","sources":["../src/database.ts"],"names":[],"mappings":"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"}
@@ -374,7 +374,9 @@ class GraffitiLocalDatabase {
374
374
  };
375
375
  }
376
376
  async *streamObjects(index, startkey, endkey, validate, session, ifModifiedSince, channels, processedIds) {
377
- const showTombstones = ifModifiedSince !== void 0;
377
+ if (ifModifiedSince !== void 0) {
378
+ ifModifiedSince -= LAST_MODIFIED_BUFFER;
379
+ }
378
380
  const result = await (await this.db).query(index, {
379
381
  startkey,
380
382
  endkey,
@@ -385,7 +387,7 @@ class GraffitiLocalDatabase {
385
387
  if (!doc) continue;
386
388
  if (processedIds?.has(doc._id)) continue;
387
389
  processedIds?.add(doc._id);
388
- if (!showTombstones && doc.tombstone) continue;
390
+ if (ifModifiedSince === void 0 && doc.tombstone) continue;
389
391
  const object = this.extractGraffitiObject(doc);
390
392
  if (channels) {
391
393
  if (!isActorAllowedGraffitiObject(object, session)) continue;
@@ -401,7 +403,18 @@ class GraffitiLocalDatabase {
401
403
  } : { object };
402
404
  }
403
405
  }
406
+ async waitToContinue(ifModifiedSince) {
407
+ if (ifModifiedSince === void 0) return;
408
+ const continueBuffer = this.options.continueBuffer ?? 1e3;
409
+ const timeElapsedSinceContinue = Date.now() - ifModifiedSince;
410
+ if (timeElapsedSinceContinue < continueBuffer) {
411
+ await new Promise(
412
+ (resolve) => setTimeout(resolve, continueBuffer - timeElapsedSinceContinue)
413
+ );
414
+ }
415
+ }
404
416
  async *discoverMeta(args, ifModifiedSince) {
417
+ await this.waitToContinue(ifModifiedSince);
405
418
  const [channels, schema, session] = args;
406
419
  const validate = compileGraffitiObjectSchema(await this.ajv, schema);
407
420
  const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(
@@ -426,9 +439,10 @@ class GraffitiLocalDatabase {
426
439
  );
427
440
  for await (const result of iterator) yield result;
428
441
  }
429
- return startTime - LAST_MODIFIED_BUFFER;
442
+ return startTime;
430
443
  }
431
444
  async *recoverOrphansMeta(args, ifModifiedSince) {
445
+ await this.waitToContinue(ifModifiedSince);
432
446
  const [schema, session] = args;
433
447
  const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(
434
448
  schema,
@@ -448,7 +462,7 @@ class GraffitiLocalDatabase {
448
462
  ifModifiedSince
449
463
  );
450
464
  for await (const result of iterator) yield result;
451
- return startTime - LAST_MODIFIED_BUFFER;
465
+ return startTime;
452
466
  }
453
467
  discoverCursor(args, ifModifiedSince) {
454
468
  return "discover:" + JSON.stringify({
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/database.ts"],
4
- "sourcesContent": ["import type {\n Graffiti,\n GraffitiObjectBase,\n GraffitiObjectUrl,\n JSONSchema,\n GraffitiSession,\n GraffitiObjectStreamContinue,\n GraffitiObjectStreamContinueEntry,\n} from \"@graffiti-garden/api\";\nimport {\n GraffitiErrorNotFound,\n GraffitiErrorSchemaMismatch,\n GraffitiErrorForbidden,\n GraffitiErrorPatchError,\n} from \"@graffiti-garden/api\";\nimport {\n randomBase64,\n applyGraffitiPatch,\n maskGraffitiObject,\n isActorAllowedGraffitiObject,\n compileGraffitiObjectSchema,\n unpackObjectUrl,\n} from \"./utilities.js\";\nimport type Ajv from \"ajv\";\nimport type { applyPatch } from \"fast-json-patch\";\n\n/**\n * Constructor options for the GraffitiPoubchDB class.\n */\nexport interface GraffitiLocalOptions {\n /**\n * Options to pass to the PouchDB constructor.\n * Defaults to `{ name: \"graffitiDb\" }`.\n *\n * See the [PouchDB documentation](https://pouchdb.com/api.html#create_database)\n * for available options.\n */\n pouchDBOptions?: PouchDB.Configuration.DatabaseConfiguration;\n /**\n * Includes the scheme and other information (possibly domain name)\n * to prefix prefixes all URLs put in the system. Defaults to `graffiti:local`.\n */\n origin?: string;\n /**\n * Whether to allow putting objects at arbtirary URLs, i.e.\n * URLs that are *not* prefixed with the origin or not generated\n * by the system. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettingArbitraryUrls?: boolean;\n /**\n * Whether to allow the user to set the lastModified field\n * when putting objects. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettinngLastModified?: boolean;\n /**\n * An optional Ajv instance to use for schema validation.\n * If not provided, an internal instance will be created.\n */\n ajv?: Ajv;\n}\n\nconst DEFAULT_ORIGIN = \"graffiti:local:\";\nconst LAST_MODIFIED_BUFFER = 60000;\n\ntype GraffitiObjectWithTombstone = GraffitiObjectBase & { tombstone: boolean };\n\n/**\n * An implementation of only the database operations of the\n * GraffitiAPI without synchronization or session management.\n */\nexport class GraffitiLocalDatabase\n implements Omit<Graffiti, \"login\" | \"logout\" | \"sessionEvents\">\n{\n protected db_:\n | Promise<PouchDB.Database<GraffitiObjectWithTombstone>>\n | undefined;\n protected applyPatch_: Promise<typeof applyPatch> | undefined;\n protected ajv_: Promise<Ajv> | undefined;\n protected readonly options: GraffitiLocalOptions;\n protected readonly origin: string;\n\n get db() {\n if (!this.db_) {\n this.db_ = (async () => {\n const { default: PouchDB } = await import(\"pouchdb\");\n const pouchDbOptions = {\n name: \"graffitiDb\",\n ...this.options.pouchDBOptions,\n };\n const db = new PouchDB<GraffitiObjectWithTombstone>(\n pouchDbOptions.name,\n pouchDbOptions,\n );\n await db\n //@ts-ignore\n .put({\n _id: \"_design/indexes\",\n views: {\n objectsPerChannelAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(channel) + \"/\" + paddedLastModified;\n //@ts-ignore\n emit(id);\n });\n }.toString(),\n },\n orphansPerActorAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.channels.length === 0) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n paddedLastModified;\n //@ts-ignore\n emit(id);\n }\n }.toString(),\n },\n channelStatsPerActor: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.tombstone) return;\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n encodeURIComponent(channel);\n //@ts-ignore\n emit(id, object.lastModified);\n });\n }.toString(),\n reduce: \"_stats\",\n },\n },\n })\n //@ts-ignore\n .catch((error) => {\n if (\n error &&\n typeof error === \"object\" &&\n \"name\" in error &&\n error.name === \"conflict\"\n ) {\n // Design document already exists\n return;\n } else {\n throw error;\n }\n });\n return db;\n })();\n }\n return this.db_;\n }\n\n protected get applyPatch() {\n if (!this.applyPatch_) {\n this.applyPatch_ = (async () => {\n const imported = await import(\"fast-json-patch\");\n return imported.applyPatch || imported.default.applyPatch;\n })();\n }\n return this.applyPatch_;\n }\n\n protected get ajv() {\n if (!this.ajv_) {\n this.ajv_ = this.options.ajv\n ? Promise.resolve(this.options.ajv)\n : (async () => {\n const { default: Ajv } = await import(\"ajv\");\n return new Ajv({ strict: false });\n })();\n }\n return this.ajv_;\n }\n\n protected extractGraffitiObject(\n object: GraffitiObjectWithTombstone,\n ): GraffitiObjectBase {\n const { value, channels, allowed, url, actor, lastModified } = object;\n return {\n value,\n channels,\n allowed,\n url,\n actor,\n lastModified,\n };\n }\n\n constructor(options?: GraffitiLocalOptions) {\n this.options = options ?? {};\n this.origin = this.options.origin ?? DEFAULT_ORIGIN;\n if (!this.origin.endsWith(\":\") && !this.origin.endsWith(\"/\")) {\n this.origin += \"/\";\n }\n }\n\n protected async allDocsAtLocation(objectUrl: string | GraffitiObjectUrl) {\n const url = unpackObjectUrl(objectUrl) + \"/\";\n const results = await (\n await this.db\n ).allDocs({\n startkey: url,\n endkey: url + \"\\uffff\", // \\uffff is the last unicode character\n include_docs: true,\n });\n const docs = results.rows\n .map((row) => row.doc)\n // Remove undefined docs\n .reduce<\n PouchDB.Core.ExistingDocument<\n GraffitiObjectWithTombstone & PouchDB.Core.AllDocsMeta\n >[]\n >((acc, doc) => {\n if (doc) acc.push(doc);\n return acc;\n }, []);\n return docs;\n }\n\n protected docId(objectUrl: GraffitiObjectUrl) {\n return objectUrl.url + \"/\" + randomBase64();\n }\n\n get: Graffiti[\"get\"] = async (...args) => {\n const [urlObject, schema, session] = args;\n\n const docsAll = await this.allDocsAtLocation(urlObject);\n\n // Filter out ones not allowed\n const docs = docsAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, session),\n );\n if (!docs.length)\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n\n // Get the most recent document\n const doc = docs.reduce((a, b) =>\n a.lastModified > b.lastModified ||\n (a.lastModified === b.lastModified && !a.tombstone && b.tombstone)\n ? a\n : b,\n );\n\n if (doc.tombstone) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n }\n\n const object = this.extractGraffitiObject(doc);\n\n // Mask out the allowed list and channels\n // if the user is not the owner\n maskGraffitiObject(object, [], session);\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n if (!validate(object)) {\n throw new GraffitiErrorSchemaMismatch();\n }\n return object;\n };\n\n /**\n * Deletes all docs at a particular location.\n * If the `keepLatest` flag is set to true,\n * the doc with the most recent timestamp will be\n * spared. If there are multiple docs with the same\n * timestamp, the one with the highest `_id` will be\n * spared.\n */\n protected async deleteAtLocation(\n url: GraffitiObjectUrl | string,\n options: {\n keepLatest?: boolean;\n session?: GraffitiSession;\n } = {\n keepLatest: false,\n },\n ) {\n const docsAtLocationAll = await this.allDocsAtLocation(url);\n const docsAtLocationAllowed = options.session\n ? docsAtLocationAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, options.session),\n )\n : docsAtLocationAll;\n if (!docsAtLocationAllowed.length) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to delete either does not exist or you are not allowed to see it\",\n );\n } else if (\n options.session &&\n docsAtLocationAllowed.some((doc) => doc.actor !== options.session?.actor)\n ) {\n throw new GraffitiErrorForbidden(\n \"You cannot delete an object owned by another actor\",\n );\n }\n const docsAtLocation = docsAtLocationAllowed.filter(\n (doc) => !doc.tombstone,\n );\n if (!docsAtLocation.length) return undefined;\n\n // Get the most recent lastModified timestamp.\n const latestModified = docsAtLocation\n .map((doc) => doc.lastModified)\n .reduce((a, b) => (a > b ? a : b));\n\n // Delete all old docs\n const docsToDelete = docsAtLocation.filter(\n (doc) => !options.keepLatest || doc.lastModified < latestModified,\n );\n\n // For docs with the same timestamp,\n // keep the one with the highest _id\n // to break concurrency ties\n const concurrentDocsAll = docsAtLocation.filter(\n (doc) => options.keepLatest && doc.lastModified === latestModified,\n );\n if (concurrentDocsAll.length) {\n const keepDocId = concurrentDocsAll\n .map((doc) => doc._id)\n .reduce((a, b) => (a > b ? a : b));\n const concurrentDocsToDelete = concurrentDocsAll.filter(\n (doc) => doc._id !== keepDocId,\n );\n docsToDelete.push(...concurrentDocsToDelete);\n }\n\n const lastModified = options.keepLatest\n ? latestModified\n : new Date().getTime();\n\n const deleteResults = await (\n await this.db\n ).bulkDocs<GraffitiObjectBase>(\n docsToDelete.map((doc) => ({\n ...doc,\n tombstone: true,\n lastModified,\n })),\n );\n\n // Get one of the docs that was deleted\n let deletedObject: GraffitiObjectBase | undefined = undefined;\n for (const resultOrError of deleteResults) {\n if (\"ok\" in resultOrError) {\n const { id } = resultOrError;\n const deletedDoc = docsToDelete.find((doc) => doc._id === id);\n if (deletedDoc) {\n deletedObject = {\n ...this.extractGraffitiObject(deletedDoc),\n lastModified,\n };\n break;\n }\n }\n }\n\n return deletedObject;\n }\n\n delete: Graffiti[\"delete\"] = async (...args) => {\n const [url, session] = args;\n const deletedObject = await this.deleteAtLocation(url, {\n session,\n });\n if (!deletedObject) {\n throw new GraffitiErrorNotFound(\"The object has already been deleted\");\n }\n return deletedObject;\n };\n\n put: Graffiti[\"put\"] = async (...args) => {\n const [objectPartial, session] = args;\n if (objectPartial.actor && objectPartial.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot put an object with a different actor than the session actor\",\n );\n }\n\n if (objectPartial.url) {\n let oldObject: GraffitiObjectBase | undefined;\n try {\n oldObject = await this.get(objectPartial.url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n if (!this.options.allowSettingArbitraryUrls) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to replace does not exist or you are not allowed to see it\",\n );\n }\n } else {\n throw e;\n }\n }\n if (oldObject?.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to replace is owned by another actor\",\n );\n }\n }\n\n const lastModified =\n ((this.options.allowSettinngLastModified ?? false) &&\n objectPartial.lastModified) ||\n new Date().getTime();\n\n const object: GraffitiObjectWithTombstone = {\n value: objectPartial.value,\n channels: objectPartial.channels,\n allowed: objectPartial.allowed,\n url: objectPartial.url ?? this.origin + randomBase64(),\n actor: session.actor,\n tombstone: false,\n lastModified,\n };\n\n await (\n await this.db\n ).put({\n _id: this.docId(object),\n ...object,\n });\n\n // Delete the old object\n const previousObject = await this.deleteAtLocation(object, {\n keepLatest: true,\n });\n if (previousObject) {\n return previousObject;\n } else {\n return {\n ...object,\n value: {},\n channels: [],\n allowed: [],\n tombstone: true,\n };\n }\n };\n\n patch: Graffiti[\"patch\"] = async (...args) => {\n const [patch, url, session] = args;\n let originalObject: GraffitiObjectBase;\n try {\n originalObject = await this.get(url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to patch does not exist or you are not allowed to see it\",\n );\n } else {\n throw e;\n }\n }\n if (originalObject.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to patch is owned by another actor\",\n );\n }\n\n // Patch it outside of the database\n const patchObject: GraffitiObjectBase = { ...originalObject };\n for (const prop of [\"value\", \"channels\", \"allowed\"] as const) {\n applyGraffitiPatch(await this.applyPatch, prop, patch, patchObject);\n }\n\n // Make sure the value is an object\n if (\n typeof patchObject.value !== \"object\" ||\n Array.isArray(patchObject.value) ||\n !patchObject.value\n ) {\n throw new GraffitiErrorPatchError(\"value is no longer an object\");\n }\n\n // Make sure the channels are an array of strings\n if (\n !Array.isArray(patchObject.channels) ||\n !patchObject.channels.every((channel) => typeof channel === \"string\")\n ) {\n throw new GraffitiErrorPatchError(\n \"channels are no longer an array of strings\",\n );\n }\n\n // Make sure the allowed list is an array of strings or undefined\n if (\n patchObject.allowed &&\n (!Array.isArray(patchObject.allowed) ||\n !patchObject.allowed.every((allowed) => typeof allowed === \"string\"))\n ) {\n throw new GraffitiErrorPatchError(\n \"allowed list is not an array of strings\",\n );\n }\n\n patchObject.lastModified = new Date().getTime();\n await (\n await this.db\n ).put({\n ...patchObject,\n tombstone: false,\n _id: this.docId(patchObject),\n });\n\n // Delete the old object\n await this.deleteAtLocation(patchObject, {\n keepLatest: true,\n });\n\n return {\n ...originalObject,\n lastModified: patchObject.lastModified,\n };\n };\n\n protected queryLastModifiedSuffixes(\n schema: JSONSchema,\n lastModified?: number,\n ) {\n // Use the index for queries over ranges of lastModified\n let startKeySuffix = \"\";\n let endKeySuffix = \"\\uffff\";\n if (\n typeof schema === \"object\" &&\n schema.properties?.lastModified &&\n typeof schema.properties.lastModified === \"object\"\n ) {\n const lastModifiedSchema = schema.properties.lastModified;\n\n const minimum =\n lastModified && lastModifiedSchema.minimum\n ? Math.max(lastModified, lastModifiedSchema.minimum)\n : (lastModified ?? lastModifiedSchema.minimum);\n const exclusiveMinimum = lastModifiedSchema.exclusiveMinimum;\n\n let intMinimum: number | undefined;\n if (exclusiveMinimum !== undefined) {\n intMinimum = Math.ceil(exclusiveMinimum);\n intMinimum === exclusiveMinimum && intMinimum++;\n } else if (minimum !== undefined) {\n intMinimum = Math.ceil(minimum);\n }\n\n if (intMinimum !== undefined) {\n startKeySuffix = intMinimum.toString().padStart(15, \"0\");\n }\n\n const maximum = lastModifiedSchema.maximum;\n const exclusiveMaximum = lastModifiedSchema.exclusiveMaximum;\n\n let intMaximum: number | undefined;\n if (exclusiveMaximum !== undefined) {\n intMaximum = Math.floor(exclusiveMaximum);\n intMaximum === exclusiveMaximum && intMaximum--;\n } else if (maximum !== undefined) {\n intMaximum = Math.floor(maximum);\n }\n\n if (intMaximum !== undefined) {\n endKeySuffix = intMaximum.toString().padStart(15, \"0\");\n }\n }\n return {\n startKeySuffix,\n endKeySuffix,\n };\n }\n\n protected async *streamObjects<Schema extends JSONSchema>(\n index: string,\n startkey: string,\n endkey: string,\n validate: ReturnType<typeof compileGraffitiObjectSchema<Schema>>,\n session: GraffitiSession | undefined | null,\n ifModifiedSince: number | undefined,\n channels?: string[],\n processedIds?: Set<string>,\n ): AsyncGenerator<GraffitiObjectStreamContinueEntry<Schema>> {\n const showTombstones = ifModifiedSince !== undefined;\n\n const result = await (\n await this.db\n ).query<GraffitiObjectWithTombstone>(index, {\n startkey,\n endkey,\n include_docs: true,\n });\n\n for (const row of result.rows) {\n const doc = row.doc;\n if (!doc) continue;\n\n if (processedIds?.has(doc._id)) continue;\n processedIds?.add(doc._id);\n\n if (!showTombstones && doc.tombstone) continue;\n\n const object = this.extractGraffitiObject(doc);\n\n if (channels) {\n if (!isActorAllowedGraffitiObject(object, session)) continue;\n maskGraffitiObject(object, channels, session);\n }\n\n if (!validate(object)) continue;\n\n yield doc.tombstone\n ? {\n tombstone: true,\n object: {\n url: object.url,\n lastModified: object.lastModified,\n },\n }\n : { object };\n }\n }\n\n protected async *discoverMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n const [channels, schema, session] = args;\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n\n const processedIds = new Set<string>();\n\n const startTime = new Date().getTime();\n\n for (const channel of channels) {\n const keyPrefix = encodeURIComponent(channel) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/objectsPerChannelAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n channels,\n processedIds,\n );\n\n for await (const result of iterator) yield result;\n }\n\n // Subtract a minute to make sure we don't miss any objects\n return startTime - LAST_MODIFIED_BUFFER;\n }\n\n protected async *recoverOrphansMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n const [schema, session] = args;\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n\n const startTime = new Date().getTime();\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/orphansPerActorAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n );\n\n for await (const result of iterator) yield result;\n\n return startTime - LAST_MODIFIED_BUFFER;\n }\n\n protected discoverCursor(\n args: Parameters<typeof Graffiti.prototype.discover<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"discover:\" +\n JSON.stringify({\n channels: args[0],\n schema: args[1],\n actor: args[2]?.actor,\n ifModifiedSince: ifModifiedSince,\n })\n );\n }\n\n protected async *discoverContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.discoverMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () => this.discoverContinue<Schema>(args, ifModifiedSince),\n cursor: this.discoverCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n discover: Graffiti[\"discover\"] = (...args) => {\n const iterator = this.discoverMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.discoverContinue<(typeof args)[1]>(args, result.value),\n cursor: this_.discoverCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n protected recoverOrphansCursor(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"orphans:\" +\n JSON.stringify({\n schema: args[0],\n actor: args[1]?.actor,\n ifModifiedSince,\n })\n );\n }\n\n protected async *recoverOrphansContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.recoverOrphansMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () =>\n this.recoverOrphansContinue<Schema>(args, ifModifiedSince),\n cursor: this.recoverOrphansCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n recoverOrphans: Graffiti[\"recoverOrphans\"] = (...args) => {\n const iterator = this.recoverOrphansMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.recoverOrphansContinue<(typeof args)[0]>(\n args,\n result.value,\n ),\n cursor: this_.recoverOrphansCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n channelStats: Graffiti[\"channelStats\"] = (session) => {\n const this_ = this;\n return (async function* () {\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const result = await (\n await this_.db\n ).query(\"indexes/channelStatsPerActor\", {\n startkey: keyPrefix,\n endkey: keyPrefix + \"\\uffff\",\n reduce: true,\n group: true,\n });\n for (const row of result.rows) {\n const channelEncoded = row.key.split(\"/\")[1];\n if (typeof channelEncoded !== \"string\") continue;\n const { count, max: lastModified } = row.value;\n if (typeof count !== \"number\" || typeof lastModified !== \"number\")\n continue;\n yield {\n value: {\n channel: decodeURIComponent(channelEncoded),\n count,\n lastModified,\n },\n };\n }\n })();\n };\n\n continueObjectStream: Graffiti[\"continueObjectStream\"] = (\n cursor,\n session,\n ) => {\n if (cursor.startsWith(\"discover:\")) {\n const { channels, schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"discover:\".length),\n );\n if (actor && actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.discoverContinue<{}>(\n [channels, schema, session],\n ifModifiedSince,\n );\n } else if (cursor.startsWith(\"orphans:\")) {\n const { schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"orphans:\".length),\n );\n if (!session || actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.recoverOrphansContinue<{}>(\n [schema, session],\n ifModifiedSince,\n );\n } else {\n throw new GraffitiErrorNotFound(\"Cursor not found\");\n }\n };\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import type {\n Graffiti,\n GraffitiObjectBase,\n GraffitiObjectUrl,\n JSONSchema,\n GraffitiSession,\n GraffitiObjectStreamContinue,\n GraffitiObjectStreamContinueEntry,\n} from \"@graffiti-garden/api\";\nimport {\n GraffitiErrorNotFound,\n GraffitiErrorSchemaMismatch,\n GraffitiErrorForbidden,\n GraffitiErrorPatchError,\n} from \"@graffiti-garden/api\";\nimport {\n randomBase64,\n applyGraffitiPatch,\n maskGraffitiObject,\n isActorAllowedGraffitiObject,\n compileGraffitiObjectSchema,\n unpackObjectUrl,\n} from \"./utilities.js\";\nimport type Ajv from \"ajv\";\nimport type { applyPatch } from \"fast-json-patch\";\n\n/**\n * Constructor options for the GraffitiPoubchDB class.\n */\nexport interface GraffitiLocalOptions {\n /**\n * Options to pass to the PouchDB constructor.\n * Defaults to `{ name: \"graffitiDb\" }`.\n *\n * See the [PouchDB documentation](https://pouchdb.com/api.html#create_database)\n * for available options.\n */\n pouchDBOptions?: PouchDB.Configuration.DatabaseConfiguration;\n /**\n * Includes the scheme and other information (possibly domain name)\n * to prefix prefixes all URLs put in the system. Defaults to `graffiti:local`.\n */\n origin?: string;\n /**\n * Whether to allow putting objects at arbtirary URLs, i.e.\n * URLs that are *not* prefixed with the origin or not generated\n * by the system. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettingArbitraryUrls?: boolean;\n /**\n * Whether to allow the user to set the lastModified field\n * when putting objects. Defaults to `false`.\n *\n * Allows this implementation to be used as a client-side cache\n * for remote sources.\n */\n allowSettinngLastModified?: boolean;\n /**\n * An optional Ajv instance to use for schema validation.\n * If not provided, an internal instance will be created.\n */\n ajv?: Ajv;\n /**\n * Wait at least this long (in milliseconds) before continuing a stream.\n * A basic form of rate limiting. Defaults to 1 seconds.\n */\n continueBuffer?: number;\n}\n\nconst DEFAULT_ORIGIN = \"graffiti:local:\";\n\n// During stream continuations, return objects\n// that have possibly already been seen over this window\nconst LAST_MODIFIED_BUFFER = 60000;\n\ntype GraffitiObjectWithTombstone = GraffitiObjectBase & { tombstone: boolean };\n\n/**\n * An implementation of only the database operations of the\n * GraffitiAPI without synchronization or session management.\n */\nexport class GraffitiLocalDatabase\n implements Omit<Graffiti, \"login\" | \"logout\" | \"sessionEvents\">\n{\n protected db_:\n | Promise<PouchDB.Database<GraffitiObjectWithTombstone>>\n | undefined;\n protected applyPatch_: Promise<typeof applyPatch> | undefined;\n protected ajv_: Promise<Ajv> | undefined;\n protected readonly options: GraffitiLocalOptions;\n protected readonly origin: string;\n\n get db() {\n if (!this.db_) {\n this.db_ = (async () => {\n const { default: PouchDB } = await import(\"pouchdb\");\n const pouchDbOptions = {\n name: \"graffitiDb\",\n ...this.options.pouchDBOptions,\n };\n const db = new PouchDB<GraffitiObjectWithTombstone>(\n pouchDbOptions.name,\n pouchDbOptions,\n );\n await db\n //@ts-ignore\n .put({\n _id: \"_design/indexes\",\n views: {\n objectsPerChannelAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(channel) + \"/\" + paddedLastModified;\n //@ts-ignore\n emit(id);\n });\n }.toString(),\n },\n orphansPerActorAndLastModified: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.channels.length === 0) {\n const paddedLastModified = object.lastModified\n .toString()\n .padStart(15, \"0\");\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n paddedLastModified;\n //@ts-ignore\n emit(id);\n }\n }.toString(),\n },\n channelStatsPerActor: {\n map: function (object: GraffitiObjectWithTombstone) {\n if (object.tombstone) return;\n object.channels.forEach(function (channel) {\n const id =\n encodeURIComponent(object.actor) +\n \"/\" +\n encodeURIComponent(channel);\n //@ts-ignore\n emit(id, object.lastModified);\n });\n }.toString(),\n reduce: \"_stats\",\n },\n },\n })\n //@ts-ignore\n .catch((error) => {\n if (\n error &&\n typeof error === \"object\" &&\n \"name\" in error &&\n error.name === \"conflict\"\n ) {\n // Design document already exists\n return;\n } else {\n throw error;\n }\n });\n return db;\n })();\n }\n return this.db_;\n }\n\n protected get applyPatch() {\n if (!this.applyPatch_) {\n this.applyPatch_ = (async () => {\n const imported = await import(\"fast-json-patch\");\n return imported.applyPatch || imported.default.applyPatch;\n })();\n }\n return this.applyPatch_;\n }\n\n protected get ajv() {\n if (!this.ajv_) {\n this.ajv_ = this.options.ajv\n ? Promise.resolve(this.options.ajv)\n : (async () => {\n const { default: Ajv } = await import(\"ajv\");\n return new Ajv({ strict: false });\n })();\n }\n return this.ajv_;\n }\n\n protected extractGraffitiObject(\n object: GraffitiObjectWithTombstone,\n ): GraffitiObjectBase {\n const { value, channels, allowed, url, actor, lastModified } = object;\n return {\n value,\n channels,\n allowed,\n url,\n actor,\n lastModified,\n };\n }\n\n constructor(options?: GraffitiLocalOptions) {\n this.options = options ?? {};\n this.origin = this.options.origin ?? DEFAULT_ORIGIN;\n if (!this.origin.endsWith(\":\") && !this.origin.endsWith(\"/\")) {\n this.origin += \"/\";\n }\n }\n\n protected async allDocsAtLocation(objectUrl: string | GraffitiObjectUrl) {\n const url = unpackObjectUrl(objectUrl) + \"/\";\n const results = await (\n await this.db\n ).allDocs({\n startkey: url,\n endkey: url + \"\\uffff\", // \\uffff is the last unicode character\n include_docs: true,\n });\n const docs = results.rows\n .map((row) => row.doc)\n // Remove undefined docs\n .reduce<\n PouchDB.Core.ExistingDocument<\n GraffitiObjectWithTombstone & PouchDB.Core.AllDocsMeta\n >[]\n >((acc, doc) => {\n if (doc) acc.push(doc);\n return acc;\n }, []);\n return docs;\n }\n\n protected docId(objectUrl: GraffitiObjectUrl) {\n return objectUrl.url + \"/\" + randomBase64();\n }\n\n get: Graffiti[\"get\"] = async (...args) => {\n const [urlObject, schema, session] = args;\n\n // TODO: Rate limit getting the same object\n // over and over\n\n const docsAll = await this.allDocsAtLocation(urlObject);\n\n // Filter out ones not allowed\n const docs = docsAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, session),\n );\n if (!docs.length)\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n\n // Get the most recent document\n const doc = docs.reduce((a, b) =>\n a.lastModified > b.lastModified ||\n (a.lastModified === b.lastModified && !a.tombstone && b.tombstone)\n ? a\n : b,\n );\n\n if (doc.tombstone) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to get either does not exist or you are not allowed to see it\",\n );\n }\n\n const object = this.extractGraffitiObject(doc);\n\n // Mask out the allowed list and channels\n // if the user is not the owner\n maskGraffitiObject(object, [], session);\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n if (!validate(object)) {\n throw new GraffitiErrorSchemaMismatch();\n }\n return object;\n };\n\n /**\n * Deletes all docs at a particular location.\n * If the `keepLatest` flag is set to true,\n * the doc with the most recent timestamp will be\n * spared. If there are multiple docs with the same\n * timestamp, the one with the highest `_id` will be\n * spared.\n */\n protected async deleteAtLocation(\n url: GraffitiObjectUrl | string,\n options: {\n keepLatest?: boolean;\n session?: GraffitiSession;\n } = {\n keepLatest: false,\n },\n ) {\n const docsAtLocationAll = await this.allDocsAtLocation(url);\n const docsAtLocationAllowed = options.session\n ? docsAtLocationAll.filter((doc) =>\n isActorAllowedGraffitiObject(doc, options.session),\n )\n : docsAtLocationAll;\n if (!docsAtLocationAllowed.length) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to delete either does not exist or you are not allowed to see it\",\n );\n } else if (\n options.session &&\n docsAtLocationAllowed.some((doc) => doc.actor !== options.session?.actor)\n ) {\n throw new GraffitiErrorForbidden(\n \"You cannot delete an object owned by another actor\",\n );\n }\n const docsAtLocation = docsAtLocationAllowed.filter(\n (doc) => !doc.tombstone,\n );\n if (!docsAtLocation.length) return undefined;\n\n // Get the most recent lastModified timestamp.\n const latestModified = docsAtLocation\n .map((doc) => doc.lastModified)\n .reduce((a, b) => (a > b ? a : b));\n\n // Delete all old docs\n const docsToDelete = docsAtLocation.filter(\n (doc) => !options.keepLatest || doc.lastModified < latestModified,\n );\n\n // For docs with the same timestamp,\n // keep the one with the highest _id\n // to break concurrency ties\n const concurrentDocsAll = docsAtLocation.filter(\n (doc) => options.keepLatest && doc.lastModified === latestModified,\n );\n if (concurrentDocsAll.length) {\n const keepDocId = concurrentDocsAll\n .map((doc) => doc._id)\n .reduce((a, b) => (a > b ? a : b));\n const concurrentDocsToDelete = concurrentDocsAll.filter(\n (doc) => doc._id !== keepDocId,\n );\n docsToDelete.push(...concurrentDocsToDelete);\n }\n\n const lastModified = options.keepLatest\n ? latestModified\n : new Date().getTime();\n\n const deleteResults = await (\n await this.db\n ).bulkDocs<GraffitiObjectBase>(\n docsToDelete.map((doc) => ({\n ...doc,\n tombstone: true,\n lastModified,\n })),\n );\n\n // Get one of the docs that was deleted\n let deletedObject: GraffitiObjectBase | undefined = undefined;\n for (const resultOrError of deleteResults) {\n if (\"ok\" in resultOrError) {\n const { id } = resultOrError;\n const deletedDoc = docsToDelete.find((doc) => doc._id === id);\n if (deletedDoc) {\n deletedObject = {\n ...this.extractGraffitiObject(deletedDoc),\n lastModified,\n };\n break;\n }\n }\n }\n\n return deletedObject;\n }\n\n delete: Graffiti[\"delete\"] = async (...args) => {\n const [url, session] = args;\n const deletedObject = await this.deleteAtLocation(url, {\n session,\n });\n if (!deletedObject) {\n throw new GraffitiErrorNotFound(\"The object has already been deleted\");\n }\n return deletedObject;\n };\n\n put: Graffiti[\"put\"] = async (...args) => {\n const [objectPartial, session] = args;\n if (objectPartial.actor && objectPartial.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot put an object with a different actor than the session actor\",\n );\n }\n\n if (objectPartial.url) {\n let oldObject: GraffitiObjectBase | undefined;\n try {\n oldObject = await this.get(objectPartial.url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n if (!this.options.allowSettingArbitraryUrls) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to replace does not exist or you are not allowed to see it\",\n );\n }\n } else {\n throw e;\n }\n }\n if (oldObject?.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to replace is owned by another actor\",\n );\n }\n }\n\n const lastModified =\n ((this.options.allowSettinngLastModified ?? false) &&\n objectPartial.lastModified) ||\n new Date().getTime();\n\n const object: GraffitiObjectWithTombstone = {\n value: objectPartial.value,\n channels: objectPartial.channels,\n allowed: objectPartial.allowed,\n url: objectPartial.url ?? this.origin + randomBase64(),\n actor: session.actor,\n tombstone: false,\n lastModified,\n };\n\n await (\n await this.db\n ).put({\n _id: this.docId(object),\n ...object,\n });\n\n // Delete the old object\n const previousObject = await this.deleteAtLocation(object, {\n keepLatest: true,\n });\n if (previousObject) {\n return previousObject;\n } else {\n return {\n ...object,\n value: {},\n channels: [],\n allowed: [],\n tombstone: true,\n };\n }\n };\n\n patch: Graffiti[\"patch\"] = async (...args) => {\n const [patch, url, session] = args;\n let originalObject: GraffitiObjectBase;\n try {\n originalObject = await this.get(url, {}, session);\n } catch (e) {\n if (e instanceof GraffitiErrorNotFound) {\n throw new GraffitiErrorNotFound(\n \"The object you are trying to patch does not exist or you are not allowed to see it\",\n );\n } else {\n throw e;\n }\n }\n if (originalObject.actor !== session.actor) {\n throw new GraffitiErrorForbidden(\n \"The object you are trying to patch is owned by another actor\",\n );\n }\n\n // Patch it outside of the database\n const patchObject: GraffitiObjectBase = { ...originalObject };\n for (const prop of [\"value\", \"channels\", \"allowed\"] as const) {\n applyGraffitiPatch(await this.applyPatch, prop, patch, patchObject);\n }\n\n // Make sure the value is an object\n if (\n typeof patchObject.value !== \"object\" ||\n Array.isArray(patchObject.value) ||\n !patchObject.value\n ) {\n throw new GraffitiErrorPatchError(\"value is no longer an object\");\n }\n\n // Make sure the channels are an array of strings\n if (\n !Array.isArray(patchObject.channels) ||\n !patchObject.channels.every((channel) => typeof channel === \"string\")\n ) {\n throw new GraffitiErrorPatchError(\n \"channels are no longer an array of strings\",\n );\n }\n\n // Make sure the allowed list is an array of strings or undefined\n if (\n patchObject.allowed &&\n (!Array.isArray(patchObject.allowed) ||\n !patchObject.allowed.every((allowed) => typeof allowed === \"string\"))\n ) {\n throw new GraffitiErrorPatchError(\n \"allowed list is not an array of strings\",\n );\n }\n\n patchObject.lastModified = new Date().getTime();\n await (\n await this.db\n ).put({\n ...patchObject,\n tombstone: false,\n _id: this.docId(patchObject),\n });\n\n // Delete the old object\n await this.deleteAtLocation(patchObject, {\n keepLatest: true,\n });\n\n return {\n ...originalObject,\n lastModified: patchObject.lastModified,\n };\n };\n\n protected queryLastModifiedSuffixes(\n schema: JSONSchema,\n lastModified?: number,\n ) {\n // Use the index for queries over ranges of lastModified\n let startKeySuffix = \"\";\n let endKeySuffix = \"\\uffff\";\n if (\n typeof schema === \"object\" &&\n schema.properties?.lastModified &&\n typeof schema.properties.lastModified === \"object\"\n ) {\n const lastModifiedSchema = schema.properties.lastModified;\n\n const minimum =\n lastModified && lastModifiedSchema.minimum\n ? Math.max(lastModified, lastModifiedSchema.minimum)\n : (lastModified ?? lastModifiedSchema.minimum);\n const exclusiveMinimum = lastModifiedSchema.exclusiveMinimum;\n\n let intMinimum: number | undefined;\n if (exclusiveMinimum !== undefined) {\n intMinimum = Math.ceil(exclusiveMinimum);\n intMinimum === exclusiveMinimum && intMinimum++;\n } else if (minimum !== undefined) {\n intMinimum = Math.ceil(minimum);\n }\n\n if (intMinimum !== undefined) {\n startKeySuffix = intMinimum.toString().padStart(15, \"0\");\n }\n\n const maximum = lastModifiedSchema.maximum;\n const exclusiveMaximum = lastModifiedSchema.exclusiveMaximum;\n\n let intMaximum: number | undefined;\n if (exclusiveMaximum !== undefined) {\n intMaximum = Math.floor(exclusiveMaximum);\n intMaximum === exclusiveMaximum && intMaximum--;\n } else if (maximum !== undefined) {\n intMaximum = Math.floor(maximum);\n }\n\n if (intMaximum !== undefined) {\n endKeySuffix = intMaximum.toString().padStart(15, \"0\");\n }\n }\n return {\n startKeySuffix,\n endKeySuffix,\n };\n }\n\n protected async *streamObjects<Schema extends JSONSchema>(\n index: string,\n startkey: string,\n endkey: string,\n validate: ReturnType<typeof compileGraffitiObjectSchema<Schema>>,\n session: GraffitiSession | undefined | null,\n ifModifiedSince: number | undefined,\n channels?: string[],\n processedIds?: Set<string>,\n ): AsyncGenerator<GraffitiObjectStreamContinueEntry<Schema>> {\n if (ifModifiedSince !== undefined) {\n // Subtract a minute to make sure we don't miss any objects\n ifModifiedSince -= LAST_MODIFIED_BUFFER;\n }\n\n const result = await (\n await this.db\n ).query<GraffitiObjectWithTombstone>(index, {\n startkey,\n endkey,\n include_docs: true,\n });\n\n for (const row of result.rows) {\n const doc = row.doc;\n if (!doc) continue;\n\n if (processedIds?.has(doc._id)) continue;\n processedIds?.add(doc._id);\n\n // If this is not a continuation, skip tombstones\n if (ifModifiedSince === undefined && doc.tombstone) continue;\n\n const object = this.extractGraffitiObject(doc);\n\n if (channels) {\n if (!isActorAllowedGraffitiObject(object, session)) continue;\n maskGraffitiObject(object, channels, session);\n }\n\n if (!validate(object)) continue;\n\n yield doc.tombstone\n ? {\n tombstone: true,\n object: {\n url: object.url,\n lastModified: object.lastModified,\n },\n }\n : { object };\n }\n }\n\n protected async waitToContinue(ifModifiedSince: number | undefined) {\n if (ifModifiedSince === undefined) return;\n const continueBuffer = this.options.continueBuffer ?? 1000;\n const timeElapsedSinceContinue = Date.now() - ifModifiedSince;\n if (timeElapsedSinceContinue < continueBuffer) {\n // Continue was called too soon,\n // wait a bit before continuing\n await new Promise((resolve) =>\n setTimeout(resolve, continueBuffer - timeElapsedSinceContinue),\n );\n }\n }\n\n protected async *discoverMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n await this.waitToContinue(ifModifiedSince);\n\n const [channels, schema, session] = args;\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n\n const processedIds = new Set<string>();\n\n const startTime = new Date().getTime();\n\n for (const channel of channels) {\n const keyPrefix = encodeURIComponent(channel) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/objectsPerChannelAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n channels,\n processedIds,\n );\n\n for await (const result of iterator) yield result;\n }\n\n return startTime;\n }\n\n protected async *recoverOrphansMeta<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): AsyncGenerator<\n GraffitiObjectStreamContinueEntry<Schema>,\n number | undefined\n > {\n await this.waitToContinue(ifModifiedSince);\n\n const [schema, session] = args;\n const { startKeySuffix, endKeySuffix } = this.queryLastModifiedSuffixes(\n schema,\n ifModifiedSince,\n );\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const startkey = keyPrefix + startKeySuffix;\n const endkey = keyPrefix + endKeySuffix;\n\n const validate = compileGraffitiObjectSchema(await this.ajv, schema);\n\n const startTime = new Date().getTime();\n\n const iterator = this.streamObjects<Schema>(\n \"indexes/orphansPerActorAndLastModified\",\n startkey,\n endkey,\n validate,\n session,\n ifModifiedSince,\n );\n\n for await (const result of iterator) yield result;\n\n return startTime;\n }\n\n protected discoverCursor(\n args: Parameters<typeof Graffiti.prototype.discover<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"discover:\" +\n JSON.stringify({\n channels: args[0],\n schema: args[1],\n actor: args[2]?.actor,\n ifModifiedSince: ifModifiedSince,\n })\n );\n }\n\n protected async *discoverContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.discover<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.discoverMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () => this.discoverContinue<Schema>(args, ifModifiedSince),\n cursor: this.discoverCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n discover: Graffiti[\"discover\"] = (...args) => {\n const iterator = this.discoverMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.discoverContinue<(typeof args)[1]>(args, result.value),\n cursor: this_.discoverCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n protected recoverOrphansCursor(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<{}>>,\n ifModifiedSince?: number,\n ): string {\n return (\n \"orphans:\" +\n JSON.stringify({\n schema: args[0],\n actor: args[1]?.actor,\n ifModifiedSince,\n })\n );\n }\n\n protected async *recoverOrphansContinue<Schema extends JSONSchema>(\n args: Parameters<typeof Graffiti.prototype.recoverOrphans<Schema>>,\n ifModifiedSince?: number,\n ): GraffitiObjectStreamContinue<Schema> {\n const iterator = this.recoverOrphansMeta(args, ifModifiedSince);\n\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n const ifModifiedSince = result.value;\n return {\n continue: () =>\n this.recoverOrphansContinue<Schema>(args, ifModifiedSince),\n cursor: this.recoverOrphansCursor(args, ifModifiedSince),\n };\n }\n yield result.value;\n }\n }\n\n recoverOrphans: Graffiti[\"recoverOrphans\"] = (...args) => {\n const iterator = this.recoverOrphansMeta(args);\n\n const this_ = this;\n return (async function* () {\n while (true) {\n const result = await iterator.next();\n if (result.done) {\n return {\n continue: () =>\n this_.recoverOrphansContinue<(typeof args)[0]>(\n args,\n result.value,\n ),\n cursor: this_.recoverOrphansCursor(args, result.value),\n };\n }\n // Make sure to filter out tombstones\n if (result.value.tombstone) continue;\n yield result.value;\n }\n })();\n };\n\n channelStats: Graffiti[\"channelStats\"] = (session) => {\n const this_ = this;\n return (async function* () {\n const keyPrefix = encodeURIComponent(session.actor) + \"/\";\n const result = await (\n await this_.db\n ).query(\"indexes/channelStatsPerActor\", {\n startkey: keyPrefix,\n endkey: keyPrefix + \"\\uffff\",\n reduce: true,\n group: true,\n });\n for (const row of result.rows) {\n const channelEncoded = row.key.split(\"/\")[1];\n if (typeof channelEncoded !== \"string\") continue;\n const { count, max: lastModified } = row.value;\n if (typeof count !== \"number\" || typeof lastModified !== \"number\")\n continue;\n yield {\n value: {\n channel: decodeURIComponent(channelEncoded),\n count,\n lastModified,\n },\n };\n }\n })();\n };\n\n continueObjectStream: Graffiti[\"continueObjectStream\"] = (\n cursor,\n session,\n ) => {\n if (cursor.startsWith(\"discover:\")) {\n const { channels, schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"discover:\".length),\n );\n if (actor && actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.discoverContinue<{}>(\n [channels, schema, session],\n ifModifiedSince,\n );\n } else if (cursor.startsWith(\"orphans:\")) {\n const { schema, actor, ifModifiedSince } = JSON.parse(\n cursor.slice(\"orphans:\".length),\n );\n if (!session || actor !== session?.actor) {\n throw new GraffitiErrorForbidden(\n \"Cannot continue a cursor for another actor\",\n );\n }\n return this.recoverOrphansContinue<{}>(\n [schema, session],\n ifModifiedSince,\n );\n } else {\n throw new GraffitiErrorNotFound(\"Cursor not found\");\n }\n };\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["doc", "ifModifiedSince"]
7
7
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@graffiti-garden/implementation-local",
3
- "version": "0.6.2",
3
+ "version": "0.6.4",
4
4
  "description": "A local implementation of the Graffiti API using PouchDB",
5
5
  "types": "./dist/index.d.ts",
6
6
  "module": "./dist/esm/index.js",
@@ -81,7 +81,7 @@
81
81
  "vitest": "^3.0.5"
82
82
  },
83
83
  "dependencies": {
84
- "@graffiti-garden/api": "^0.6.0",
84
+ "@graffiti-garden/api": "^0.6.3",
85
85
  "@types/pouchdb": "^6.4.2",
86
86
  "ajv": "^8.17.1",
87
87
  "fast-json-patch": "^3.1.1",