/* Extension to common Nostr types */ import { PrivateKey, PublicKey } from "./lib/nostr-ts/key.ts"; import * as nostr from "./lib/nostr-ts/nostr.ts"; import { groupBy, NostrKind, TagPubKey } from "./lib/nostr-ts/nostr.ts"; import { ProfileData } from "./features/profile.ts"; import { ContentItem } from "./UI/message.ts"; import { prepareEncryptedNostrEvent, prepareNormalNostrEvent } from "./lib/nostr-ts/event.ts"; type TotolChunks = string; type ChunkIndex = string; // 0-indexed type GroupLeadEventID = string; export type TagImage = ["image", GroupLeadEventID, TotolChunks, ChunkIndex]; export type TagClient = ["client", "blowater"]; export type TagLamportTimestamp = ["lamport", string]; export type TagReply = ["e", nostr.EventID, RelayURL, Marker]; type Marker = "reply" | "root" | "mention"; type RelayURL = string; export type Tag = nostr.Tag | TagImage | TagClient | TagLamportTimestamp | TagReply; type Tags = { image?: [GroupLeadEventID, TotolChunks, ChunkIndex]; lamport_timestamp?: number; reply?: [nostr.EventID, RelayURL, "reply"]; root?: [nostr.EventID, RelayURL, "root"]; } & nostr.Tags; type Event = nostr.NostrEvent; export type parsedTagsEvent = nostr.NostrEvent & { readonly parsedTags: Tags; }; export type Parsed_Event = parsedTagsEvent & { readonly publicKey: PublicKey; }; // content is either JSON, encrypted or other format that's should not be rendered directly export type Encrypted_Kind = NostrKind.CustomAppData | NostrKind.DIRECT_MESSAGE; export type Non_Plain_Text_Kind = Encrypted_Kind | NostrKind.META_DATA; export type Text_Note_Event = Parsed_Event & { parsedContentItems: ContentItem[]; }; export type Profile_Nostr_Event = Parsed_Event & { profile: ProfileData; }; export type CustomAppData_Event = Parsed_Event & { readonly customAppData: CustomAppData; }; export type DirectedMessage_Event = Parsed_Event & { decryptedContent: string; parsedContentItems: ContentItem[]; }; export type Encrypted_Event = DirectedMessage_Event | CustomAppData_Event; export type CustomAppData = PinContact | UnpinContact | UserLogin; export type PinContact = { type: "PinContact"; pubkey: string; }; export type UnpinContact = { type: "UnpinContact"; pubkey: string; }; export type UserLogin = { type: "UserLogin"; }; export function getTags(event: Event): Tags { const tags: Tags = { p: [], e: [], }; for (const tag of event.tags) { switch (tag[0]) { case "p": tags.p.push(tag[1]); break; case "e": if (tag[3] == "reply") { const [_1, EventID, RelayURL, _2] = tag; tags.reply = [EventID, RelayURL as string, "reply"]; } else if (tag[3] == "root") { const [_1, EventID, RelayURL, _2] = tag; tags.root = [EventID, RelayURL as string, "root"]; } else if (tag[1] != "") { tags.e.push(tag[1]); } break; case "image": const [_, GroupLeadEventID, TotolChunks, ChunkIndex] = tag; tags.image = [GroupLeadEventID, TotolChunks, ChunkIndex]; break; case "client": tags.client = tag[1]; break; case "lamport": tags.lamport_timestamp = Number(tag[1]); break; } } return tags; } export async function prepareNostrImageEvent( sender: nostr.NostrAccountContext, receiverPublicKey: PublicKey, blob: Blob, kind: nostr.NostrKind, tags?: Tag[], ): Promise<[nostr.NostrEvent, string] | Error> { // prepare nostr event // read the blob const binaryContent = await nostr.blobToBase64(blob); const limit = 64 * 1024; if (binaryContent.length > limit) { return new Error(`content size ${binaryContent.length} > limit ${limit}`); } const encrypted = await sender.encrypt(receiverPublicKey.hex, binaryContent); if (encrypted instanceof Error) { return encrypted; } const GroupLeadEventID = PrivateKey.Generate().hex; const event: nostr.UnsignedNostrEvent = { created_at: Math.floor(Date.now() / 1000), kind: kind, pubkey: sender.publicKey.hex, tags: [ ["p", receiverPublicKey.hex], ["image", GroupLeadEventID, "1", "0"], ...(tags || []), ], content: encrypted, }; const signedEvent = await sender.signEvent(event); return [signedEvent, GroupLeadEventID]; } export function reassembleBase64ImageFromEvents( events: nostr.NostrEvent[], ) { if (events.length === 0) { return ""; } const firstEvent = events[0]; const imageTag = getTags(firstEvent).image; if (imageTag == undefined) { return new Error(`${firstEvent.id} is not an image event`); } const [_2, chunkCount, _4] = imageTag; const chunks = new Array(Number(chunkCount)); chunks.fill(null); for (const event of events) { const imageTag = getTags(event).image; if (imageTag == undefined) { return new Error(`${event.id} is not an image event`); } const [_2, _3, chunkIndex] = imageTag; const cIndex = Number(chunkIndex); chunks[cIndex] = event.content; } if (chunks.includes(null)) { const miss = chunks.filter((c) => c === null).length; return new Error( `not enough chunks for image event ${firstEvent.id}, need ${Number(chunkCount)}, miss ${miss}`, ); } return chunks.join(""); } export function groupImageEvents(events: Iterable) { return groupBy(events, (event) => { const tags = event.parsedTags; const imageTag = tags.image; if (imageTag == undefined) { return undefined; } const groupID = imageTag[0]; return groupID; }); } export async function prepareReplyEvent( sender: nostr.NostrAccountContext, targetEvent: nostr.NostrEvent, tags: Tag[], content: string, ): Promise { const ps = getTags(targetEvent).p; if (targetEvent.kind == NostrKind.DIRECT_MESSAGE) { const replyTo = PublicKey.FromHex(targetEvent.pubkey); if (replyTo instanceof Error) { return replyTo; } return prepareEncryptedNostrEvent( sender, replyTo, targetEvent.kind, [ [ "e", targetEvent.id, "", "reply", ], ...ps.map((p) => [ "p", p, ] as TagPubKey ), ...tags, ], content, ); } return prepareNormalNostrEvent( sender, targetEvent.kind, [ [ "e", targetEvent.id, "", "reply", ], ...ps.map((p) => [ "p", p, ] as TagPubKey ), ...tags, ], content, ); } export function compare(a: parsedTagsEvent, b: parsedTagsEvent) { if (a.parsedTags.lamport_timestamp && b.parsedTags.lamport_timestamp) { return a.parsedTags.lamport_timestamp - b.parsedTags.lamport_timestamp; } return a.created_at - b.created_at; } export function computeThreads(events: T[]) { events.sort(compare); const idsMap = new Map(); for (const event of events) { if (!idsMap.has(event.id)) { idsMap.set(event.id, event); } const tags = event.parsedTags; if (tags.image && tags.image[2] == "0" && !idsMap.has(tags.image[0])) { idsMap.set(tags.image[0], event); } } const relationsMap = new Map(); for (const event of events) { let id = event.id; const replyTags = event.parsedTags.root || event.parsedTags.reply || event.parsedTags.e; const imageTags = event.parsedTags.image; if (replyTags && replyTags.length > 0) { id = replyTags[0]; } else if (imageTags && imageTags.length > 0) { id = imageTags[0]; } const idsEvent = idsMap.get(id); if (idsEvent) { const relationEvent = relationsMap.get(idsEvent); if (!relationEvent) { relationsMap.set(event, idsEvent); } else { relationsMap.set(event, relationEvent); } } else { relationsMap.set(event, id); } } const resMap = new Map(); for (const event of events) { const relationEvent = relationsMap.get(event); if (!relationEvent) { throw Error("Impossible"); } const id = typeof relationEvent == "string" ? relationEvent : relationEvent.id; const res = resMap.get(id); if (res) { res.push(event); } else { resMap.set(id, [event]); } } return Array.from(resMap.values()); }