"use server"; import * as Y from "yjs"; import * as base64 from "base64-js"; import { restoreOAuthSession, OAuthSessionError } from "src/atproto-oauth"; import { getIdentityData } from "actions/getIdentityData"; import { AtpBaseClient, PubLeafletBlocksHeader, PubLeafletBlocksImage, PubLeafletBlocksText, PubLeafletBlocksUnorderedList, PubLeafletDocument, PubLeafletPagesLinearDocument, PubLeafletPagesCanvas, PubLeafletRichtextFacet, PubLeafletBlocksWebsite, PubLeafletBlocksCode, PubLeafletBlocksMath, PubLeafletBlocksHorizontalRule, PubLeafletBlocksBskyPost, PubLeafletBlocksBlockquote, PubLeafletBlocksIframe, PubLeafletBlocksPage, PubLeafletBlocksPoll, PubLeafletBlocksButton, PubLeafletPollDefinition, } from "lexicons/api"; import { Block } from "components/Blocks/Block"; import { TID } from "@atproto/common"; import { supabaseServerClient } from "supabase/serverClient"; import { scanIndexLocal } from "src/replicache/utils"; import type { Fact } from "src/replicache"; import type { Attribute } from "src/replicache/attributes"; import { Delta, YJSFragmentToString } from "src/utils/yjsFragmentToString"; import { ids } from "lexicons/api/lexicons"; import { BlobRef } from "@atproto/lexicon"; import { AtUri } from "@atproto/syntax"; import { Json } from "supabase/database.types"; import { $Typed, UnicodeString } from "@atproto/api"; import { List, parseBlocksToList } from "src/utils/parseBlocksToList"; import { getBlocksWithTypeLocal } from "src/hooks/queries/useBlocks"; import { Lock } from "src/utils/lock"; import type { PubLeafletPublication } from "lexicons/api"; import { ColorToRGB, ColorToRGBA, } from "components/ThemeManager/colorToLexicons"; import { parseColor } from "@react-stately/color"; import { Notification, pingIdentityToUpdateNotification, } from "src/notifications"; import { v7 } from "uuid"; type PublishResult = | { success: true; rkey: string; record: PubLeafletDocument.Record } | { success: false; error: OAuthSessionError }; export async function publishToPublication({ root_entity, publication_uri, leaflet_id, title, description, tags, cover_image, entitiesToDelete, publishedAt, }: { root_entity: string; publication_uri?: string; leaflet_id: string; title?: string; description?: string; tags?: string[]; cover_image?: string | null; entitiesToDelete?: string[]; publishedAt?: string; }): Promise { let identity = await getIdentityData(); if (!identity || !identity.atp_did) { return { success: false, error: { type: "oauth_session_expired", message: "Not authenticated", did: "", }, }; } const sessionResult = await restoreOAuthSession(identity.atp_did); if (!sessionResult.ok) { return { success: false, error: sessionResult.error }; } let credentialSession = sessionResult.value; let agent = new AtpBaseClient( credentialSession.fetchHandler.bind(credentialSession), ); // Check if we're publishing to a publication or standalone let draft: any = null; let existingDocUri: string | null = null; if (publication_uri) { // Publishing to a publication - use leaflets_in_publications let { data, error } = await supabaseServerClient .from("publications") .select("*, leaflets_in_publications(*, documents(*))") .eq("uri", publication_uri) .eq("leaflets_in_publications.leaflet", leaflet_id) .single(); console.log(error); if (!data || identity.atp_did !== data?.identity_did) throw new Error("No draft or not publisher"); draft = data.leaflets_in_publications[0]; existingDocUri = draft?.doc; } else { // Publishing standalone - use leaflets_to_documents let { data } = await supabaseServerClient .from("leaflets_to_documents") .select("*, documents(*)") .eq("leaflet", leaflet_id) .single(); draft = data; existingDocUri = draft?.document; } // Heuristic: Remove title entities if this is the first time publishing // (when coming from a standalone leaflet with entitiesToDelete passed in) if (entitiesToDelete && entitiesToDelete.length > 0 && !existingDocUri) { await supabaseServerClient .from("entities") .delete() .in("id", entitiesToDelete); } let { data } = await supabaseServerClient.rpc("get_facts", { root: root_entity, }); let facts = (data as unknown as Fact[]) || []; let { pages } = await processBlocksToPages( facts, agent, root_entity, credentialSession.did!, ); let existingRecord = draft?.documents?.data as | PubLeafletDocument.Record | undefined; // Extract theme for standalone documents (not for publications) let theme: PubLeafletPublication.Theme | undefined; if (!publication_uri) { theme = await extractThemeFromFacts(facts, root_entity, agent); } // Upload cover image if provided let coverImageBlob: BlobRef | undefined; if (cover_image) { let scan = scanIndexLocal(facts); let [imageData] = scan.eav(cover_image, "block/image"); if (imageData) { let imageResponse = await fetch(imageData.data.src); if (imageResponse.status === 200) { let binary = await imageResponse.blob(); let blob = await agent.com.atproto.repo.uploadBlob(binary, { headers: { "Content-Type": binary.type }, }); coverImageBlob = blob.data.blob; } } } let record: PubLeafletDocument.Record = { $type: "pub.leaflet.document", author: credentialSession.did!, ...(publication_uri && { publication: publication_uri }), ...(theme && { theme }), title: title || "Untitled", description: description || "", ...(tags !== undefined && { tags }), // Include tags if provided (even if empty array to clear tags) ...(coverImageBlob && { coverImage: coverImageBlob }), // Include cover image if uploaded pages: pages.map((p) => { if (p.type === "canvas") { return { $type: "pub.leaflet.pages.canvas" as const, id: p.id, blocks: p.blocks as PubLeafletPagesCanvas.Block[], }; } else { return { $type: "pub.leaflet.pages.linearDocument" as const, id: p.id, blocks: p.blocks as PubLeafletPagesLinearDocument.Block[], }; } }), publishedAt: publishedAt || existingRecord?.publishedAt || new Date().toISOString(), }; // Keep the same rkey if updating an existing document let rkey = existingDocUri ? new AtUri(existingDocUri).rkey : TID.nextStr(); let { data: result } = await agent.com.atproto.repo.putRecord({ rkey, repo: credentialSession.did!, collection: record.$type, record, validate: false, //TODO publish the lexicon so we can validate! }); // Optimistically create database entries await supabaseServerClient.from("documents").upsert({ uri: result.uri, data: record as Json, }); if (publication_uri) { // Publishing to a publication - update both tables await Promise.all([ supabaseServerClient.from("documents_in_publications").upsert({ publication: publication_uri, document: result.uri, }), supabaseServerClient.from("leaflets_in_publications").upsert({ doc: result.uri, leaflet: leaflet_id, publication: publication_uri, title: title, description: description, }), ]); } else { // Publishing standalone - update leaflets_to_documents await supabaseServerClient.from("leaflets_to_documents").upsert({ leaflet: leaflet_id, document: result.uri, title: title || "Untitled", description: description || "", }); // Heuristic: Remove title entities if this is the first time publishing standalone // (when entitiesToDelete is provided and there's no existing document) if (entitiesToDelete && entitiesToDelete.length > 0 && !existingDocUri) { await supabaseServerClient .from("entities") .delete() .in("id", entitiesToDelete); } } // Create notifications for mentions (only on first publish) if (!existingDocUri) { await createMentionNotifications( result.uri, record, credentialSession.did!, ); } return { success: true, rkey, record: JSON.parse(JSON.stringify(record)) }; } async function processBlocksToPages( facts: Fact[], agent: AtpBaseClient, root_entity: string, did: string, ) { let scan = scanIndexLocal(facts); let pages: { id: string; blocks: | PubLeafletPagesLinearDocument.Block[] | PubLeafletPagesCanvas.Block[]; type: "doc" | "canvas"; }[] = []; // Create a lock to serialize image uploads const uploadLock = new Lock(); let firstEntity = scan.eav(root_entity, "root/page")?.[0]; if (!firstEntity) throw new Error("No root page"); // Check if the first page is a canvas or linear document let [pageType] = scan.eav(firstEntity.data.value, "page/type"); if (pageType?.data.value === "canvas") { // First page is a canvas let canvasBlocks = await canvasBlocksToRecord(firstEntity.data.value, did); pages.unshift({ id: firstEntity.data.value, blocks: canvasBlocks, type: "canvas", }); } else { // First page is a linear document let blocks = getBlocksWithTypeLocal(facts, firstEntity?.data.value); let b = await blocksToRecord(blocks, did); pages.unshift({ id: firstEntity.data.value, blocks: b, type: "doc", }); } return { pages }; async function uploadImage(src: string) { let data = await fetch(src); if (data.status !== 200) return; let binary = await data.blob(); return uploadLock.withLock(async () => { let blob = await agent.com.atproto.repo.uploadBlob(binary, { headers: { "Content-Type": binary.type }, }); return blob.data.blob; }); } async function blocksToRecord( blocks: Block[], did: string, ): Promise { let parsedBlocks = parseBlocksToList(blocks); return ( await Promise.all( parsedBlocks.map(async (blockOrList) => { if (blockOrList.type === "block") { let alignmentValue = scan.eav( blockOrList.block.value, "block/text-alignment", )[0]?.data.value; let alignment: ExcludeString< PubLeafletPagesLinearDocument.Block["alignment"] > = alignmentValue === "center" ? "lex:pub.leaflet.pages.linearDocument#textAlignCenter" : alignmentValue === "right" ? "lex:pub.leaflet.pages.linearDocument#textAlignRight" : alignmentValue === "justify" ? "lex:pub.leaflet.pages.linearDocument#textAlignJustify" : alignmentValue === "left" ? "lex:pub.leaflet.pages.linearDocument#textAlignLeft" : undefined; let b = await blockToRecord(blockOrList.block, did); if (!b) return []; let block: PubLeafletPagesLinearDocument.Block = { $type: "pub.leaflet.pages.linearDocument#block", block: b, }; if (alignment) block.alignment = alignment; return [block]; } else { let block: PubLeafletPagesLinearDocument.Block = { $type: "pub.leaflet.pages.linearDocument#block", block: { $type: "pub.leaflet.blocks.unorderedList", children: await childrenToRecord(blockOrList.children, did), }, }; return [block]; } }), ) ).flat(); } async function childrenToRecord(children: List[], did: string) { return ( await Promise.all( children.map(async (child) => { let content = await blockToRecord(child.block, did); if (!content) return []; let record: PubLeafletBlocksUnorderedList.ListItem = { $type: "pub.leaflet.blocks.unorderedList#listItem", content, children: await childrenToRecord(child.children, did), }; return record; }), ) ).flat(); } async function blockToRecord(b: Block, did: string) { const getBlockContent = (b: string) => { let [content] = scan.eav(b, "block/text"); if (!content) return ["", [] as PubLeafletRichtextFacet.Main[]] as const; let doc = new Y.Doc(); const update = base64.toByteArray(content.data.value); Y.applyUpdate(doc, update); let nodes = doc.getXmlElement("prosemirror").toArray(); let stringValue = YJSFragmentToString(nodes[0]); let { facets } = YJSFragmentToFacets(nodes[0]); return [stringValue, facets] as const; }; if (b.type === "card") { let [page] = scan.eav(b.value, "block/card"); if (!page) return; let [pageType] = scan.eav(page.data.value, "page/type"); if (pageType?.data.value === "canvas") { let canvasBlocks = await canvasBlocksToRecord(page.data.value, did); pages.push({ id: page.data.value, blocks: canvasBlocks, type: "canvas", }); } else { let blocks = getBlocksWithTypeLocal(facts, page.data.value); pages.push({ id: page.data.value, blocks: await blocksToRecord(blocks, did), type: "doc", }); } let block: $Typed = { $type: "pub.leaflet.blocks.page", id: page.data.value, }; return block; } if (b.type === "bluesky-post") { let [post] = scan.eav(b.value, "block/bluesky-post"); if (!post || !post.data.value.post) return; let block: $Typed = { $type: ids.PubLeafletBlocksBskyPost, postRef: { uri: post.data.value.post.uri, cid: post.data.value.post.cid, }, }; return block; } if (b.type === "horizontal-rule") { let block: $Typed = { $type: ids.PubLeafletBlocksHorizontalRule, }; return block; } if (b.type === "heading") { let [headingLevel] = scan.eav(b.value, "block/heading-level"); let [stringValue, facets] = getBlockContent(b.value); let block: $Typed = { $type: "pub.leaflet.blocks.header", level: Math.floor(headingLevel?.data.value || 1), plaintext: stringValue, facets, }; return block; } if (b.type === "blockquote") { let [stringValue, facets] = getBlockContent(b.value); let block: $Typed = { $type: ids.PubLeafletBlocksBlockquote, plaintext: stringValue, facets, }; return block; } if (b.type == "text") { let [stringValue, facets] = getBlockContent(b.value); let [textSize] = scan.eav(b.value, "block/text-size"); let block: $Typed = { $type: ids.PubLeafletBlocksText, plaintext: stringValue, facets, ...(textSize && { textSize: textSize.data.value }), }; return block; } if (b.type === "embed") { let [url] = scan.eav(b.value, "embed/url"); let [height] = scan.eav(b.value, "embed/height"); if (!url) return; let block: $Typed = { $type: "pub.leaflet.blocks.iframe", url: url.data.value, height: Math.floor(height?.data.value || 600), }; return block; } if (b.type == "image") { let [image] = scan.eav(b.value, "block/image"); if (!image) return; let [altText] = scan.eav(b.value, "image/alt"); let blobref = await uploadImage(image.data.src); if (!blobref) return; let block: $Typed = { $type: "pub.leaflet.blocks.image", image: blobref, aspectRatio: { height: Math.floor(image.data.height), width: Math.floor(image.data.width), }, alt: altText ? altText.data.value : undefined, }; return block; } if (b.type === "link") { let [previewImage] = scan.eav(b.value, "link/preview"); let [description] = scan.eav(b.value, "link/description"); let [src] = scan.eav(b.value, "link/url"); if (!src) return; let blobref = previewImage ? await uploadImage(previewImage?.data.src) : undefined; let [title] = scan.eav(b.value, "link/title"); let block: $Typed = { $type: "pub.leaflet.blocks.website", previewImage: blobref, src: src.data.value, description: description?.data.value, title: title?.data.value, }; return block; } if (b.type === "code") { let [language] = scan.eav(b.value, "block/code-language"); let [code] = scan.eav(b.value, "block/code"); let [theme] = scan.eav(root_entity, "theme/code-theme"); let block: $Typed = { $type: "pub.leaflet.blocks.code", language: language?.data.value, plaintext: code?.data.value || "", syntaxHighlightingTheme: theme?.data.value, }; return block; } if (b.type === "math") { let [math] = scan.eav(b.value, "block/math"); let block: $Typed = { $type: "pub.leaflet.blocks.math", tex: math?.data.value || "", }; return block; } if (b.type === "poll") { // Get poll options from the entity let pollOptions = scan.eav(b.value, "poll/options"); let options: PubLeafletPollDefinition.Option[] = pollOptions.map( (opt) => { let optionName = scan.eav(opt.data.value, "poll-option/name")?.[0]; return { $type: "pub.leaflet.poll.definition#option", text: optionName?.data.value || "", }; }, ); // Create the poll definition record let pollRecord: PubLeafletPollDefinition.Record = { $type: "pub.leaflet.poll.definition", name: "Poll", // Default name, can be customized options, }; // Upload the poll record let { data: pollResult } = await agent.com.atproto.repo.putRecord({ //use the entity id as the rkey so we can associate it in the editor rkey: b.value, repo: did, collection: pollRecord.$type, record: pollRecord, validate: false, }); // Optimistically write poll definition to database console.log( await supabaseServerClient.from("atp_poll_records").upsert({ uri: pollResult.uri, cid: pollResult.cid, record: pollRecord as Json, }), ); // Return a poll block with reference to the poll record let block: $Typed = { $type: "pub.leaflet.blocks.poll", pollRef: { uri: pollResult.uri, cid: pollResult.cid, }, }; return block; } if (b.type === "button") { let [text] = scan.eav(b.value, "button/text"); let [url] = scan.eav(b.value, "button/url"); if (!text || !url) return; let block: $Typed = { $type: "pub.leaflet.blocks.button", text: text.data.value, url: url.data.value, }; return block; } return; } async function canvasBlocksToRecord( pageID: string, did: string, ): Promise { let canvasBlocks = scan.eav(pageID, "canvas/block"); return ( await Promise.all( canvasBlocks.map(async (canvasBlock) => { let blockEntity = canvasBlock.data.value; let position = canvasBlock.data.position; // Get the block content let blockType = scan.eav(blockEntity, "block/type")?.[0]; if (!blockType) return null; let block: Block = { type: blockType.data.value, value: blockEntity, parent: pageID, position: "", factID: canvasBlock.id, }; let content = await blockToRecord(block, did); if (!content) return null; // Get canvas-specific properties let width = scan.eav(blockEntity, "canvas/block/width")?.[0]?.data.value || 360; let rotation = scan.eav(blockEntity, "canvas/block/rotation")?.[0] ?.data.value; let canvasBlockRecord: PubLeafletPagesCanvas.Block = { $type: "pub.leaflet.pages.canvas#block", block: content, x: Math.floor(position.x), y: Math.floor(position.y), width: Math.floor(width), ...(rotation !== undefined && { rotation: Math.floor(rotation) }), }; return canvasBlockRecord; }), ) ).filter((b): b is PubLeafletPagesCanvas.Block => b !== null); } } function YJSFragmentToFacets( node: Y.XmlElement | Y.XmlText | Y.XmlHook, byteOffset: number = 0, ): { facets: PubLeafletRichtextFacet.Main[]; byteLength: number } { if (node.constructor === Y.XmlElement) { // Handle inline mention nodes if (node.nodeName === "didMention") { const text = node.getAttribute("text") || ""; const unicodestring = new UnicodeString(text); const facet: PubLeafletRichtextFacet.Main = { index: { byteStart: byteOffset, byteEnd: byteOffset + unicodestring.length, }, features: [ { $type: "pub.leaflet.richtext.facet#didMention", did: node.getAttribute("did"), }, ], }; return { facets: [facet], byteLength: unicodestring.length }; } if (node.nodeName === "atMention") { const text = node.getAttribute("text") || ""; const unicodestring = new UnicodeString(text); const facet: PubLeafletRichtextFacet.Main = { index: { byteStart: byteOffset, byteEnd: byteOffset + unicodestring.length, }, features: [ { $type: "pub.leaflet.richtext.facet#atMention", atURI: node.getAttribute("atURI"), }, ], }; return { facets: [facet], byteLength: unicodestring.length }; } if (node.nodeName === "hard_break") { const unicodestring = new UnicodeString("\n"); return { facets: [], byteLength: unicodestring.length }; } // For other elements (like paragraph), process children let allFacets: PubLeafletRichtextFacet.Main[] = []; let currentOffset = byteOffset; for (const child of node.toArray()) { const result = YJSFragmentToFacets(child, currentOffset); allFacets.push(...result.facets); currentOffset += result.byteLength; } return { facets: allFacets, byteLength: currentOffset - byteOffset }; } if (node.constructor === Y.XmlText) { let facets: PubLeafletRichtextFacet.Main[] = []; let delta = node.toDelta() as Delta[]; let byteStart = byteOffset; let totalLength = 0; for (let d of delta) { let unicodestring = new UnicodeString(d.insert); let facet: PubLeafletRichtextFacet.Main = { index: { byteStart, byteEnd: byteStart + unicodestring.length, }, features: [], }; if (d.attributes?.strikethrough) facet.features.push({ $type: "pub.leaflet.richtext.facet#strikethrough", }); if (d.attributes?.code) facet.features.push({ $type: "pub.leaflet.richtext.facet#code" }); if (d.attributes?.highlight) facet.features.push({ $type: "pub.leaflet.richtext.facet#highlight" }); if (d.attributes?.underline) facet.features.push({ $type: "pub.leaflet.richtext.facet#underline" }); if (d.attributes?.strong) facet.features.push({ $type: "pub.leaflet.richtext.facet#bold" }); if (d.attributes?.em) facet.features.push({ $type: "pub.leaflet.richtext.facet#italic" }); if (d.attributes?.link) facet.features.push({ $type: "pub.leaflet.richtext.facet#link", uri: d.attributes.link.href, }); if (facet.features.length > 0) facets.push(facet); byteStart += unicodestring.length; totalLength += unicodestring.length; } return { facets, byteLength: totalLength }; } return { facets: [], byteLength: 0 }; } type ExcludeString = T extends string ? string extends T ? never : T /* maybe literal, not the whole `string` */ : T; /* not a string */ async function extractThemeFromFacts( facts: Fact[], root_entity: string, agent: AtpBaseClient, ): Promise { let scan = scanIndexLocal(facts); let pageBackground = scan.eav(root_entity, "theme/page-background")?.[0]?.data .value; let cardBackground = scan.eav(root_entity, "theme/card-background")?.[0]?.data .value; let primary = scan.eav(root_entity, "theme/primary")?.[0]?.data.value; let accentBackground = scan.eav(root_entity, "theme/accent-background")?.[0] ?.data.value; let accentText = scan.eav(root_entity, "theme/accent-text")?.[0]?.data.value; let showPageBackground = !scan.eav( root_entity, "theme/card-border-hidden", )?.[0]?.data.value; let backgroundImage = scan.eav(root_entity, "theme/background-image")?.[0]; let backgroundImageRepeat = scan.eav( root_entity, "theme/background-image-repeat", )?.[0]; let pageWidth = scan.eav(root_entity, "theme/page-width")?.[0]; let theme: PubLeafletPublication.Theme = { showPageBackground: showPageBackground ?? true, }; if (pageWidth) theme.pageWidth = pageWidth.data.value; if (pageBackground) theme.backgroundColor = ColorToRGBA(parseColor(`hsba(${pageBackground})`)); if (cardBackground) theme.pageBackground = ColorToRGBA(parseColor(`hsba(${cardBackground})`)); if (primary) theme.primary = ColorToRGB(parseColor(`hsba(${primary})`)); if (accentBackground) theme.accentBackground = ColorToRGB( parseColor(`hsba(${accentBackground})`), ); if (accentText) theme.accentText = ColorToRGB(parseColor(`hsba(${accentText})`)); // Upload background image if present if (backgroundImage?.data) { let imageData = await fetch(backgroundImage.data.src); if (imageData.status === 200) { let binary = await imageData.blob(); let blob = await agent.com.atproto.repo.uploadBlob(binary, { headers: { "Content-Type": binary.type }, }); theme.backgroundImage = { $type: "pub.leaflet.theme.backgroundImage", image: blob.data.blob, repeat: backgroundImageRepeat?.data.value ? true : false, ...(backgroundImageRepeat?.data.value && { width: Math.floor(backgroundImageRepeat.data.value), }), }; } } // Only return theme if at least one property is set if (Object.keys(theme).length > 1 || theme.showPageBackground !== true) { return theme; } return undefined; } /** * Extract mentions from a published document and create notifications */ async function createMentionNotifications( documentUri: string, record: PubLeafletDocument.Record, authorDid: string, ) { const mentionedDids = new Set(); const mentionedPublications = new Map(); // Map of DID -> publication URI const mentionedDocuments = new Map(); // Map of DID -> document URI // Extract mentions from all text blocks in all pages for (const page of record.pages) { if (page.$type === "pub.leaflet.pages.linearDocument") { const linearPage = page as PubLeafletPagesLinearDocument.Main; for (const blockWrapper of linearPage.blocks) { const block = blockWrapper.block; if (block.$type === "pub.leaflet.blocks.text") { const textBlock = block as PubLeafletBlocksText.Main; if (textBlock.facets) { for (const facet of textBlock.facets) { for (const feature of facet.features) { // Check for DID mentions if (PubLeafletRichtextFacet.isDidMention(feature)) { if (feature.did !== authorDid) { mentionedDids.add(feature.did); } } // Check for AT URI mentions (publications and documents) if (PubLeafletRichtextFacet.isAtMention(feature)) { const uri = new AtUri(feature.atURI); if (uri.collection === "pub.leaflet.publication") { // Get the publication owner's DID const { data: publication } = await supabaseServerClient .from("publications") .select("identity_did") .eq("uri", feature.atURI) .single(); if (publication && publication.identity_did !== authorDid) { mentionedPublications.set( publication.identity_did, feature.atURI, ); } } else if (uri.collection === "pub.leaflet.document") { // Get the document owner's DID const { data: document } = await supabaseServerClient .from("documents") .select("uri, data") .eq("uri", feature.atURI) .single(); if (document) { const docRecord = document.data as PubLeafletDocument.Record; if (docRecord.author !== authorDid) { mentionedDocuments.set(docRecord.author, feature.atURI); } } } } } } } } } } } // Create notifications for DID mentions for (const did of mentionedDids) { const notification: Notification = { id: v7(), recipient: did, data: { type: "mention", document_uri: documentUri, mention_type: "did", }, }; await supabaseServerClient.from("notifications").insert(notification); await pingIdentityToUpdateNotification(did); } // Create notifications for publication mentions for (const [recipientDid, publicationUri] of mentionedPublications) { const notification: Notification = { id: v7(), recipient: recipientDid, data: { type: "mention", document_uri: documentUri, mention_type: "publication", mentioned_uri: publicationUri, }, }; await supabaseServerClient.from("notifications").insert(notification); await pingIdentityToUpdateNotification(recipientDid); } // Create notifications for document mentions for (const [recipientDid, mentionedDocUri] of mentionedDocuments) { const notification: Notification = { id: v7(), recipient: recipientDid, data: { type: "mention", document_uri: documentUri, mention_type: "document", mentioned_uri: mentionedDocUri, }, }; await supabaseServerClient.from("notifications").insert(notification); await pingIdentityToUpdateNotification(recipientDid); } }