blowater/features/profile.ts
BlowaterNostr bd96687205
Optimize Parse Event Content (#83)
Co-authored-by: Foodstr <foodstr@proton.me>
2023-07-16 23:04:23 +08:00

163 lines
4.5 KiB
TypeScript

import * as csp from "https://raw.githubusercontent.com/BlowaterNostr/csp/master/csp.ts";
import { Database_Contextual_View } from "../database.ts";
import {
ConnectionPool,
newSubID,
} from "https://raw.githubusercontent.com/BlowaterNostr/nostr.ts/main/relay.ts";
import { PublicKey } from "https://raw.githubusercontent.com/BlowaterNostr/nostr.ts/main/key.ts";
import {
groupBy,
NostrAccountContext,
NostrEvent,
NostrKind,
prepareNormalNostrEvent,
} from "https://raw.githubusercontent.com/BlowaterNostr/nostr.ts/main/nostr.ts";
import { Parsed_Event, Profile_Nostr_Event } from "../nostr.ts";
// nip01 meta data
// https://github.com/nostr-protocol/nips/blob/master/05.md
// get the profile from a public key
export class UserMetaDataNotExist {
constructor(
public readonly publicKey: string,
public readonly relay: string,
) {}
}
export function profilesStream(
publicKeys: Iterable<string>,
pool: ConnectionPool,
) {
const chan = csp.chan<[NostrEvent, string]>();
(async () => {
let subId = newSubID();
let resp = await pool.newSub(
subId,
{
authors: Array.from(publicKeys),
kinds: [NostrKind.META_DATA],
},
);
if (resp instanceof Error) {
throw resp;
}
for await (let { res: nostrMessage, url: relayUrl } of resp) {
if (nostrMessage.type === "EVENT" && nostrMessage.event.content) {
await chan.put([
nostrMessage.event,
relayUrl,
]);
}
}
await chan.close(`pool sub has been clsoed`);
})();
return chan;
}
export async function saveProfile(
profile: ProfileData,
sender: NostrAccountContext,
pool: ConnectionPool,
) {
const event = await prepareNormalNostrEvent(
sender,
NostrKind.META_DATA,
[],
JSON.stringify(profile),
);
pool.sendEvent(event);
}
export function getProfileEvent(
db: Database_Contextual_View,
pubkey: PublicKey,
): Profile_Nostr_Event | undefined {
const events: Profile_Nostr_Event[] = [];
for (const e of db.events) {
if (e.kind === NostrKind.META_DATA && e.pubkey === pubkey.hex) {
events.push(e);
}
}
if (events.length == 0) {
return undefined;
}
events.sort((e1, e2) => e2.created_at - e1.created_at);
const newest = events[0];
return newest;
}
export function getProfilesByName(db: Database_Contextual_View, name: string): Profile_Nostr_Event[] {
const events: Profile_Nostr_Event[] = [];
for (const e of db.events) {
if (e.kind === NostrKind.META_DATA) {
events.push(e);
}
}
if (events.length == 0) {
return [];
}
const profilesPerUser = groupBy(events, (e) => e.pubkey);
const result = [];
for (const events of profilesPerUser.values()) {
events.sort((e1, e2) => e2.created_at - e1.created_at);
const p = events[0];
if (p.profile.name && p.profile.name?.toLocaleLowerCase().indexOf(name.toLowerCase()) != -1) {
result.push(p);
}
}
return result;
}
export function getProfiles(
db: Database_Contextual_View,
pubkeys: Set<string>,
): Map<string, /*pubkey*/ Profile_Nostr_Event | undefined> {
const contacts: Map<string, Profile_Nostr_Event | undefined> = new Map();
for (const key of pubkeys) {
const event = getProfileEvent(db, PublicKey.FromHex(key) as PublicKey);
contacts.set(key, event);
}
return contacts;
}
// aka user profile
export interface ProfileData {
name?: string;
picture?: string;
about?: string;
website?: string;
banner?: string;
[key: string]: string | undefined;
}
export function ProfileFromNostrEvent(
event: Parsed_Event<NostrKind.META_DATA>,
) {
const profileData = parseProfileData(event.content);
if (profileData instanceof Error) {
return profileData;
}
const e: Profile_Nostr_Event = {
kind: event.kind,
id: event.id,
sig: event.sig,
created_at: event.created_at,
pubkey: event.pubkey,
tags: event.tags,
content: event.content,
parsedTags: event.parsedTags,
profile: profileData,
publicKey: event.publicKey,
};
return e;
}
export function parseProfileData(content: string) {
try {
return JSON.parse(content) as ProfileData;
} catch (e) {
return e as Error;
}
}